π File detail
components/InvalidConfigDialog.tsx
π§© .tsxπ 156 linesπΎ 15,252 bytesπ text
β Back to All Filesπ― Use case
This file lives under βcomponents/β, which covers shared React UI pieces. On the API surface it exposes showInvalidConfigDialog β mainly functions, hooks, or classes. Dependencies touch React UI. It composes internal code from ink, keybindings, state, utils, and CustomSelect (relative imports).
Generated from folder role, exports, dependency roots, and inline comments β not hand-reviewed for every path.
π§ Inline summary
import { c as _c } from "react/compiler-runtime"; import React from 'react'; import { Box, render, Text } from '../ink.js'; import { KeybindingSetup } from '../keybindings/KeybindingProviderSetup.js'; import { AppStateProvider } from '../state/AppState.js';
π€ Exports (heuristic)
showInvalidConfigDialog
π External import roots
Package roots from from "β¦" (relative paths omitted).
react
π₯οΈ Source preview
import { c as _c } from "react/compiler-runtime";
import React from 'react';
import { Box, render, Text } from '../ink.js';
import { KeybindingSetup } from '../keybindings/KeybindingProviderSetup.js';
import { AppStateProvider } from '../state/AppState.js';
import type { ConfigParseError } from '../utils/errors.js';
import { getBaseRenderOptions } from '../utils/renderOptions.js';
import { jsonStringify, writeFileSync_DEPRECATED } from '../utils/slowOperations.js';
import type { ThemeName } from '../utils/theme.js';
import { Select } from './CustomSelect/index.js';
import { Dialog } from './design-system/Dialog.js';
interface InvalidConfigHandlerProps {
error: ConfigParseError;
}
interface InvalidConfigDialogProps {
filePath: string;
errorDescription: string;
onExit: () => void;
onReset: () => void;
}
/**
* Dialog shown when the Claude config file contains invalid JSON
*/
function InvalidConfigDialog(t0) {
const $ = _c(19);
const {
filePath,
errorDescription,
onExit,
onReset
} = t0;
let t1;
if ($[0] !== onExit || $[1] !== onReset) {
t1 = value => {
if (value === "exit") {
onExit();
} else {
onReset();
}
};
$[0] = onExit;
$[1] = onReset;
$[2] = t1;
} else {
t1 = $[2];
}
const handleSelect = t1;
let t2;
if ($[3] !== filePath) {
t2 = <Text>The configuration file at <Text bold={true}>{filePath}</Text> contains invalid JSON.</Text>;
$[3] = filePath;
$[4] = t2;
} else {
t2 = $[4];
}
let t3;
if ($[5] !== errorDescription) {
t3 = <Text>{errorDescription}</Text>;
$[5] = errorDescription;
$[6] = t3;
} else {
t3 = $[6];
}
let t4;
if ($[7] !== t2 || $[8] !== t3) {
t4 = <Box flexDirection="column" gap={1}>{t2}{t3}</Box>;
$[7] = t2;
$[8] = t3;
$[9] = t4;
} else {
t4 = $[9];
}
let t5;
if ($[10] === Symbol.for("react.memo_cache_sentinel")) {
t5 = <Text bold={true}>Choose an option:</Text>;
$[10] = t5;
} else {
t5 = $[10];
}
let t6;
if ($[11] === Symbol.for("react.memo_cache_sentinel")) {
t6 = [{
label: "Exit and fix manually",
value: "exit"
}, {
label: "Reset with default configuration",
value: "reset"
}];
$[11] = t6;
} else {
t6 = $[11];
}
let t7;
if ($[12] !== handleSelect || $[13] !== onExit) {
t7 = <Box flexDirection="column">{t5}<Select options={t6} onChange={handleSelect} onCancel={onExit} /></Box>;
$[12] = handleSelect;
$[13] = onExit;
$[14] = t7;
} else {
t7 = $[14];
}
let t8;
if ($[15] !== onExit || $[16] !== t4 || $[17] !== t7) {
t8 = <Dialog title="Configuration Error" color="error" onCancel={onExit}>{t4}{t7}</Dialog>;
$[15] = onExit;
$[16] = t4;
$[17] = t7;
$[18] = t8;
} else {
t8 = $[18];
}
return t8;
}
/**
* Safe fallback theme name for error dialogs to avoid circular dependency.
* Uses a hardcoded dark theme that doesn't require reading from config.
*/
const SAFE_ERROR_THEME_NAME: ThemeName = 'dark';
export async function showInvalidConfigDialog({
error
}: InvalidConfigHandlerProps): Promise<void> {
// Extend RenderOptions with theme property for this specific usage
type SafeRenderOptions = Parameters<typeof render>[1] & {
theme?: ThemeName;
};
const renderOptions: SafeRenderOptions = {
...getBaseRenderOptions(false),
// IMPORTANT: Use hardcoded theme name to avoid circular dependency with getGlobalConfig()
// This allows the error dialog to show even when config file has JSON syntax errors
theme: SAFE_ERROR_THEME_NAME
};
await new Promise<void>(async resolve => {
const {
unmount
} = await render(<AppStateProvider>
<KeybindingSetup>
<InvalidConfigDialog filePath={error.filePath} errorDescription={error.message} onExit={() => {
unmount();
void resolve();
process.exit(1);
}} onReset={() => {
writeFileSync_DEPRECATED(error.filePath, jsonStringify(error.defaultConfig, null, 2), {
flush: false,
encoding: 'utf8'
});
unmount();
void resolve();
process.exit(0);
}} />
</KeybindingSetup>
</AppStateProvider>, renderOptions);
});
}
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["React","Box","render","Text","KeybindingSetup","AppStateProvider","ConfigParseError","getBaseRenderOptions","jsonStringify","writeFileSync_DEPRECATED","ThemeName","Select","Dialog","InvalidConfigHandlerProps","error","InvalidConfigDialogProps","filePath","errorDescription","onExit","onReset","InvalidConfigDialog","t0","$","_c","t1","value","handleSelect","t2","t3","t4","t5","Symbol","for","t6","label","t7","t8","SAFE_ERROR_THEME_NAME","showInvalidConfigDialog","Promise","SafeRenderOptions","Parameters","theme","renderOptions","resolve","unmount","message","process","exit","defaultConfig","flush","encoding"],"sources":["InvalidConfigDialog.tsx"],"sourcesContent":["import React from 'react'\nimport { Box, render, Text } from '../ink.js'\nimport { KeybindingSetup } from '../keybindings/KeybindingProviderSetup.js'\nimport { AppStateProvider } from '../state/AppState.js'\nimport type { ConfigParseError } from '../utils/errors.js'\nimport { getBaseRenderOptions } from '../utils/renderOptions.js'\nimport {\n  jsonStringify,\n  writeFileSync_DEPRECATED,\n} from '../utils/slowOperations.js'\nimport type { ThemeName } from '../utils/theme.js'\nimport { Select } from './CustomSelect/index.js'\nimport { Dialog } from './design-system/Dialog.js'\n\ninterface InvalidConfigHandlerProps {\n  error: ConfigParseError\n}\n\ninterface InvalidConfigDialogProps {\n  filePath: string\n  errorDescription: string\n  onExit: () => void\n  onReset: () => void\n}\n\n/**\n * Dialog shown when the Claude config file contains invalid JSON\n */\nfunction InvalidConfigDialog({\n  filePath,\n  errorDescription,\n  onExit,\n  onReset,\n}: InvalidConfigDialogProps): React.ReactNode {\n  // Handler for Select onChange\n  const handleSelect = (value: string) => {\n    if (value === 'exit') {\n      onExit()\n    } else {\n      onReset()\n    }\n  }\n\n  return (\n    <Dialog title=\"Configuration Error\" color=\"error\" onCancel={onExit}>\n      <Box flexDirection=\"column\" gap={1}>\n        <Text>\n          The configuration file at <Text bold>{filePath}</Text> contains\n          invalid JSON.\n        </Text>\n        <Text>{errorDescription}</Text>\n      </Box>\n      <Box flexDirection=\"column\">\n        <Text bold>Choose an option:</Text>\n        <Select\n          options={[\n            { label: 'Exit and fix manually', value: 'exit' },\n            { label: 'Reset with default configuration', value: 'reset' },\n          ]}\n          onChange={handleSelect}\n          onCancel={onExit}\n        />\n      </Box>\n    </Dialog>\n  )\n}\n\n/**\n * Safe fallback theme name for error dialogs to avoid circular dependency.\n * Uses a hardcoded dark theme that doesn't require reading from config.\n */\nconst SAFE_ERROR_THEME_NAME: ThemeName = 'dark'\n\nexport async function showInvalidConfigDialog({\n  error,\n}: InvalidConfigHandlerProps): Promise<void> {\n  // Extend RenderOptions with theme property for this specific usage\n  type SafeRenderOptions = Parameters<typeof render>[1] & { theme?: ThemeName }\n\n  const renderOptions: SafeRenderOptions = {\n    ...getBaseRenderOptions(false),\n    // IMPORTANT: Use hardcoded theme name to avoid circular dependency with getGlobalConfig()\n    // This allows the error dialog to show even when config file has JSON syntax errors\n    theme: SAFE_ERROR_THEME_NAME,\n  }\n\n  await new Promise<void>(async resolve => {\n    const { unmount } = await render(\n      <AppStateProvider>\n        <KeybindingSetup>\n          <InvalidConfigDialog\n            filePath={error.filePath}\n            errorDescription={error.message}\n            onExit={() => {\n              unmount()\n              void resolve()\n              process.exit(1)\n            }}\n            onReset={() => {\n              writeFileSync_DEPRECATED(\n                error.filePath,\n                jsonStringify(error.defaultConfig, null, 2),\n                { flush: false, encoding: 'utf8' },\n              )\n              unmount()\n              void resolve()\n              process.exit(0)\n            }}\n          />\n        </KeybindingSetup>\n      </AppStateProvider>,\n      renderOptions,\n    )\n  })\n}\n"],"mappings":"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","ignoreList":[]}