DesktopHandoff.tsx
components/DesktopHandoff.tsx
193
Lines
19335
Bytes
2
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 193 lines, 10 detected imports, and 2 detected exports.
Important relationships
Detected exports
getDownloadUrlDesktopHandoff
Keywords
desktopondoneelsetextclaudesetstatereactinpututilschecking
Detected imports
react/compiler-runtimereact../commands.js../ink.js../utils/browser.js../utils/desktopDeepLink.js../utils/errors.js../utils/gracefulShutdown.js../utils/sessionStorage.js./design-system/LoadingState.js
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 React, { useEffect, useState } from 'react';
import type { CommandResultDisplay } from '../commands.js';
// eslint-disable-next-line custom-rules/prefer-use-keybindings -- raw input for "any key" dismiss and y/n prompt
import { Box, Text, useInput } from '../ink.js';
import { openBrowser } from '../utils/browser.js';
import { getDesktopInstallStatus, openCurrentSessionInDesktop } from '../utils/desktopDeepLink.js';
import { errorMessage } from '../utils/errors.js';
import { gracefulShutdown } from '../utils/gracefulShutdown.js';
import { flushSessionStorage } from '../utils/sessionStorage.js';
import { LoadingState } from './design-system/LoadingState.js';
const DESKTOP_DOCS_URL = 'https://clau.de/desktop';
export function getDownloadUrl(): string {
switch (process.platform) {
case 'win32':
return 'https://claude.ai/api/desktop/win32/x64/exe/latest/redirect';
default:
return 'https://claude.ai/api/desktop/darwin/universal/dmg/latest/redirect';
}
}
type DesktopHandoffState = 'checking' | 'prompt-download' | 'flushing' | 'opening' | 'success' | 'error';
type Props = {
onDone: (result?: string, options?: {
display?: CommandResultDisplay;
}) => void;
};
export function DesktopHandoff(t0) {
const $ = _c(20);
const {
onDone
} = t0;
const [state, setState] = useState("checking");
const [error, setError] = useState(null);
const [downloadMessage, setDownloadMessage] = useState("");
let t1;
if ($[0] !== error || $[1] !== onDone || $[2] !== state) {
t1 = input => {
if (state === "error") {
onDone(error ?? "Unknown error", {
display: "system"
});
return;
}
if (state === "prompt-download") {
if (input === "y" || input === "Y") {
openBrowser(getDownloadUrl()).catch(_temp);
onDone(`Starting download. Re-run /desktop once you\u2019ve installed the app.\nLearn more at ${DESKTOP_DOCS_URL}`, {
display: "system"
});
} else {
if (input === "n" || input === "N") {
onDone(`The desktop app is required for /desktop. Learn more at ${DESKTOP_DOCS_URL}`, {
display: "system"
});
}
}
}
};
$[0] = error;
$[1] = onDone;
$[2] = state;
$[3] = t1;
} else {
t1 = $[3];
}
useInput(t1);
let t2;
let t3;
if ($[4] !== onDone) {
t2 = () => {
const performHandoff = async function performHandoff() {
setState("checking");
const installStatus = await getDesktopInstallStatus();
if (installStatus.status === "not-installed") {
setDownloadMessage("Claude Desktop is not installed.");
setState("prompt-download");
return;
}
if (installStatus.status === "version-too-old") {
setDownloadMessage(`Claude Desktop needs to be updated (found v${installStatus.version}, need v1.1.2396+).`);
setState("prompt-download");
return;
}
setState("flushing");
await flushSessionStorage();
setState("opening");
const result = await openCurrentSessionInDesktop();
if (!result.success) {
setError(result.error ?? "Failed to open Claude Desktop");
setState("error");
return;
}
setState("success");
setTimeout(_temp2, 500, onDone);
};
performHandoff().catch(err => {
setError(errorMessage(err));
setState("error");
});
};
t3 = [onDone];
$[4] = onDone;
$[5] = t2;
$[6] = t3;
} else {
t2 = $[5];
t3 = $[6];
}
useEffect(t2, t3);
if (state === "error") {
let t4;
if ($[7] !== error) {
t4 = <Text color="error">Error: {error}</Text>;
$[7] = error;
$[8] = t4;
} else {
t4 = $[8];
}
let t5;
if ($[9] === Symbol.for("react.memo_cache_sentinel")) {
t5 = <Text dimColor={true}>Press any key to continue…</Text>;
$[9] = t5;
} else {
t5 = $[9];
}
let t6;
if ($[10] !== t4) {
t6 = <Box flexDirection="column" paddingX={2}>{t4}{t5}</Box>;
$[10] = t4;
$[11] = t6;
} else {
t6 = $[11];
}
return t6;
}
if (state === "prompt-download") {
let t4;
if ($[12] !== downloadMessage) {
t4 = <Text>{downloadMessage}</Text>;
$[12] = downloadMessage;
$[13] = t4;
} else {
t4 = $[13];
}
let t5;
if ($[14] === Symbol.for("react.memo_cache_sentinel")) {
t5 = <Text>Download now? (y/n)</Text>;
$[14] = t5;
} else {
t5 = $[14];
}
let t6;
if ($[15] !== t4) {
t6 = <Box flexDirection="column" paddingX={2}>{t4}{t5}</Box>;
$[15] = t4;
$[16] = t6;
} else {
t6 = $[16];
}
return t6;
}
let t4;
if ($[17] === Symbol.for("react.memo_cache_sentinel")) {
t4 = {
checking: "Checking for Claude Desktop\u2026",
flushing: "Saving session\u2026",
opening: "Opening Claude Desktop\u2026",
success: "Opening in Claude Desktop\u2026"
};
$[17] = t4;
} else {
t4 = $[17];
}
const messages = t4;
const t5 = messages[state];
let t6;
if ($[18] !== t5) {
t6 = <LoadingState message={t5} />;
$[18] = t5;
$[19] = t6;
} else {
t6 = $[19];
}
return t6;
}
async function _temp2(onDone_0) {
onDone_0("Session transferred to Claude Desktop", {
display: "system"
});
await gracefulShutdown(0, "other");
}
function _temp() {}
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["React","useEffect","useState","CommandResultDisplay","Box","Text","useInput","openBrowser","getDesktopInstallStatus","openCurrentSessionInDesktop","errorMessage","gracefulShutdown","flushSessionStorage","LoadingState","DESKTOP_DOCS_URL","getDownloadUrl","process","platform","DesktopHandoffState","Props","onDone","result","options","display","DesktopHandoff","t0","$","_c","state","setState","error","setError","downloadMessage","setDownloadMessage","t1","input","catch","_temp","t2","t3","performHandoff","installStatus","status","version","success","setTimeout","_temp2","err","t4","t5","Symbol","for","t6","checking","flushing","opening","messages","onDone_0"],"sources":["DesktopHandoff.tsx"],"sourcesContent":["import React, { useEffect, useState } from 'react'\nimport type { CommandResultDisplay } from '../commands.js'\n// eslint-disable-next-line custom-rules/prefer-use-keybindings -- raw input for \"any key\" dismiss and y/n prompt\nimport { Box, Text, useInput } from '../ink.js'\nimport { openBrowser } from '../utils/browser.js'\nimport {\n  getDesktopInstallStatus,\n  openCurrentSessionInDesktop,\n} from '../utils/desktopDeepLink.js'\nimport { errorMessage } from '../utils/errors.js'\nimport { gracefulShutdown } from '../utils/gracefulShutdown.js'\nimport { flushSessionStorage } from '../utils/sessionStorage.js'\nimport { LoadingState } from './design-system/LoadingState.js'\n\nconst DESKTOP_DOCS_URL = 'https://clau.de/desktop'\n\nexport function getDownloadUrl(): string {\n  switch (process.platform) {\n    case 'win32':\n      return 'https://claude.ai/api/desktop/win32/x64/exe/latest/redirect'\n    default:\n      return 'https://claude.ai/api/desktop/darwin/universal/dmg/latest/redirect'\n  }\n}\n\ntype DesktopHandoffState =\n  | 'checking'\n  | 'prompt-download'\n  | 'flushing'\n  | 'opening'\n  | 'success'\n  | 'error'\n\ntype Props = {\n  onDone: (\n    result?: string,\n    options?: { display?: CommandResultDisplay },\n  ) => void\n}\n\nexport function DesktopHandoff({ onDone }: Props): React.ReactNode {\n  const [state, setState] = useState<DesktopHandoffState>('checking')\n  const [error, setError] = useState<string | null>(null)\n  const [downloadMessage, setDownloadMessage] = useState<string>('')\n\n  // Handle keyboard input for error and prompt-download states\n  useInput(input => {\n    if (state === 'error') {\n      onDone(error ?? 'Unknown error', { display: 'system' })\n      return\n    }\n    if (state === 'prompt-download') {\n      if (input === 'y' || input === 'Y') {\n        openBrowser(getDownloadUrl()).catch(() => {})\n        onDone(\n          `Starting download. Re-run /desktop once you\\u2019ve installed the app.\\nLearn more at ${DESKTOP_DOCS_URL}`,\n          { display: 'system' },\n        )\n      } else if (input === 'n' || input === 'N') {\n        onDone(\n          `The desktop app is required for /desktop. Learn more at ${DESKTOP_DOCS_URL}`,\n          { display: 'system' },\n        )\n      }\n    }\n  })\n\n  useEffect(() => {\n    async function performHandoff(): Promise<void> {\n      // Check Desktop install status\n      setState('checking')\n      const installStatus = await getDesktopInstallStatus()\n\n      if (installStatus.status === 'not-installed') {\n        setDownloadMessage('Claude Desktop is not installed.')\n        setState('prompt-download')\n        return\n      }\n\n      if (installStatus.status === 'version-too-old') {\n        setDownloadMessage(\n          `Claude Desktop needs to be updated (found v${installStatus.version}, need v1.1.2396+).`,\n        )\n        setState('prompt-download')\n        return\n      }\n\n      // Flush session storage to ensure transcript is fully written\n      setState('flushing')\n      await flushSessionStorage()\n\n      // Open the deep link (uses claude-dev:// in dev mode)\n      setState('opening')\n      const result = await openCurrentSessionInDesktop()\n\n      if (!result.success) {\n        setError(result.error ?? 'Failed to open Claude Desktop')\n        setState('error')\n        return\n      }\n\n      // Success - exit the CLI\n      setState('success')\n\n      // Give the user a moment to see the success message\n      setTimeout(\n        async (onDone: Props['onDone']) => {\n          onDone('Session transferred to Claude Desktop', { display: 'system' })\n          await gracefulShutdown(0, 'other')\n        },\n        500,\n        onDone,\n      )\n    }\n\n    performHandoff().catch(err => {\n      setError(errorMessage(err))\n      setState('error')\n    })\n  }, [onDone])\n\n  if (state === 'error') {\n    return (\n      <Box flexDirection=\"column\" paddingX={2}>\n        <Text color=\"error\">Error: {error}</Text>\n        <Text dimColor>Press any key to continue…</Text>\n      </Box>\n    )\n  }\n\n  if (state === 'prompt-download') {\n    return (\n      <Box flexDirection=\"column\" paddingX={2}>\n        <Text>{downloadMessage}</Text>\n        <Text>Download now? (y/n)</Text>\n      </Box>\n    )\n  }\n\n  const messages: Record<\n    Exclude<DesktopHandoffState, 'error' | 'prompt-download'>,\n    string\n  > = {\n    checking: 'Checking for Claude Desktop…',\n    flushing: 'Saving session…',\n    opening: 'Opening Claude Desktop…',\n    success: 'Opening in Claude Desktop…',\n  }\n\n  return <LoadingState message={messages[state]} />\n}\n"],"mappings":";AAAA,OAAOA,KAAK,IAAIC,SAAS,EAAEC,QAAQ,QAAQ,OAAO;AAClD,cAAcC,oBAAoB,QAAQ,gBAAgB;AAC1D;AACA,SAASC,GAAG,EAAEC,IAAI,EAAEC,QAAQ,QAAQ,WAAW;AAC/C,SAASC,WAAW,QAAQ,qBAAqB;AACjD,SACEC,uBAAuB,EACvBC,2BAA2B,QACtB,6BAA6B;AACpC,SAASC,YAAY,QAAQ,oBAAoB;AACjD,SAASC,gBAAgB,QAAQ,8BAA8B;AAC/D,SAASC,mBAAmB,QAAQ,4BAA4B;AAChE,SAASC,YAAY,QAAQ,iCAAiC;AAE9D,MAAMC,gBAAgB,GAAG,yBAAyB;AAElD,OAAO,SAASC,cAAcA,CAAA,CAAE,EAAE,MAAM,CAAC;EACvC,QAAQC,OAAO,CAACC,QAAQ;IACtB,KAAK,OAAO;MACV,OAAO,6DAA6D;IACtE;MACE,OAAO,oEAAoE;EAC/E;AACF;AAEA,KAAKC,mBAAmB,GACpB,UAAU,GACV,iBAAiB,GACjB,UAAU,GACV,SAAS,GACT,SAAS,GACT,OAAO;AAEX,KAAKC,KAAK,GAAG;EACXC,MAAM,EAAE,CACNC,MAAe,CAAR,EAAE,MAAM,EACfC,OAA4C,CAApC,EAAE;IAAEC,OAAO,CAAC,EAAEpB,oBAAoB;EAAC,CAAC,EAC5C,GAAG,IAAI;AACX,CAAC;AAED,OAAO,SAAAqB,eAAAC,EAAA;EAAA,MAAAC,CAAA,GAAAC,EAAA;EAAwB;IAAAP;EAAA,IAAAK,EAAiB;EAC9C,OAAAG,KAAA,EAAAC,QAAA,IAA0B3B,QAAQ,CAAsB,UAAU,CAAC;EACnE,OAAA4B,KAAA,EAAAC,QAAA,IAA0B7B,QAAQ,CAAgB,IAAI,CAAC;EACvD,OAAA8B,eAAA,EAAAC,kBAAA,IAA8C/B,QAAQ,CAAS,EAAE,CAAC;EAAA,IAAAgC,EAAA;EAAA,IAAAR,CAAA,QAAAI,KAAA,IAAAJ,CAAA,QAAAN,MAAA,IAAAM,CAAA,QAAAE,KAAA;IAGzDM,EAAA,GAAAC,KAAA;MACP,IAAIP,KAAK,KAAK,OAAO;QACnBR,MAAM,CAACU,KAAwB,IAAxB,eAAwB,EAAE;UAAAP,OAAA,EAAW;QAAS,CAAC,CAAC;QAAA;MAAA;MAGzD,IAAIK,KAAK,KAAK,iBAAiB;QAC7B,IAAIO,KAAK,KAAK,GAAoB,IAAbA,KAAK,KAAK,GAAG;UAChC5B,WAAW,CAACQ,cAAc,CAAC,CAAC,CAAC,CAAAqB,KAAM,CAACC,KAAQ,CAAC;UAC7CjB,MAAM,CACJ,yFAAyFN,gBAAgB,EAAE,EAC3G;YAAAS,OAAA,EAAW;UAAS,CACtB,CAAC;QAAA;UACI,IAAIY,KAAK,KAAK,GAAoB,IAAbA,KAAK,KAAK,GAAG;YACvCf,MAAM,CACJ,2DAA2DN,gBAAgB,EAAE,EAC7E;cAAAS,OAAA,EAAW;YAAS,CACtB,CAAC;UAAA;QACF;MAAA;IACF,CACF;IAAAG,CAAA,MAAAI,KAAA;IAAAJ,CAAA,MAAAN,MAAA;IAAAM,CAAA,MAAAE,KAAA;IAAAF,CAAA,MAAAQ,EAAA;EAAA;IAAAA,EAAA,GAAAR,CAAA;EAAA;EAnBDpB,QAAQ,CAAC4B,EAmBR,CAAC;EAAA,IAAAI,EAAA;EAAA,IAAAC,EAAA;EAAA,IAAAb,CAAA,QAAAN,MAAA;IAEQkB,EAAA,GAAAA,CAAA;MACR,MAAAE,cAAA,kBAAAA,eAAA;QAEEX,QAAQ,CAAC,UAAU,CAAC;QACpB,MAAAY,aAAA,GAAsB,MAAMjC,uBAAuB,CAAC,CAAC;QAErD,IAAIiC,aAAa,CAAAC,MAAO,KAAK,eAAe;UAC1CT,kBAAkB,CAAC,kCAAkC,CAAC;UACtDJ,QAAQ,CAAC,iBAAiB,CAAC;UAAA;QAAA;QAI7B,IAAIY,aAAa,CAAAC,MAAO,KAAK,iBAAiB;UAC5CT,kBAAkB,CAChB,8CAA8CQ,aAAa,CAAAE,OAAQ,qBACrE,CAAC;UACDd,QAAQ,CAAC,iBAAiB,CAAC;UAAA;QAAA;QAK7BA,QAAQ,CAAC,UAAU,CAAC;QACpB,MAAMjB,mBAAmB,CAAC,CAAC;QAG3BiB,QAAQ,CAAC,SAAS,CAAC;QACnB,MAAAR,MAAA,GAAe,MAAMZ,2BAA2B,CAAC,CAAC;QAElD,IAAI,CAACY,MAAM,CAAAuB,OAAQ;UACjBb,QAAQ,CAACV,MAAM,CAAAS,KAAyC,IAA/C,+BAA+C,CAAC;UACzDD,QAAQ,CAAC,OAAO,CAAC;UAAA;QAAA;QAKnBA,QAAQ,CAAC,SAAS,CAAC;QAGnBgB,UAAU,CACRC,MAGC,EACD,GAAG,EACH1B,MACF,CAAC;MAAA,CACF;MAEDoB,cAAc,CAAC,CAAC,CAAAJ,KAAM,CAACW,GAAA;QACrBhB,QAAQ,CAACrB,YAAY,CAACqC,GAAG,CAAC,CAAC;QAC3BlB,QAAQ,CAAC,OAAO,CAAC;MAAA,CAClB,CAAC;IAAA,CACH;IAAEU,EAAA,IAACnB,MAAM,CAAC;IAAAM,CAAA,MAAAN,MAAA;IAAAM,CAAA,MAAAY,EAAA;IAAAZ,CAAA,MAAAa,EAAA;EAAA;IAAAD,EAAA,GAAAZ,CAAA;IAAAa,EAAA,GAAAb,CAAA;EAAA;EApDXzB,SAAS,CAACqC,EAoDT,EAAEC,EAAQ,CAAC;EAEZ,IAAIX,KAAK,KAAK,OAAO;IAAA,IAAAoB,EAAA;IAAA,IAAAtB,CAAA,QAAAI,KAAA;MAGfkB,EAAA,IAAC,IAAI,CAAO,KAAO,CAAP,OAAO,CAAC,OAAQlB,MAAI,CAAE,EAAjC,IAAI,CAAoC;MAAAJ,CAAA,MAAAI,KAAA;MAAAJ,CAAA,MAAAsB,EAAA;IAAA;MAAAA,EAAA,GAAAtB,CAAA;IAAA;IAAA,IAAAuB,EAAA;IAAA,IAAAvB,CAAA,QAAAwB,MAAA,CAAAC,GAAA;MACzCF,EAAA,IAAC,IAAI,CAAC,QAAQ,CAAR,KAAO,CAAC,CAAC,0BAA0B,EAAxC,IAAI,CAA2C;MAAAvB,CAAA,MAAAuB,EAAA;IAAA;MAAAA,EAAA,GAAAvB,CAAA;IAAA;IAAA,IAAA0B,EAAA;IAAA,IAAA1B,CAAA,SAAAsB,EAAA;MAFlDI,EAAA,IAAC,GAAG,CAAe,aAAQ,CAAR,QAAQ,CAAW,QAAC,CAAD,GAAC,CACrC,CAAAJ,EAAwC,CACxC,CAAAC,EAA+C,CACjD,EAHC,GAAG,CAGE;MAAAvB,CAAA,OAAAsB,EAAA;MAAAtB,CAAA,OAAA0B,EAAA;IAAA;MAAAA,EAAA,GAAA1B,CAAA;IAAA;IAAA,OAHN0B,EAGM;EAAA;EAIV,IAAIxB,KAAK,KAAK,iBAAiB;IAAA,IAAAoB,EAAA;IAAA,IAAAtB,CAAA,SAAAM,eAAA;MAGzBgB,EAAA,IAAC,IAAI,CAAEhB,gBAAc,CAAE,EAAtB,IAAI,CAAyB;MAAAN,CAAA,OAAAM,eAAA;MAAAN,CAAA,OAAAsB,EAAA;IAAA;MAAAA,EAAA,GAAAtB,CAAA;IAAA;IAAA,IAAAuB,EAAA;IAAA,IAAAvB,CAAA,SAAAwB,MAAA,CAAAC,GAAA;MAC9BF,EAAA,IAAC,IAAI,CAAC,mBAAmB,EAAxB,IAAI,CAA2B;MAAAvB,CAAA,OAAAuB,EAAA;IAAA;MAAAA,EAAA,GAAAvB,CAAA;IAAA;IAAA,IAAA0B,EAAA;IAAA,IAAA1B,CAAA,SAAAsB,EAAA;MAFlCI,EAAA,IAAC,GAAG,CAAe,aAAQ,CAAR,QAAQ,CAAW,QAAC,CAAD,GAAC,CACrC,CAAAJ,EAA6B,CAC7B,CAAAC,EAA+B,CACjC,EAHC,GAAG,CAGE;MAAAvB,CAAA,OAAAsB,EAAA;MAAAtB,CAAA,OAAA0B,EAAA;IAAA;MAAAA,EAAA,GAAA1B,CAAA;IAAA;IAAA,OAHN0B,EAGM;EAAA;EAET,IAAAJ,EAAA;EAAA,IAAAtB,CAAA,SAAAwB,MAAA,CAAAC,GAAA;IAKGH,EAAA;MAAAK,QAAA,EACQ,mCAA8B;MAAAC,QAAA,EAC9B,sBAAiB;MAAAC,OAAA,EAClB,8BAAyB;MAAAX,OAAA,EACzB;IACX,CAAC;IAAAlB,CAAA,OAAAsB,EAAA;EAAA;IAAAA,EAAA,GAAAtB,CAAA;EAAA;EARD,MAAA8B,QAAA,GAGIR,EAKH;EAE6B,MAAAC,EAAA,GAAAO,QAAQ,CAAC5B,KAAK,CAAC;EAAA,IAAAwB,EAAA;EAAA,IAAA1B,CAAA,SAAAuB,EAAA;IAAtCG,EAAA,IAAC,YAAY,CAAU,OAAe,CAAf,CAAAH,EAAc,CAAC,GAAI;IAAAvB,CAAA,OAAAuB,EAAA;IAAAvB,CAAA,OAAA0B,EAAA;EAAA;IAAAA,EAAA,GAAA1B,CAAA;EAAA;EAAA,OAA1C0B,EAA0C;AAAA;AA7G5C,eAAAN,OAAAW,QAAA;EAmEGrC,QAAM,CAAC,uCAAuC,EAAE;IAAAG,OAAA,EAAW;EAAS,CAAC,CAAC;EACtE,MAAMZ,gBAAgB,CAAC,CAAC,EAAE,OAAO,CAAC;AAAA;AApErC,SAAA0B,MAAA","ignoreList":[]}