Filemedium importancesource

DesktopUpsellStartup.tsx

components/DesktopUpsell/DesktopUpsellStartup.tsx

171
Lines
15520
Bytes
3
Exports
10
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 171 lines, 10 detected imports, and 3 detected exports.

Important relationships

  • No related files detected.

Detected exports

  • getDesktopUpsellConfig
  • shouldShowDesktopUpsellStartup
  • DesktopUpsellStartup

Keywords

ondonereactelsehandleselectsymbolmemo_cache_sentinelconfigpermissiondialogdesktopupsellseencountnot-now

Detected imports

  • react/compiler-runtime
  • react
  • react
  • ../../ink.js
  • ../../services/analytics/growthbook.js
  • ../../services/analytics/index.js
  • ../../utils/config.js
  • ../CustomSelect/select.js
  • ../DesktopHandoff.js
  • ../permissions/PermissionDialog.js

Source notes

This page embeds the full file contents. Small or leaf files are still indexed honestly instead of being over-explained.

Open parent directory

Full source

import { c as _c } from "react/compiler-runtime";
import * as React from 'react';
import { useEffect, useState } from 'react';
import { Box, Text } from '../../ink.js';
import { getDynamicConfig_CACHED_MAY_BE_STALE } from '../../services/analytics/growthbook.js';
import { logEvent } from '../../services/analytics/index.js';
import { getGlobalConfig, saveGlobalConfig } from '../../utils/config.js';
import { Select } from '../CustomSelect/select.js';
import { DesktopHandoff } from '../DesktopHandoff.js';
import { PermissionDialog } from '../permissions/PermissionDialog.js';
type DesktopUpsellConfig = {
  enable_shortcut_tip: boolean;
  enable_startup_dialog: boolean;
};
const DESKTOP_UPSELL_DEFAULT: DesktopUpsellConfig = {
  enable_shortcut_tip: false,
  enable_startup_dialog: false
};
export function getDesktopUpsellConfig(): DesktopUpsellConfig {
  return getDynamicConfig_CACHED_MAY_BE_STALE('tengu_desktop_upsell', DESKTOP_UPSELL_DEFAULT);
}
function isSupportedPlatform(): boolean {
  return process.platform === 'darwin' || process.platform === 'win32' && process.arch === 'x64';
}
export function shouldShowDesktopUpsellStartup(): boolean {
  if (!isSupportedPlatform()) return false;
  if (!getDesktopUpsellConfig().enable_startup_dialog) return false;
  const config = getGlobalConfig();
  if (config.desktopUpsellDismissed) return false;
  if ((config.desktopUpsellSeenCount ?? 0) >= 3) return false;
  return true;
}
type DesktopUpsellSelection = 'try' | 'not-now' | 'never';
type Props = {
  onDone: () => void;
};
export function DesktopUpsellStartup(t0) {
  const $ = _c(14);
  const {
    onDone
  } = t0;
  const [showHandoff, setShowHandoff] = useState(false);
  let t1;
  if ($[0] === Symbol.for("react.memo_cache_sentinel")) {
    t1 = [];
    $[0] = t1;
  } else {
    t1 = $[0];
  }
  useEffect(_temp, t1);
  if (showHandoff) {
    let t2;
    if ($[1] !== onDone) {
      t2 = <DesktopHandoff onDone={() => onDone()} />;
      $[1] = onDone;
      $[2] = t2;
    } else {
      t2 = $[2];
    }
    return t2;
  }
  let t2;
  if ($[3] !== onDone) {
    t2 = function handleSelect(value) {
      switch (value) {
        case "try":
          {
            setShowHandoff(true);
            return;
          }
        case "never":
          {
            saveGlobalConfig(_temp2);
            onDone();
            return;
          }
        case "not-now":
          {
            onDone();
            return;
          }
      }
    };
    $[3] = onDone;
    $[4] = t2;
  } else {
    t2 = $[4];
  }
  const handleSelect = t2;
  let t3;
  if ($[5] === Symbol.for("react.memo_cache_sentinel")) {
    t3 = {
      label: "Open in Claude Code Desktop",
      value: "try" as const
    };
    $[5] = t3;
  } else {
    t3 = $[5];
  }
  let t4;
  if ($[6] === Symbol.for("react.memo_cache_sentinel")) {
    t4 = {
      label: "Not now",
      value: "not-now" as const
    };
    $[6] = t4;
  } else {
    t4 = $[6];
  }
  let t5;
  if ($[7] === Symbol.for("react.memo_cache_sentinel")) {
    t5 = [t3, t4, {
      label: "Don't ask again",
      value: "never" as const
    }];
    $[7] = t5;
  } else {
    t5 = $[7];
  }
  const options = t5;
  let t6;
  if ($[8] === Symbol.for("react.memo_cache_sentinel")) {
    t6 = <Box marginBottom={1}><Text>Same Claude Code with visual diffs, live app preview, parallel sessions, and more.</Text></Box>;
    $[8] = t6;
  } else {
    t6 = $[8];
  }
  let t7;
  if ($[9] !== handleSelect) {
    t7 = () => handleSelect("not-now");
    $[9] = handleSelect;
    $[10] = t7;
  } else {
    t7 = $[10];
  }
  let t8;
  if ($[11] !== handleSelect || $[12] !== t7) {
    t8 = <PermissionDialog title="Try Claude Code Desktop"><Box flexDirection="column" paddingX={2} paddingY={1}>{t6}<Select options={options} onChange={handleSelect} onCancel={t7} /></Box></PermissionDialog>;
    $[11] = handleSelect;
    $[12] = t7;
    $[13] = t8;
  } else {
    t8 = $[13];
  }
  return t8;
}
function _temp2(prev_0) {
  if (prev_0.desktopUpsellDismissed) {
    return prev_0;
  }
  return {
    ...prev_0,
    desktopUpsellDismissed: true
  };
}
function _temp() {
  const newCount = (getGlobalConfig().desktopUpsellSeenCount ?? 0) + 1;
  saveGlobalConfig(prev => {
    if ((prev.desktopUpsellSeenCount ?? 0) >= newCount) {
      return prev;
    }
    return {
      ...prev,
      desktopUpsellSeenCount: newCount
    };
  });
  logEvent("tengu_desktop_upsell_shown", {
    seen_count: newCount
  });
}
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["React","useEffect","useState","Box","Text","getDynamicConfig_CACHED_MAY_BE_STALE","logEvent","getGlobalConfig","saveGlobalConfig","Select","DesktopHandoff","PermissionDialog","DesktopUpsellConfig","enable_shortcut_tip","enable_startup_dialog","DESKTOP_UPSELL_DEFAULT","getDesktopUpsellConfig","isSupportedPlatform","process","platform","arch","shouldShowDesktopUpsellStartup","config","desktopUpsellDismissed","desktopUpsellSeenCount","DesktopUpsellSelection","Props","onDone","DesktopUpsellStartup","t0","$","_c","showHandoff","setShowHandoff","t1","Symbol","for","_temp","t2","handleSelect","value","_temp2","t3","label","const","t4","t5","options","t6","t7","t8","prev_0","prev","newCount","seen_count"],"sources":["DesktopUpsellStartup.tsx"],"sourcesContent":["import * as React from 'react'\nimport { useEffect, useState } from 'react'\nimport { Box, Text } from '../../ink.js'\nimport { getDynamicConfig_CACHED_MAY_BE_STALE } from '../../services/analytics/growthbook.js'\nimport { logEvent } from '../../services/analytics/index.js'\nimport { getGlobalConfig, saveGlobalConfig } from '../../utils/config.js'\nimport { Select } from '../CustomSelect/select.js'\nimport { DesktopHandoff } from '../DesktopHandoff.js'\nimport { PermissionDialog } from '../permissions/PermissionDialog.js'\n\ntype DesktopUpsellConfig = {\n  enable_shortcut_tip: boolean\n  enable_startup_dialog: boolean\n}\n\nconst DESKTOP_UPSELL_DEFAULT: DesktopUpsellConfig = {\n  enable_shortcut_tip: false,\n  enable_startup_dialog: false,\n}\n\nexport function getDesktopUpsellConfig(): DesktopUpsellConfig {\n  return getDynamicConfig_CACHED_MAY_BE_STALE(\n    'tengu_desktop_upsell',\n    DESKTOP_UPSELL_DEFAULT,\n  )\n}\n\nfunction isSupportedPlatform(): boolean {\n  return (\n    process.platform === 'darwin' ||\n    (process.platform === 'win32' && process.arch === 'x64')\n  )\n}\n\nexport function shouldShowDesktopUpsellStartup(): boolean {\n  if (!isSupportedPlatform()) return false\n  if (!getDesktopUpsellConfig().enable_startup_dialog) return false\n  const config = getGlobalConfig()\n  if (config.desktopUpsellDismissed) return false\n  if ((config.desktopUpsellSeenCount ?? 0) >= 3) return false\n  return true\n}\n\ntype DesktopUpsellSelection = 'try' | 'not-now' | 'never'\n\ntype Props = {\n  onDone: () => void\n}\n\nexport function DesktopUpsellStartup({ onDone }: Props): React.ReactNode {\n  const [showHandoff, setShowHandoff] = useState(false)\n\n  // Increment seen count on mount (guard in updater for StrictMode safety)\n  useEffect(() => {\n    const newCount = (getGlobalConfig().desktopUpsellSeenCount ?? 0) + 1\n    saveGlobalConfig(prev => {\n      if ((prev.desktopUpsellSeenCount ?? 0) >= newCount) return prev\n      return { ...prev, desktopUpsellSeenCount: newCount }\n    })\n    logEvent('tengu_desktop_upsell_shown', { seen_count: newCount })\n  }, [])\n\n  if (showHandoff) {\n    return <DesktopHandoff onDone={() => onDone()} />\n  }\n\n  function handleSelect(value: DesktopUpsellSelection): void {\n    switch (value) {\n      case 'try':\n        setShowHandoff(true)\n        return\n      case 'never':\n        saveGlobalConfig(prev => {\n          if (prev.desktopUpsellDismissed) return prev\n          return { ...prev, desktopUpsellDismissed: true }\n        })\n        onDone()\n        return\n      case 'not-now':\n        onDone()\n        return\n    }\n  }\n\n  const options = [\n    { label: 'Open in Claude Code Desktop', value: 'try' as const },\n    { label: 'Not now', value: 'not-now' as const },\n    { label: \"Don't ask again\", value: 'never' as const },\n  ]\n\n  return (\n    <PermissionDialog title=\"Try Claude Code Desktop\">\n      <Box flexDirection=\"column\" paddingX={2} paddingY={1}>\n        <Box marginBottom={1}>\n          <Text>\n            Same Claude Code with visual diffs, live app preview, parallel\n            sessions, and more.\n          </Text>\n        </Box>\n        <Select\n          options={options}\n          onChange={handleSelect}\n          onCancel={() => handleSelect('not-now')}\n        />\n      </Box>\n    </PermissionDialog>\n  )\n}\n"],"mappings":";AAAA,OAAO,KAAKA,KAAK,MAAM,OAAO;AAC9B,SAASC,SAAS,EAAEC,QAAQ,QAAQ,OAAO;AAC3C,SAASC,GAAG,EAAEC,IAAI,QAAQ,cAAc;AACxC,SAASC,oCAAoC,QAAQ,wCAAwC;AAC7F,SAASC,QAAQ,QAAQ,mCAAmC;AAC5D,SAASC,eAAe,EAAEC,gBAAgB,QAAQ,uBAAuB;AACzE,SAASC,MAAM,QAAQ,2BAA2B;AAClD,SAASC,cAAc,QAAQ,sBAAsB;AACrD,SAASC,gBAAgB,QAAQ,oCAAoC;AAErE,KAAKC,mBAAmB,GAAG;EACzBC,mBAAmB,EAAE,OAAO;EAC5BC,qBAAqB,EAAE,OAAO;AAChC,CAAC;AAED,MAAMC,sBAAsB,EAAEH,mBAAmB,GAAG;EAClDC,mBAAmB,EAAE,KAAK;EAC1BC,qBAAqB,EAAE;AACzB,CAAC;AAED,OAAO,SAASE,sBAAsBA,CAAA,CAAE,EAAEJ,mBAAmB,CAAC;EAC5D,OAAOP,oCAAoC,CACzC,sBAAsB,EACtBU,sBACF,CAAC;AACH;AAEA,SAASE,mBAAmBA,CAAA,CAAE,EAAE,OAAO,CAAC;EACtC,OACEC,OAAO,CAACC,QAAQ,KAAK,QAAQ,IAC5BD,OAAO,CAACC,QAAQ,KAAK,OAAO,IAAID,OAAO,CAACE,IAAI,KAAK,KAAM;AAE5D;AAEA,OAAO,SAASC,8BAA8BA,CAAA,CAAE,EAAE,OAAO,CAAC;EACxD,IAAI,CAACJ,mBAAmB,CAAC,CAAC,EAAE,OAAO,KAAK;EACxC,IAAI,CAACD,sBAAsB,CAAC,CAAC,CAACF,qBAAqB,EAAE,OAAO,KAAK;EACjE,MAAMQ,MAAM,GAAGf,eAAe,CAAC,CAAC;EAChC,IAAIe,MAAM,CAACC,sBAAsB,EAAE,OAAO,KAAK;EAC/C,IAAI,CAACD,MAAM,CAACE,sBAAsB,IAAI,CAAC,KAAK,CAAC,EAAE,OAAO,KAAK;EAC3D,OAAO,IAAI;AACb;AAEA,KAAKC,sBAAsB,GAAG,KAAK,GAAG,SAAS,GAAG,OAAO;AAEzD,KAAKC,KAAK,GAAG;EACXC,MAAM,EAAE,GAAG,GAAG,IAAI;AACpB,CAAC;AAED,OAAO,SAAAC,qBAAAC,EAAA;EAAA,MAAAC,CAAA,GAAAC,EAAA;EAA8B;IAAAJ;EAAA,IAAAE,EAAiB;EACpD,OAAAG,WAAA,EAAAC,cAAA,IAAsC/B,QAAQ,CAAC,KAAK,CAAC;EAAA,IAAAgC,EAAA;EAAA,IAAAJ,CAAA,QAAAK,MAAA,CAAAC,GAAA;IAUlDF,EAAA,KAAE;IAAAJ,CAAA,MAAAI,EAAA;EAAA;IAAAA,EAAA,GAAAJ,CAAA;EAAA;EAPL7B,SAAS,CAACoC,KAOT,EAAEH,EAAE,CAAC;EAEN,IAAIF,WAAW;IAAA,IAAAM,EAAA;IAAA,IAAAR,CAAA,QAAAH,MAAA;MACNW,EAAA,IAAC,cAAc,CAAS,MAAc,CAAd,OAAMX,MAAM,CAAC,EAAC,GAAI;MAAAG,CAAA,MAAAH,MAAA;MAAAG,CAAA,MAAAQ,EAAA;IAAA;MAAAA,EAAA,GAAAR,CAAA;IAAA;IAAA,OAA1CQ,EAA0C;EAAA;EAClD,IAAAA,EAAA;EAAA,IAAAR,CAAA,QAAAH,MAAA;IAEDW,EAAA,YAAAC,aAAAC,KAAA;MACE,QAAQA,KAAK;QAAA,KACN,KAAK;UAAA;YACRP,cAAc,CAAC,IAAI,CAAC;YAAA;UAAA;QAAA,KAEjB,OAAO;UAAA;YACVzB,gBAAgB,CAACiC,MAGhB,CAAC;YACFd,MAAM,CAAC,CAAC;YAAA;UAAA;QAAA,KAEL,SAAS;UAAA;YACZA,MAAM,CAAC,CAAC;YAAA;UAAA;MAEZ;IAAC,CACF;IAAAG,CAAA,MAAAH,MAAA;IAAAG,CAAA,MAAAQ,EAAA;EAAA;IAAAA,EAAA,GAAAR,CAAA;EAAA;EAhBD,MAAAS,YAAA,GAAAD,EAgBC;EAAA,IAAAI,EAAA;EAAA,IAAAZ,CAAA,QAAAK,MAAA,CAAAC,GAAA;IAGCM,EAAA;MAAAC,KAAA,EAAS,6BAA6B;MAAAH,KAAA,EAAS,KAAK,IAAII;IAAM,CAAC;IAAAd,CAAA,MAAAY,EAAA;EAAA;IAAAA,EAAA,GAAAZ,CAAA;EAAA;EAAA,IAAAe,EAAA;EAAA,IAAAf,CAAA,QAAAK,MAAA,CAAAC,GAAA;IAC/DS,EAAA;MAAAF,KAAA,EAAS,SAAS;MAAAH,KAAA,EAAS,SAAS,IAAII;IAAM,CAAC;IAAAd,CAAA,MAAAe,EAAA;EAAA;IAAAA,EAAA,GAAAf,CAAA;EAAA;EAAA,IAAAgB,EAAA;EAAA,IAAAhB,CAAA,QAAAK,MAAA,CAAAC,GAAA;IAFjCU,EAAA,IACdJ,EAA+D,EAC/DG,EAA+C,EAC/C;MAAAF,KAAA,EAAS,iBAAiB;MAAAH,KAAA,EAAS,OAAO,IAAII;IAAM,CAAC,CACtD;IAAAd,CAAA,MAAAgB,EAAA;EAAA;IAAAA,EAAA,GAAAhB,CAAA;EAAA;EAJD,MAAAiB,OAAA,GAAgBD,EAIf;EAAA,IAAAE,EAAA;EAAA,IAAAlB,CAAA,QAAAK,MAAA,CAAAC,GAAA;IAKKY,EAAA,IAAC,GAAG,CAAe,YAAC,CAAD,GAAC,CAClB,CAAC,IAAI,CAAC,kFAGN,EAHC,IAAI,CAIP,EALC,GAAG,CAKE;IAAAlB,CAAA,MAAAkB,EAAA;EAAA;IAAAA,EAAA,GAAAlB,CAAA;EAAA;EAAA,IAAAmB,EAAA;EAAA,IAAAnB,CAAA,QAAAS,YAAA;IAIMU,EAAA,GAAAA,CAAA,KAAMV,YAAY,CAAC,SAAS,CAAC;IAAAT,CAAA,MAAAS,YAAA;IAAAT,CAAA,OAAAmB,EAAA;EAAA;IAAAA,EAAA,GAAAnB,CAAA;EAAA;EAAA,IAAAoB,EAAA;EAAA,IAAApB,CAAA,SAAAS,YAAA,IAAAT,CAAA,SAAAmB,EAAA;IAX7CC,EAAA,IAAC,gBAAgB,CAAO,KAAyB,CAAzB,yBAAyB,CAC/C,CAAC,GAAG,CAAe,aAAQ,CAAR,QAAQ,CAAW,QAAC,CAAD,GAAC,CAAY,QAAC,CAAD,GAAC,CAClD,CAAAF,EAKK,CACL,CAAC,MAAM,CACID,OAAO,CAAPA,QAAM,CAAC,CACNR,QAAY,CAAZA,aAAW,CAAC,CACZ,QAA6B,CAA7B,CAAAU,EAA4B,CAAC,GAE3C,EAZC,GAAG,CAaN,EAdC,gBAAgB,CAcE;IAAAnB,CAAA,OAAAS,YAAA;IAAAT,CAAA,OAAAmB,EAAA;IAAAnB,CAAA,OAAAoB,EAAA;EAAA;IAAAA,EAAA,GAAApB,CAAA;EAAA;EAAA,OAdnBoB,EAcmB;AAAA;AAxDhB,SAAAT,OAAAU,MAAA;EAwBG,IAAIC,MAAI,CAAA7B,sBAAuB;IAAA,OAAS6B,MAAI;EAAA;EAAA,OACrC;IAAA,GAAKA,MAAI;IAAA7B,sBAAA,EAA0B;EAAK,CAAC;AAAA;AAzBnD,SAAAc,MAAA;EAKH,MAAAgB,QAAA,GAAiB,CAAC9C,eAAe,CAAC,CAAC,CAAAiB,sBAA4B,IAA7C,CAA6C,IAAI,CAAC;EACpEhB,gBAAgB,CAAC4C,IAAA;IACf,IAAI,CAACA,IAAI,CAAA5B,sBAA4B,IAAhC,CAAgC,KAAK6B,QAAQ;MAAA,OAASD,IAAI;IAAA;IAAA,OACxD;MAAA,GAAKA,IAAI;MAAA5B,sBAAA,EAA0B6B;IAAS,CAAC;EAAA,CACrD,CAAC;EACF/C,QAAQ,CAAC,4BAA4B,EAAE;IAAAgD,UAAA,EAAcD;EAAS,CAAC,CAAC;AAAA","ignoreList":[]}