OverageCreditUpsell.tsx
components/LogoV2/OverageCreditUpsell.tsx
166
Lines
18347
Bytes
7
Exports
12
Imports
10
Keywords
What this is
This page documents one file from the repository and includes its full source so you can read it without leaving the docs site.
Beginner explanation
This file is one piece of the larger system. Its name, directory, imports, and exports show where it fits. Start by reading the exports and related files first.
How it is used
Start from the exports list and related files. Those are the easiest clues for where this file fits into the system.
Expert explanation
Architecturally, this file intersects with ui-flow. It contains 166 lines, 12 detected imports, and 7 detected exports.
Important relationships
Detected exports
isEligibleForOverageCreditGrantshouldShowOverageCreditUpsellmaybeRefreshOverageCreditCacheuseShowOverageCreditUpsellincrementOverageCreditUpsellSeenCountOverageCreditUpsellcreateOverageCreditFeed
Keywords
textinfoamountmaxwidthtitlereactcopygetcachedoveragecreditgrantconfigusage
Detected imports
react/compiler-runtimereactreact../../ink.js../../services/analytics/index.js../../services/api/overageCreditGrant.js../../utils/config.js../../utils/format.js./Feed.jsOC & Bulk Overages copyOC & Bulk Overages copyOC & Bulk Overages copy
Source notes
This page embeds the full file contents. Small or leaf files are still indexed honestly instead of being over-explained.
Full source
import { c as _c } from "react/compiler-runtime";
import * as React from 'react';
import { useState } from 'react';
import { Text } from '../../ink.js';
import { logEvent } from '../../services/analytics/index.js';
import { formatGrantAmount, getCachedOverageCreditGrant, refreshOverageCreditGrantCache } from '../../services/api/overageCreditGrant.js';
import { getGlobalConfig, saveGlobalConfig } from '../../utils/config.js';
import { truncate } from '../../utils/format.js';
import type { FeedConfig } from './Feed.js';
const MAX_IMPRESSIONS = 3;
/**
* Whether to show the overage credit upsell on any surface.
*
* Eligibility comes entirely from the backend GET /overage_credit_grant
* response — the CLI doesn't replicate tier/threshold/role checks. The
* backend returns available: false for Team members who aren't admins,
* so they don't see an upsell they can't act on.
*
* isEligibleForOverageCreditGrant — just the backend eligibility. Use for
* persistent reference surfaces (/usage) where the info should show
* whenever eligible, no impression cap.
* shouldShowOverageCreditUpsell — adds the 3-impression cap and
* hasVisitedExtraUsage dismiss. Use for promotional surfaces
* (welcome feed, tips).
*/
export function isEligibleForOverageCreditGrant(): boolean {
const info = getCachedOverageCreditGrant();
if (!info || !info.available || info.granted) return false;
return formatGrantAmount(info) !== null;
}
export function shouldShowOverageCreditUpsell(): boolean {
if (!isEligibleForOverageCreditGrant()) return false;
const config = getGlobalConfig();
if (config.hasVisitedExtraUsage) return false;
if ((config.overageCreditUpsellSeenCount ?? 0) >= MAX_IMPRESSIONS) return false;
return true;
}
/**
* Kick off a background fetch if the cache is empty. Safe to call
* unconditionally on mount — it no-ops if cache is fresh.
*/
export function maybeRefreshOverageCreditCache(): void {
if (getCachedOverageCreditGrant() !== null) return;
void refreshOverageCreditGrantCache();
}
export function useShowOverageCreditUpsell() {
const [show] = useState(_temp);
return show;
}
function _temp() {
maybeRefreshOverageCreditCache();
return shouldShowOverageCreditUpsell();
}
export function incrementOverageCreditUpsellSeenCount(): void {
let newCount = 0;
saveGlobalConfig(prev => {
newCount = (prev.overageCreditUpsellSeenCount ?? 0) + 1;
return {
...prev,
overageCreditUpsellSeenCount: newCount
};
});
logEvent('tengu_overage_credit_upsell_shown', {
seen_count: newCount
});
}
// Copy from "OC & Bulk Overages copy" doc (#6 — CLI /usage)
function getUsageText(amount: string): string {
return `${amount} in extra usage for third-party apps · /extra-usage`;
}
// Copy from "OC & Bulk Overages copy" doc (#4 — CLI Welcome screen).
// Char budgets: title ≤19, subtitle ≤48.
const FEED_SUBTITLE = 'On us. Works on third-party apps · /extra-usage';
function getFeedTitle(amount: string): string {
return `${amount} in extra usage`;
}
type Props = {
maxWidth?: number;
twoLine?: boolean;
};
export function OverageCreditUpsell(t0) {
const $ = _c(8);
const {
maxWidth,
twoLine
} = t0;
let t1;
let t2;
if ($[0] !== maxWidth || $[1] !== twoLine) {
t2 = Symbol.for("react.early_return_sentinel");
bb0: {
const info = getCachedOverageCreditGrant();
if (!info) {
t2 = null;
break bb0;
}
const amount = formatGrantAmount(info);
if (!amount) {
t2 = null;
break bb0;
}
if (twoLine) {
const title = getFeedTitle(amount);
let t3;
if ($[4] !== maxWidth) {
t3 = maxWidth ? truncate(FEED_SUBTITLE, maxWidth) : FEED_SUBTITLE;
$[4] = maxWidth;
$[5] = t3;
} else {
t3 = $[5];
}
let t4;
if ($[6] !== t3) {
t4 = <Text dimColor={true}>{t3}</Text>;
$[6] = t3;
$[7] = t4;
} else {
t4 = $[7];
}
t2 = <><Text color="claude">{maxWidth ? truncate(title, maxWidth) : title}</Text>{t4}</>;
break bb0;
}
const text = getUsageText(amount);
const display = maxWidth ? truncate(text, maxWidth) : text;
const highlightLen = Math.min(getFeedTitle(amount).length, display.length);
t1 = <Text dimColor={true}><Text color="claude">{display.slice(0, highlightLen)}</Text>{display.slice(highlightLen)}</Text>;
}
$[0] = maxWidth;
$[1] = twoLine;
$[2] = t1;
$[3] = t2;
} else {
t1 = $[2];
t2 = $[3];
}
if (t2 !== Symbol.for("react.early_return_sentinel")) {
return t2;
}
return t1;
}
/**
* Feed config for the homescreen rotating feed. Mirrors
* createGuestPassesFeed in feedConfigs.tsx.
*
* Copy from "OC & Bulk Overages copy" doc (#4 — CLI Welcome screen).
* Char budgets: title ≤19, subtitle ≤48.
*/
export function createOverageCreditFeed(): FeedConfig {
const info = getCachedOverageCreditGrant();
const amount = info ? formatGrantAmount(info) : null;
const title = amount ? getFeedTitle(amount) : 'extra usage credit';
return {
title,
lines: [],
customContent: {
content: <Text dimColor>{FEED_SUBTITLE}</Text>,
width: Math.max(title.length, FEED_SUBTITLE.length)
}
};
}
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["React","useState","Text","logEvent","formatGrantAmount","getCachedOverageCreditGrant","refreshOverageCreditGrantCache","getGlobalConfig","saveGlobalConfig","truncate","FeedConfig","MAX_IMPRESSIONS","isEligibleForOverageCreditGrant","info","available","granted","shouldShowOverageCreditUpsell","config","hasVisitedExtraUsage","overageCreditUpsellSeenCount","maybeRefreshOverageCreditCache","useShowOverageCreditUpsell","show","_temp","incrementOverageCreditUpsellSeenCount","newCount","prev","seen_count","getUsageText","amount","FEED_SUBTITLE","getFeedTitle","Props","maxWidth","twoLine","OverageCreditUpsell","t0","$","_c","t1","t2","Symbol","for","bb0","title","t3","t4","text","display","highlightLen","Math","min","length","slice","createOverageCreditFeed","lines","customContent","content","width","max"],"sources":["OverageCreditUpsell.tsx"],"sourcesContent":["import * as React from 'react'\nimport { useState } from 'react'\nimport { Text } from '../../ink.js'\nimport { logEvent } from '../../services/analytics/index.js'\nimport {\n  formatGrantAmount,\n  getCachedOverageCreditGrant,\n  refreshOverageCreditGrantCache,\n} from '../../services/api/overageCreditGrant.js'\nimport { getGlobalConfig, saveGlobalConfig } from '../../utils/config.js'\nimport { truncate } from '../../utils/format.js'\nimport type { FeedConfig } from './Feed.js'\n\nconst MAX_IMPRESSIONS = 3\n\n/**\n * Whether to show the overage credit upsell on any surface.\n *\n * Eligibility comes entirely from the backend GET /overage_credit_grant\n * response — the CLI doesn't replicate tier/threshold/role checks. The\n * backend returns available: false for Team members who aren't admins,\n * so they don't see an upsell they can't act on.\n *\n * isEligibleForOverageCreditGrant — just the backend eligibility. Use for\n *   persistent reference surfaces (/usage) where the info should show\n *   whenever eligible, no impression cap.\n * shouldShowOverageCreditUpsell — adds the 3-impression cap and\n *   hasVisitedExtraUsage dismiss. Use for promotional surfaces\n *   (welcome feed, tips).\n */\nexport function isEligibleForOverageCreditGrant(): boolean {\n  const info = getCachedOverageCreditGrant()\n  if (!info || !info.available || info.granted) return false\n  return formatGrantAmount(info) !== null\n}\n\nexport function shouldShowOverageCreditUpsell(): boolean {\n  if (!isEligibleForOverageCreditGrant()) return false\n\n  const config = getGlobalConfig()\n  if (config.hasVisitedExtraUsage) return false\n  if ((config.overageCreditUpsellSeenCount ?? 0) >= MAX_IMPRESSIONS)\n    return false\n\n  return true\n}\n\n/**\n * Kick off a background fetch if the cache is empty. Safe to call\n * unconditionally on mount — it no-ops if cache is fresh.\n */\nexport function maybeRefreshOverageCreditCache(): void {\n  if (getCachedOverageCreditGrant() !== null) return\n  void refreshOverageCreditGrantCache()\n}\n\nexport function useShowOverageCreditUpsell(): boolean {\n  const [show] = useState(() => {\n    maybeRefreshOverageCreditCache()\n    return shouldShowOverageCreditUpsell()\n  })\n  return show\n}\n\nexport function incrementOverageCreditUpsellSeenCount(): void {\n  let newCount = 0\n  saveGlobalConfig(prev => {\n    newCount = (prev.overageCreditUpsellSeenCount ?? 0) + 1\n    return {\n      ...prev,\n      overageCreditUpsellSeenCount: newCount,\n    }\n  })\n  logEvent('tengu_overage_credit_upsell_shown', { seen_count: newCount })\n}\n\n// Copy from \"OC & Bulk Overages copy\" doc (#6 — CLI /usage)\nfunction getUsageText(amount: string): string {\n  return `${amount} in extra usage for third-party apps · /extra-usage`\n}\n\n// Copy from \"OC & Bulk Overages copy\" doc (#4 — CLI Welcome screen).\n// Char budgets: title ≤19, subtitle ≤48.\nconst FEED_SUBTITLE = 'On us. Works on third-party apps · /extra-usage'\n\nfunction getFeedTitle(amount: string): string {\n  return `${amount} in extra usage`\n}\n\ntype Props = { maxWidth?: number; twoLine?: boolean }\n\nexport function OverageCreditUpsell({\n  maxWidth,\n  twoLine,\n}: Props): React.ReactNode {\n  const info = getCachedOverageCreditGrant()\n  if (!info) return null\n  const amount = formatGrantAmount(info)\n  if (!amount) return null\n\n  if (twoLine) {\n    const title = getFeedTitle(amount)\n    return (\n      <>\n        <Text color=\"claude\">\n          {maxWidth ? truncate(title, maxWidth) : title}\n        </Text>\n        <Text dimColor>\n          {maxWidth ? truncate(FEED_SUBTITLE, maxWidth) : FEED_SUBTITLE}\n        </Text>\n      </>\n    )\n  }\n\n  const text = getUsageText(amount)\n  const display = maxWidth ? truncate(text, maxWidth) : text\n  const highlightLen = Math.min(getFeedTitle(amount).length, display.length)\n\n  return (\n    <Text dimColor>\n      <Text color=\"claude\">{display.slice(0, highlightLen)}</Text>\n      {display.slice(highlightLen)}\n    </Text>\n  )\n}\n\n/**\n * Feed config for the homescreen rotating feed. Mirrors\n * createGuestPassesFeed in feedConfigs.tsx.\n *\n * Copy from \"OC & Bulk Overages copy\" doc (#4 — CLI Welcome screen).\n * Char budgets: title ≤19, subtitle ≤48.\n */\nexport function createOverageCreditFeed(): FeedConfig {\n  const info = getCachedOverageCreditGrant()\n  const amount = info ? formatGrantAmount(info) : null\n  const title = amount ? getFeedTitle(amount) : 'extra usage credit'\n  return {\n    title,\n    lines: [],\n    customContent: {\n      content: <Text dimColor>{FEED_SUBTITLE}</Text>,\n      width: Math.max(title.length, FEED_SUBTITLE.length),\n    },\n  }\n}\n"],"mappings":"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","ignoreList":[]}