@google/gemini-cli 0.7.0-nightly.20250918.2722473a → 0.7.0-preview.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/package.json +4 -3
- package/dist/src/commands/extensions/disable.d.ts +1 -2
- package/dist/src/commands/extensions/disable.js +17 -3
- package/dist/src/commands/extensions/disable.js.map +1 -1
- package/dist/src/commands/extensions/enable.d.ts +1 -2
- package/dist/src/commands/extensions/enable.js +17 -4
- package/dist/src/commands/extensions/enable.js.map +1 -1
- package/dist/src/commands/extensions/install.d.ts +1 -0
- package/dist/src/commands/extensions/install.js +10 -2
- package/dist/src/commands/extensions/install.js.map +1 -1
- package/dist/src/commands/extensions/install.test.js +12 -2
- package/dist/src/commands/extensions/install.test.js.map +1 -1
- package/dist/src/commands/extensions/link.js +1 -1
- package/dist/src/commands/extensions/link.js.map +1 -1
- package/dist/src/commands/extensions/new.test.js +2 -2
- package/dist/src/commands/extensions/new.test.js.map +1 -1
- package/dist/src/commands/extensions/uninstall.js +1 -1
- package/dist/src/commands/extensions/uninstall.js.map +1 -1
- package/dist/src/commands/extensions/uninstall.test.js +4 -1
- package/dist/src/commands/extensions/uninstall.test.js.map +1 -1
- package/dist/src/commands/extensions/update.js +38 -17
- package/dist/src/commands/extensions/update.js.map +1 -1
- package/dist/src/commands/mcp/add.js +6 -1
- package/dist/src/commands/mcp/add.js.map +1 -1
- package/dist/src/config/config.d.ts +2 -0
- package/dist/src/config/config.js +85 -65
- package/dist/src/config/config.js.map +1 -1
- package/dist/src/config/extension.d.ts +5 -18
- package/dist/src/config/extension.js +95 -201
- package/dist/src/config/extension.js.map +1 -1
- package/dist/src/config/extensions/extensionEnablement.d.ts +14 -0
- package/dist/src/config/extensions/extensionEnablement.js +81 -33
- package/dist/src/config/extensions/extensionEnablement.js.map +1 -1
- package/dist/src/config/extensions/extensionEnablement.test.js +164 -19
- package/dist/src/config/extensions/extensionEnablement.test.js.map +1 -1
- package/dist/src/config/extensions/github.d.ts +29 -0
- package/dist/src/config/extensions/github.js +301 -0
- package/dist/src/config/extensions/github.js.map +1 -0
- package/dist/src/config/extensions/github.test.d.ts +6 -0
- package/dist/src/config/extensions/github.test.js +271 -0
- package/dist/src/config/extensions/github.test.js.map +1 -0
- package/dist/src/config/extensions/update.d.ts +20 -0
- package/dist/src/config/extensions/update.js +94 -0
- package/dist/src/config/extensions/update.js.map +1 -0
- package/dist/src/config/extensions/update.test.d.ts +6 -0
- package/dist/src/config/extensions/update.test.js +327 -0
- package/dist/src/config/extensions/update.test.js.map +1 -0
- package/dist/src/config/policy-engine.integration.test.d.ts +6 -0
- package/dist/src/config/policy-engine.integration.test.js +270 -0
- package/dist/src/config/policy-engine.integration.test.js.map +1 -0
- package/dist/src/config/policy.d.ts +8 -0
- package/dist/src/config/policy.js +150 -0
- package/dist/src/config/policy.js.map +1 -0
- package/dist/src/config/policy.test.d.ts +6 -0
- package/dist/src/config/policy.test.js +336 -0
- package/dist/src/config/policy.test.js.map +1 -0
- package/dist/src/config/settings.js +19 -4
- package/dist/src/config/settings.js.map +1 -1
- package/dist/src/config/settingsSchema.d.ts +29 -11
- package/dist/src/config/settingsSchema.js +29 -11
- package/dist/src/config/settingsSchema.js.map +1 -1
- package/dist/src/config/settingsSchema.test.js +1 -1
- package/dist/src/config/settingsSchema.test.js.map +1 -1
- package/dist/src/config/trustedFolders.d.ts +10 -1
- package/dist/src/config/trustedFolders.js +40 -14
- package/dist/src/config/trustedFolders.js.map +1 -1
- package/dist/src/config/trustedFolders.test.js +95 -14
- package/dist/src/config/trustedFolders.test.js.map +1 -1
- package/dist/src/gemini.js +102 -127
- package/dist/src/gemini.js.map +1 -1
- package/dist/src/gemini.test.js +74 -5
- package/dist/src/gemini.test.js.map +1 -1
- package/dist/src/generated/git-commit.d.ts +2 -2
- package/dist/src/generated/git-commit.js +2 -2
- package/dist/src/generated/git-commit.js.map +1 -1
- package/dist/src/nonInteractiveCli.d.ts +2 -1
- package/dist/src/nonInteractiveCli.js +29 -16
- package/dist/src/nonInteractiveCli.js.map +1 -1
- package/dist/src/nonInteractiveCliCommands.d.ts +17 -0
- package/dist/src/nonInteractiveCliCommands.js +79 -0
- package/dist/src/nonInteractiveCliCommands.js.map +1 -0
- package/dist/src/services/BuiltinCommandLoader.js +4 -0
- package/dist/src/services/BuiltinCommandLoader.js.map +1 -1
- package/dist/src/services/BuiltinCommandLoader.test.js +50 -1
- package/dist/src/services/BuiltinCommandLoader.test.js.map +1 -1
- package/dist/src/test-utils/createExtension.d.ts +15 -0
- package/dist/src/test-utils/createExtension.js +25 -0
- package/dist/src/test-utils/createExtension.js.map +1 -0
- package/dist/src/test-utils/mockCommandContext.js +2 -0
- package/dist/src/test-utils/mockCommandContext.js.map +1 -1
- package/dist/src/test-utils/render.d.ts +3 -1
- package/dist/src/test-utils/render.js +2 -1
- package/dist/src/test-utils/render.js.map +1 -1
- package/dist/src/ui/App.js +3 -2
- package/dist/src/ui/App.js.map +1 -1
- package/dist/src/ui/AppContainer.js +55 -42
- package/dist/src/ui/AppContainer.js.map +1 -1
- package/dist/src/ui/AppContainer.test.js +130 -0
- package/dist/src/ui/AppContainer.test.js.map +1 -1
- package/dist/src/ui/IdeIntegrationNudge.d.ts +2 -2
- package/dist/src/ui/IdeIntegrationNudge.js +1 -2
- package/dist/src/ui/IdeIntegrationNudge.js.map +1 -1
- package/dist/src/ui/commands/chatCommand.js +14 -3
- package/dist/src/ui/commands/chatCommand.js.map +1 -1
- package/dist/src/ui/commands/clearCommand.js +1 -1
- package/dist/src/ui/commands/clearCommand.js.map +1 -1
- package/dist/src/ui/commands/extensionsCommand.js +22 -6
- package/dist/src/ui/commands/extensionsCommand.js.map +1 -1
- package/dist/src/ui/commands/ideCommand.js +1 -1
- package/dist/src/ui/commands/ideCommand.js.map +1 -1
- package/dist/src/ui/commands/mcpCommand.js +9 -2
- package/dist/src/ui/commands/mcpCommand.js.map +1 -1
- package/dist/src/ui/commands/modelCommand.d.ts +7 -0
- package/dist/src/ui/commands/modelCommand.js +16 -0
- package/dist/src/ui/commands/modelCommand.js.map +1 -0
- package/dist/src/ui/commands/modelCommand.test.d.ts +6 -0
- package/dist/src/ui/commands/modelCommand.test.js +30 -0
- package/dist/src/ui/commands/modelCommand.test.js.map +1 -0
- package/dist/src/ui/commands/permissionsCommand.d.ts +7 -0
- package/dist/src/ui/commands/permissionsCommand.js +16 -0
- package/dist/src/ui/commands/permissionsCommand.js.map +1 -0
- package/dist/src/ui/commands/permissionsCommand.test.d.ts +6 -0
- package/dist/src/ui/commands/permissionsCommand.test.js +30 -0
- package/dist/src/ui/commands/permissionsCommand.test.js.map +1 -0
- package/dist/src/ui/commands/types.d.ts +3 -3
- package/dist/src/ui/commands/types.js.map +1 -1
- package/dist/src/ui/components/AppHeader.js +2 -5
- package/dist/src/ui/components/AppHeader.js.map +1 -1
- package/dist/src/ui/components/Composer.js +4 -10
- package/dist/src/ui/components/Composer.js.map +1 -1
- package/dist/src/ui/components/Composer.test.js +13 -21
- package/dist/src/ui/components/Composer.test.js.map +1 -1
- package/dist/src/ui/components/DialogManager.d.ts +6 -1
- package/dist/src/ui/components/DialogManager.js +10 -1
- package/dist/src/ui/components/DialogManager.js.map +1 -1
- package/dist/src/ui/components/HistoryItemDisplay.d.ts +1 -1
- package/dist/src/ui/components/HistoryItemDisplay.js +1 -1
- package/dist/src/ui/components/HistoryItemDisplay.js.map +1 -1
- package/dist/src/ui/components/InputPrompt.d.ts +1 -1
- package/dist/src/ui/components/InputPrompt.js +17 -8
- package/dist/src/ui/components/InputPrompt.js.map +1 -1
- package/dist/src/ui/components/MainContent.js +1 -1
- package/dist/src/ui/components/MainContent.js.map +1 -1
- package/dist/src/ui/components/ModelDialog.d.ts +11 -0
- package/dist/src/ui/components/ModelDialog.js +53 -0
- package/dist/src/ui/components/ModelDialog.js.map +1 -0
- package/dist/src/ui/components/ModelDialog.test.d.ts +6 -0
- package/dist/src/ui/components/ModelDialog.test.js +153 -0
- package/dist/src/ui/components/ModelDialog.test.js.map +1 -0
- package/dist/src/ui/components/PermissionsModifyTrustDialog.d.ts +13 -0
- package/dist/src/ui/components/PermissionsModifyTrustDialog.js +45 -0
- package/dist/src/ui/components/PermissionsModifyTrustDialog.js.map +1 -0
- package/dist/src/ui/components/PermissionsModifyTrustDialog.test.d.ts +6 -0
- package/dist/src/ui/components/PermissionsModifyTrustDialog.test.js +158 -0
- package/dist/src/ui/components/PermissionsModifyTrustDialog.test.js.map +1 -0
- package/dist/src/ui/components/QueuedMessageDisplay.d.ts +9 -0
- package/dist/src/ui/components/QueuedMessageDisplay.js +20 -0
- package/dist/src/ui/components/QueuedMessageDisplay.js.map +1 -0
- package/dist/src/ui/components/QueuedMessageDisplay.test.d.ts +6 -0
- package/dist/src/ui/components/QueuedMessageDisplay.test.js +56 -0
- package/dist/src/ui/components/QueuedMessageDisplay.test.js.map +1 -0
- package/dist/src/ui/components/SettingsDialog.js +1 -1
- package/dist/src/ui/components/SettingsDialog.js.map +1 -1
- package/dist/src/ui/components/SettingsDialog.test.js +51 -17
- package/dist/src/ui/components/SettingsDialog.test.js.map +1 -1
- package/dist/src/ui/components/messages/ToolGroupMessage.d.ts +1 -1
- package/dist/src/ui/components/messages/ToolGroupMessage.js +11 -9
- package/dist/src/ui/components/messages/ToolGroupMessage.js.map +1 -1
- package/dist/src/ui/components/messages/ToolMessage.d.ts +1 -1
- package/dist/src/ui/components/messages/ToolMessage.js +13 -6
- package/dist/src/ui/components/messages/ToolMessage.js.map +1 -1
- package/dist/src/ui/components/shared/BaseSelectionList.d.ts +43 -0
- package/dist/src/ui/components/shared/BaseSelectionList.js +72 -0
- package/dist/src/ui/components/shared/BaseSelectionList.js.map +1 -0
- package/dist/src/ui/components/shared/BaseSelectionList.test.d.ts +6 -0
- package/dist/src/ui/components/shared/BaseSelectionList.test.js +374 -0
- package/dist/src/ui/components/shared/BaseSelectionList.test.js.map +1 -0
- package/dist/src/ui/components/shared/DescriptiveRadioButtonSelect.d.ts +36 -0
- package/dist/src/ui/components/shared/DescriptiveRadioButtonSelect.js +13 -0
- package/dist/src/ui/components/shared/DescriptiveRadioButtonSelect.js.map +1 -0
- package/dist/src/ui/components/shared/DescriptiveRadioButtonSelect.test.d.ts +6 -0
- package/dist/src/ui/components/shared/DescriptiveRadioButtonSelect.test.js +68 -0
- package/dist/src/ui/components/shared/DescriptiveRadioButtonSelect.test.js.map +1 -0
- package/dist/src/ui/components/shared/RadioButtonSelect.js +9 -104
- package/dist/src/ui/components/shared/RadioButtonSelect.js.map +1 -1
- package/dist/src/ui/components/shared/RadioButtonSelect.test.js +113 -92
- package/dist/src/ui/components/shared/RadioButtonSelect.test.js.map +1 -1
- package/dist/src/ui/components/shared/text-buffer.js +63 -2
- package/dist/src/ui/components/shared/text-buffer.js.map +1 -1
- package/dist/src/ui/constants.d.ts +1 -0
- package/dist/src/ui/constants.js +1 -0
- package/dist/src/ui/constants.js.map +1 -1
- package/dist/src/ui/contexts/KeypressContext.js +3 -0
- package/dist/src/ui/contexts/KeypressContext.js.map +1 -1
- package/dist/src/ui/contexts/SessionContext.d.ts +6 -0
- package/dist/src/ui/contexts/SessionContext.js +107 -5
- package/dist/src/ui/contexts/SessionContext.js.map +1 -1
- package/dist/src/ui/contexts/ShellFocusContext.d.ts +7 -0
- package/dist/src/ui/contexts/ShellFocusContext.js +9 -0
- package/dist/src/ui/contexts/ShellFocusContext.js.map +1 -0
- package/dist/src/ui/contexts/UIActionsContext.d.ts +2 -0
- package/dist/src/ui/contexts/UIActionsContext.js.map +1 -1
- package/dist/src/ui/contexts/UIStateContext.d.ts +7 -4
- package/dist/src/ui/contexts/UIStateContext.js +1 -0
- package/dist/src/ui/contexts/UIStateContext.js.map +1 -1
- package/dist/src/ui/hooks/shellCommandProcessor.js +7 -5
- package/dist/src/ui/hooks/shellCommandProcessor.js.map +1 -1
- package/dist/src/ui/hooks/shellCommandProcessor.test.js +1 -1
- package/dist/src/ui/hooks/shellCommandProcessor.test.js.map +1 -1
- package/dist/src/ui/hooks/slashCommandProcessor.d.ts +4 -1
- package/dist/src/ui/hooks/slashCommandProcessor.js +7 -1
- package/dist/src/ui/hooks/slashCommandProcessor.js.map +1 -1
- package/dist/src/ui/hooks/useExtensionUpdates.d.ts +12 -0
- package/dist/src/ui/hooks/useExtensionUpdates.js +64 -0
- package/dist/src/ui/hooks/useExtensionUpdates.js.map +1 -0
- package/dist/src/ui/hooks/useExtensionUpdates.test.d.ts +6 -0
- package/dist/src/ui/hooks/useExtensionUpdates.test.js +165 -0
- package/dist/src/ui/hooks/useExtensionUpdates.test.js.map +1 -0
- package/dist/src/ui/hooks/useFocus.js +10 -0
- package/dist/src/ui/hooks/useFocus.js.map +1 -1
- package/dist/src/ui/hooks/useFolderTrust.d.ts +1 -1
- package/dist/src/ui/hooks/useFolderTrust.js +2 -7
- package/dist/src/ui/hooks/useFolderTrust.js.map +1 -1
- package/dist/src/ui/hooks/useGeminiStream.js +7 -6
- package/dist/src/ui/hooks/useGeminiStream.js.map +1 -1
- package/dist/src/ui/hooks/useGitBranchName.test.js.map +1 -1
- package/dist/src/ui/hooks/useModelCommand.d.ts +12 -0
- package/dist/src/ui/hooks/useModelCommand.js +21 -0
- package/dist/src/ui/hooks/useModelCommand.js.map +1 -0
- package/dist/src/ui/hooks/useModelCommand.test.d.ts +6 -0
- package/dist/src/ui/hooks/useModelCommand.test.js +35 -0
- package/dist/src/ui/hooks/useModelCommand.test.js.map +1 -0
- package/dist/src/ui/hooks/usePermissionsModifyTrust.d.ts +17 -0
- package/dist/src/ui/hooks/usePermissionsModifyTrust.js +78 -0
- package/dist/src/ui/hooks/usePermissionsModifyTrust.js.map +1 -0
- package/dist/src/ui/hooks/usePermissionsModifyTrust.test.d.ts +6 -0
- package/dist/src/ui/hooks/usePermissionsModifyTrust.test.js +182 -0
- package/dist/src/ui/hooks/usePermissionsModifyTrust.test.js.map +1 -0
- package/dist/src/ui/hooks/useSelectionList.d.ts +33 -0
- package/dist/src/ui/hooks/useSelectionList.js +252 -0
- package/dist/src/ui/hooks/useSelectionList.js.map +1 -0
- package/dist/src/ui/hooks/useSelectionList.test.d.ts +6 -0
- package/dist/src/ui/hooks/useSelectionList.test.js +651 -0
- package/dist/src/ui/hooks/useSelectionList.test.js.map +1 -0
- package/dist/src/ui/hooks/useSlashCompletion.js +7 -2
- package/dist/src/ui/hooks/useSlashCompletion.js.map +1 -1
- package/dist/src/ui/hooks/useSlashCompletion.test.js +33 -0
- package/dist/src/ui/hooks/useSlashCompletion.test.js.map +1 -1
- package/dist/src/ui/hooks/useStateAndRef.d.ts +1 -1
- package/dist/src/ui/hooks/useStateAndRef.js +2 -2
- package/dist/src/ui/hooks/useStateAndRef.js.map +1 -1
- package/dist/src/ui/noninteractive/nonInteractiveUi.d.ts +12 -0
- package/dist/src/ui/noninteractive/nonInteractiveUi.js +27 -0
- package/dist/src/ui/noninteractive/nonInteractiveUi.js.map +1 -0
- package/dist/src/ui/privacy/CloudFreePrivacyNotice.js +3 -3
- package/dist/src/ui/privacy/CloudFreePrivacyNotice.js.map +1 -1
- package/dist/src/ui/state/extensions.d.ts +2 -1
- package/dist/src/ui/state/extensions.js +1 -0
- package/dist/src/ui/state/extensions.js.map +1 -1
- package/dist/src/utils/deepMerge.d.ts +2 -3
- package/dist/src/utils/events.d.ts +2 -1
- package/dist/src/utils/events.js +1 -0
- package/dist/src/utils/events.js.map +1 -1
- package/dist/src/utils/relaunch.d.ts +7 -0
- package/dist/src/utils/relaunch.js +57 -0
- package/dist/src/utils/relaunch.js.map +1 -0
- package/dist/src/utils/relaunch.test.d.ts +6 -0
- package/dist/src/utils/relaunch.test.js +273 -0
- package/dist/src/utils/relaunch.test.js.map +1 -0
- package/dist/src/zed-integration/schema.d.ts +22 -22
- package/dist/src/zed-integration/zedIntegration.d.ts +7 -0
- package/dist/src/zed-integration/zedIntegration.js +14 -2
- package/dist/src/zed-integration/zedIntegration.js.map +1 -1
- package/dist/tsconfig.tsbuildinfo +1 -1
- package/package.json +5 -4
|
@@ -0,0 +1,68 @@
|
|
|
1
|
+
import { jsx as _jsx } from "react/jsx-runtime";
|
|
2
|
+
/**
|
|
3
|
+
* @license
|
|
4
|
+
* Copyright 2025 Google LLC
|
|
5
|
+
* SPDX-License-Identifier: Apache-2.0
|
|
6
|
+
*/
|
|
7
|
+
import { describe, it, expect, vi, beforeEach } from 'vitest';
|
|
8
|
+
import { renderWithProviders } from '../../../test-utils/render.js';
|
|
9
|
+
import { DescriptiveRadioButtonSelect, } from './DescriptiveRadioButtonSelect.js';
|
|
10
|
+
vi.mock('./BaseSelectionList.js', async (importOriginal) => {
|
|
11
|
+
const actual = await importOriginal();
|
|
12
|
+
return {
|
|
13
|
+
...actual,
|
|
14
|
+
BaseSelectionList: vi.fn(({ children, ...props }) => (_jsx(actual.BaseSelectionList, { ...props, children: children }))),
|
|
15
|
+
};
|
|
16
|
+
});
|
|
17
|
+
vi.mock('../../semantic-colors.js', () => ({
|
|
18
|
+
theme: {
|
|
19
|
+
text: {
|
|
20
|
+
primary: 'COLOR_PRIMARY',
|
|
21
|
+
secondary: 'COLOR_SECONDARY',
|
|
22
|
+
},
|
|
23
|
+
status: {
|
|
24
|
+
success: 'COLOR_SUCCESS',
|
|
25
|
+
},
|
|
26
|
+
},
|
|
27
|
+
}));
|
|
28
|
+
describe('DescriptiveRadioButtonSelect', () => {
|
|
29
|
+
const mockOnSelect = vi.fn();
|
|
30
|
+
const mockOnHighlight = vi.fn();
|
|
31
|
+
const ITEMS = [
|
|
32
|
+
{ title: 'Foo Title', description: 'This is Foo.', value: 'foo' },
|
|
33
|
+
{ title: 'Bar Title', description: 'This is Bar.', value: 'bar' },
|
|
34
|
+
{
|
|
35
|
+
title: 'Baz Title',
|
|
36
|
+
description: 'This is Baz.',
|
|
37
|
+
value: 'baz',
|
|
38
|
+
disabled: true,
|
|
39
|
+
},
|
|
40
|
+
];
|
|
41
|
+
const renderComponent = (props = {}) => {
|
|
42
|
+
const defaultProps = {
|
|
43
|
+
items: ITEMS,
|
|
44
|
+
onSelect: mockOnSelect,
|
|
45
|
+
...props,
|
|
46
|
+
};
|
|
47
|
+
return renderWithProviders(_jsx(DescriptiveRadioButtonSelect, { ...defaultProps }));
|
|
48
|
+
};
|
|
49
|
+
beforeEach(() => {
|
|
50
|
+
vi.clearAllMocks();
|
|
51
|
+
});
|
|
52
|
+
it('should render correctly with default props', () => {
|
|
53
|
+
const { lastFrame } = renderComponent();
|
|
54
|
+
expect(lastFrame()).toMatchSnapshot();
|
|
55
|
+
});
|
|
56
|
+
it('should render correctly with custom props', () => {
|
|
57
|
+
const { lastFrame } = renderComponent({
|
|
58
|
+
initialIndex: 1,
|
|
59
|
+
isFocused: false,
|
|
60
|
+
showScrollArrows: true,
|
|
61
|
+
maxItemsToShow: 5,
|
|
62
|
+
showNumbers: true,
|
|
63
|
+
onHighlight: mockOnHighlight,
|
|
64
|
+
});
|
|
65
|
+
expect(lastFrame()).toMatchSnapshot();
|
|
66
|
+
});
|
|
67
|
+
});
|
|
68
|
+
//# sourceMappingURL=DescriptiveRadioButtonSelect.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"DescriptiveRadioButtonSelect.test.js","sourceRoot":"","sources":["../../../../../src/ui/components/shared/DescriptiveRadioButtonSelect.test.tsx"],"names":[],"mappings":";AAAA;;;;GAIG;AAEH,OAAO,EAAE,QAAQ,EAAE,EAAE,EAAE,MAAM,EAAE,EAAE,EAAE,UAAU,EAAE,MAAM,QAAQ,CAAC;AAC9D,OAAO,EAAE,mBAAmB,EAAE,MAAM,+BAA+B,CAAC;AACpE,OAAO,EACL,4BAA4B,GAG7B,MAAM,mCAAmC,CAAC;AAE3C,EAAE,CAAC,IAAI,CAAC,wBAAwB,EAAE,KAAK,EAAE,cAAc,EAAE,EAAE;IACzD,MAAM,MAAM,GACV,MAAM,cAAc,EAA2C,CAAC;IAClE,OAAO;QACL,GAAG,MAAM;QACT,iBAAiB,EAAE,EAAE,CAAC,EAAE,CAAC,CAAC,EAAE,QAAQ,EAAE,GAAG,KAAK,EAAE,EAAE,EAAE,CAAC,CACnD,KAAC,MAAM,CAAC,iBAAiB,OAAK,KAAK,YAAG,QAAQ,GAA4B,CAC3E,CAAC;KACH,CAAC;AACJ,CAAC,CAAC,CAAC;AAEH,EAAE,CAAC,IAAI,CAAC,0BAA0B,EAAE,GAAG,EAAE,CAAC,CAAC;IACzC,KAAK,EAAE;QACL,IAAI,EAAE;YACJ,OAAO,EAAE,eAAe;YACxB,SAAS,EAAE,iBAAiB;SAC7B;QACD,MAAM,EAAE;YACN,OAAO,EAAE,eAAe;SACzB;KACF;CACF,CAAC,CAAC,CAAC;AAEJ,QAAQ,CAAC,8BAA8B,EAAE,GAAG,EAAE;IAC5C,MAAM,YAAY,GAAG,EAAE,CAAC,EAAE,EAAE,CAAC;IAC7B,MAAM,eAAe,GAAG,EAAE,CAAC,EAAE,EAAE,CAAC;IAEhC,MAAM,KAAK,GAA8C;QACvD,EAAE,KAAK,EAAE,WAAW,EAAE,WAAW,EAAE,cAAc,EAAE,KAAK,EAAE,KAAK,EAAE;QACjE,EAAE,KAAK,EAAE,WAAW,EAAE,WAAW,EAAE,cAAc,EAAE,KAAK,EAAE,KAAK,EAAE;QACjE;YACE,KAAK,EAAE,WAAW;YAClB,WAAW,EAAE,cAAc;YAC3B,KAAK,EAAE,KAAK;YACZ,QAAQ,EAAE,IAAI;SACf;KACF,CAAC;IAEF,MAAM,eAAe,GAAG,CACtB,QAA4D,EAAE,EAC9D,EAAE;QACF,MAAM,YAAY,GAA8C;YAC9D,KAAK,EAAE,KAAK;YACZ,QAAQ,EAAE,YAAY;YACtB,GAAG,KAAK;SACT,CAAC;QACF,OAAO,mBAAmB,CACxB,KAAC,4BAA4B,OAAK,YAAY,GAAI,CACnD,CAAC;IACJ,CAAC,CAAC;IAEF,UAAU,CAAC,GAAG,EAAE;QACd,EAAE,CAAC,aAAa,EAAE,CAAC;IACrB,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,4CAA4C,EAAE,GAAG,EAAE;QACpD,MAAM,EAAE,SAAS,EAAE,GAAG,eAAe,EAAE,CAAC;QACxC,MAAM,CAAC,SAAS,EAAE,CAAC,CAAC,eAAe,EAAE,CAAC;IACxC,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,2CAA2C,EAAE,GAAG,EAAE;QACnD,MAAM,EAAE,SAAS,EAAE,GAAG,eAAe,CAAC;YACpC,YAAY,EAAE,CAAC;YACf,SAAS,EAAE,KAAK;YAChB,gBAAgB,EAAE,IAAI;YACtB,cAAc,EAAE,CAAC;YACjB,WAAW,EAAE,IAAI;YACjB,WAAW,EAAE,eAAe;SAC7B,CAAC,CAAC;QACH,MAAM,CAAC,SAAS,EAAE,CAAC,CAAC,eAAe,EAAE,CAAC;IACxC,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC"}
|
|
@@ -1,8 +1,7 @@
|
|
|
1
1
|
import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
|
|
2
|
-
import {
|
|
3
|
-
import { Text, Box } from 'ink';
|
|
2
|
+
import { Text } from 'ink';
|
|
4
3
|
import { theme } from '../../semantic-colors.js';
|
|
5
|
-
import {
|
|
4
|
+
import { BaseSelectionList } from './BaseSelectionList.js';
|
|
6
5
|
/**
|
|
7
6
|
* A custom component that displays a list of items with radio buttons,
|
|
8
7
|
* supporting scrolling and keyboard navigation.
|
|
@@ -10,107 +9,13 @@ import { useKeypress } from '../../hooks/useKeypress.js';
|
|
|
10
9
|
* @template T The type of the value associated with each radio item.
|
|
11
10
|
*/
|
|
12
11
|
export function RadioButtonSelect({ items, initialIndex = 0, onSelect, onHighlight, isFocused = true, showScrollArrows = false, maxItemsToShow = 10, showNumbers = true, }) {
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
useEffect(() => {
|
|
18
|
-
const newScrollOffset = Math.max(0, Math.min(activeIndex - maxItemsToShow + 1, items.length - maxItemsToShow));
|
|
19
|
-
if (activeIndex < scrollOffset) {
|
|
20
|
-
setScrollOffset(activeIndex);
|
|
21
|
-
}
|
|
22
|
-
else if (activeIndex >= scrollOffset + maxItemsToShow) {
|
|
23
|
-
setScrollOffset(newScrollOffset);
|
|
24
|
-
}
|
|
25
|
-
}, [activeIndex, items.length, scrollOffset, maxItemsToShow]);
|
|
26
|
-
useEffect(() => () => {
|
|
27
|
-
if (numberInputTimer.current) {
|
|
28
|
-
clearTimeout(numberInputTimer.current);
|
|
29
|
-
}
|
|
30
|
-
}, []);
|
|
31
|
-
useKeypress((key) => {
|
|
32
|
-
const { sequence, name } = key;
|
|
33
|
-
const isNumeric = showNumbers && /^[0-9]$/.test(sequence);
|
|
34
|
-
// Any key press that is not a digit should clear the number input buffer.
|
|
35
|
-
if (!isNumeric && numberInputTimer.current) {
|
|
36
|
-
clearTimeout(numberInputTimer.current);
|
|
37
|
-
setNumberInput('');
|
|
38
|
-
}
|
|
39
|
-
if (name === 'k' || name === 'up') {
|
|
40
|
-
const newIndex = activeIndex > 0 ? activeIndex - 1 : items.length - 1;
|
|
41
|
-
setActiveIndex(newIndex);
|
|
42
|
-
onHighlight?.(items[newIndex].value);
|
|
43
|
-
return;
|
|
44
|
-
}
|
|
45
|
-
if (name === 'j' || name === 'down') {
|
|
46
|
-
const newIndex = activeIndex < items.length - 1 ? activeIndex + 1 : 0;
|
|
47
|
-
setActiveIndex(newIndex);
|
|
48
|
-
onHighlight?.(items[newIndex].value);
|
|
49
|
-
return;
|
|
50
|
-
}
|
|
51
|
-
if (name === 'return') {
|
|
52
|
-
onSelect(items[activeIndex].value);
|
|
53
|
-
return;
|
|
54
|
-
}
|
|
55
|
-
// Handle numeric input for selection.
|
|
56
|
-
if (isNumeric) {
|
|
57
|
-
if (numberInputTimer.current) {
|
|
58
|
-
clearTimeout(numberInputTimer.current);
|
|
12
|
+
return (_jsx(BaseSelectionList, { items: items, initialIndex: initialIndex, onSelect: onSelect, onHighlight: onHighlight, isFocused: isFocused, showNumbers: showNumbers, showScrollArrows: showScrollArrows, maxItemsToShow: maxItemsToShow, renderItem: (item, { titleColor }) => {
|
|
13
|
+
// Handle special theme display case for ThemeDialog compatibility
|
|
14
|
+
if (item.themeNameDisplay && item.themeTypeDisplay) {
|
|
15
|
+
return (_jsxs(Text, { color: titleColor, wrap: "truncate", children: [item.themeNameDisplay, ' ', _jsx(Text, { color: theme.text.secondary, children: item.themeTypeDisplay })] }));
|
|
59
16
|
}
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
// A single '0' is not a valid selection since items are 1-indexed.
|
|
64
|
-
if (newNumberInput === '0') {
|
|
65
|
-
numberInputTimer.current = setTimeout(() => setNumberInput(''), 350);
|
|
66
|
-
return;
|
|
67
|
-
}
|
|
68
|
-
if (targetIndex >= 0 && targetIndex < items.length) {
|
|
69
|
-
const targetItem = items[targetIndex];
|
|
70
|
-
setActiveIndex(targetIndex);
|
|
71
|
-
onHighlight?.(targetItem.value);
|
|
72
|
-
// If the typed number can't be a prefix for another valid number,
|
|
73
|
-
// select it immediately. Otherwise, wait for more input.
|
|
74
|
-
const potentialNextNumber = Number.parseInt(newNumberInput + '0', 10);
|
|
75
|
-
if (potentialNextNumber > items.length) {
|
|
76
|
-
onSelect(targetItem.value);
|
|
77
|
-
setNumberInput('');
|
|
78
|
-
}
|
|
79
|
-
else {
|
|
80
|
-
numberInputTimer.current = setTimeout(() => {
|
|
81
|
-
onSelect(targetItem.value);
|
|
82
|
-
setNumberInput('');
|
|
83
|
-
}, 350); // Debounce time for multi-digit input.
|
|
84
|
-
}
|
|
85
|
-
}
|
|
86
|
-
else {
|
|
87
|
-
// The typed number is out of bounds, clear the buffer
|
|
88
|
-
setNumberInput('');
|
|
89
|
-
}
|
|
90
|
-
}
|
|
91
|
-
}, { isActive: !!(isFocused && items.length > 0) });
|
|
92
|
-
const visibleItems = items.slice(scrollOffset, scrollOffset + maxItemsToShow);
|
|
93
|
-
return (_jsxs(Box, { flexDirection: "column", children: [showScrollArrows && (_jsx(Text, { color: scrollOffset > 0 ? theme.text.primary : theme.text.secondary, children: "\u25B2" })), visibleItems.map((item, index) => {
|
|
94
|
-
const itemIndex = scrollOffset + index;
|
|
95
|
-
const isSelected = activeIndex === itemIndex;
|
|
96
|
-
let textColor = theme.text.primary;
|
|
97
|
-
let numberColor = theme.text.primary;
|
|
98
|
-
if (isSelected) {
|
|
99
|
-
textColor = theme.status.success;
|
|
100
|
-
numberColor = theme.status.success;
|
|
101
|
-
}
|
|
102
|
-
else if (item.disabled) {
|
|
103
|
-
textColor = theme.text.secondary;
|
|
104
|
-
numberColor = theme.text.secondary;
|
|
105
|
-
}
|
|
106
|
-
if (!showNumbers) {
|
|
107
|
-
numberColor = theme.text.secondary;
|
|
108
|
-
}
|
|
109
|
-
const numberColumnWidth = String(items.length).length;
|
|
110
|
-
const itemNumberText = `${String(itemIndex + 1).padStart(numberColumnWidth)}.`;
|
|
111
|
-
return (_jsxs(Box, { alignItems: "center", children: [_jsx(Box, { minWidth: 2, flexShrink: 0, children: _jsx(Text, { color: isSelected ? theme.status.success : theme.text.primary, "aria-hidden": true, children: isSelected ? '●' : ' ' }) }), _jsx(Box, { marginRight: 1, flexShrink: 0, minWidth: itemNumberText.length, "aria-state": { checked: isSelected }, children: _jsx(Text, { color: numberColor, children: itemNumberText }) }), item.themeNameDisplay && item.themeTypeDisplay ? (_jsxs(Text, { color: textColor, wrap: "truncate", children: [item.themeNameDisplay, ' ', _jsx(Text, { color: theme.text.secondary, children: item.themeTypeDisplay })] })) : (_jsx(Text, { color: textColor, wrap: "truncate", children: item.label }))] }, item.label));
|
|
112
|
-
}), showScrollArrows && (_jsx(Text, { color: scrollOffset + maxItemsToShow < items.length
|
|
113
|
-
? theme.text.primary
|
|
114
|
-
: theme.text.secondary, children: "\u25BC" }))] }));
|
|
17
|
+
// Regular label display
|
|
18
|
+
return (_jsx(Text, { color: titleColor, wrap: "truncate", children: item.label }));
|
|
19
|
+
} }));
|
|
115
20
|
}
|
|
116
21
|
//# sourceMappingURL=RadioButtonSelect.js.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"RadioButtonSelect.js","sourceRoot":"","sources":["../../../../../src/ui/components/shared/RadioButtonSelect.tsx"],"names":[],"mappings":";AAOA,OAAO,EAAE,
|
|
1
|
+
{"version":3,"file":"RadioButtonSelect.js","sourceRoot":"","sources":["../../../../../src/ui/components/shared/RadioButtonSelect.tsx"],"names":[],"mappings":";AAOA,OAAO,EAAE,IAAI,EAAE,MAAM,KAAK,CAAC;AAC3B,OAAO,EAAE,KAAK,EAAE,MAAM,0BAA0B,CAAC;AACjD,OAAO,EAAE,iBAAiB,EAAE,MAAM,wBAAwB,CAAC;AAqC3D;;;;;GAKG;AACH,MAAM,UAAU,iBAAiB,CAAI,EACnC,KAAK,EACL,YAAY,GAAG,CAAC,EAChB,QAAQ,EACR,WAAW,EACX,SAAS,GAAG,IAAI,EAChB,gBAAgB,GAAG,KAAK,EACxB,cAAc,GAAG,EAAE,EACnB,WAAW,GAAG,IAAI,GACQ;IAC1B,OAAO,CACL,KAAC,iBAAiB,IAChB,KAAK,EAAE,KAAK,EACZ,YAAY,EAAE,YAAY,EAC1B,QAAQ,EAAE,QAAQ,EAClB,WAAW,EAAE,WAAW,EACxB,SAAS,EAAE,SAAS,EACpB,WAAW,EAAE,WAAW,EACxB,gBAAgB,EAAE,gBAAgB,EAClC,cAAc,EAAE,cAAc,EAC9B,UAAU,EAAE,CAAC,IAAI,EAAE,EAAE,UAAU,EAAE,EAAE,EAAE;YACnC,kEAAkE;YAClE,IAAI,IAAI,CAAC,gBAAgB,IAAI,IAAI,CAAC,gBAAgB,EAAE,CAAC;gBACnD,OAAO,CACL,MAAC,IAAI,IAAC,KAAK,EAAE,UAAU,EAAE,IAAI,EAAC,UAAU,aACrC,IAAI,CAAC,gBAAgB,EAAE,GAAG,EAC3B,KAAC,IAAI,IAAC,KAAK,EAAE,KAAK,CAAC,IAAI,CAAC,SAAS,YAAG,IAAI,CAAC,gBAAgB,GAAQ,IAC5D,CACR,CAAC;YACJ,CAAC;YACD,wBAAwB;YACxB,OAAO,CACL,KAAC,IAAI,IAAC,KAAK,EAAE,UAAU,EAAE,IAAI,EAAC,UAAU,YACrC,IAAI,CAAC,KAAK,GACN,CACR,CAAC;QACJ,CAAC,GACD,CACH,CAAC;AACJ,CAAC"}
|
|
@@ -4,107 +4,128 @@ import { jsx as _jsx } from "react/jsx-runtime";
|
|
|
4
4
|
* Copyright 2025 Google LLC
|
|
5
5
|
* SPDX-License-Identifier: Apache-2.0
|
|
6
6
|
*/
|
|
7
|
+
import { describe, it, expect, vi, beforeEach } from 'vitest';
|
|
7
8
|
import { renderWithProviders } from '../../../test-utils/render.js';
|
|
8
|
-
import { waitFor } from '@testing-library/react';
|
|
9
9
|
import { RadioButtonSelect, } from './RadioButtonSelect.js';
|
|
10
|
-
import {
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
10
|
+
import { BaseSelectionList, } from './BaseSelectionList.js';
|
|
11
|
+
vi.mock('./BaseSelectionList.js', () => ({
|
|
12
|
+
BaseSelectionList: vi.fn(() => null),
|
|
13
|
+
}));
|
|
14
|
+
vi.mock('../../semantic-colors.js', () => ({
|
|
15
|
+
theme: {
|
|
16
|
+
text: { secondary: 'COLOR_SECONDARY' },
|
|
17
|
+
},
|
|
18
|
+
}));
|
|
19
|
+
const MockedBaseSelectionList = vi.mocked(BaseSelectionList);
|
|
20
|
+
const extractRenderItem = () => {
|
|
21
|
+
const mockCalls = MockedBaseSelectionList.mock.calls;
|
|
22
|
+
if (mockCalls.length === 0) {
|
|
23
|
+
throw new Error('BaseSelectionList was not called. Ensure RadioButtonSelect is rendered before calling extractRenderItem.');
|
|
24
|
+
}
|
|
25
|
+
const props = mockCalls[0][0];
|
|
26
|
+
if (typeof props.renderItem !== 'function') {
|
|
27
|
+
throw new Error('renderItem prop was not found on BaseSelectionList call.');
|
|
28
|
+
}
|
|
29
|
+
return props.renderItem;
|
|
30
|
+
};
|
|
31
|
+
describe('RadioButtonSelect', () => {
|
|
32
|
+
const mockOnSelect = vi.fn();
|
|
33
|
+
const mockOnHighlight = vi.fn();
|
|
34
|
+
const ITEMS = [
|
|
35
|
+
{ label: 'Option 1', value: 'one' },
|
|
36
|
+
{ label: 'Option 2', value: 'two' },
|
|
37
|
+
{ label: 'Option 3', value: 'three', disabled: true },
|
|
38
|
+
];
|
|
39
|
+
const renderComponent = (props = {}) => {
|
|
40
|
+
const defaultProps = {
|
|
41
|
+
items: ITEMS,
|
|
42
|
+
onSelect: mockOnSelect,
|
|
43
|
+
...props,
|
|
44
|
+
};
|
|
45
|
+
return renderWithProviders(_jsx(RadioButtonSelect, { ...defaultProps }));
|
|
46
|
+
};
|
|
47
|
+
beforeEach(() => {
|
|
48
|
+
vi.clearAllMocks();
|
|
20
49
|
});
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
50
|
+
describe('Prop forwarding to BaseSelectionList', () => {
|
|
51
|
+
it('should forward all props correctly when provided', () => {
|
|
52
|
+
const props = {
|
|
53
|
+
items: ITEMS,
|
|
54
|
+
initialIndex: 1,
|
|
55
|
+
onSelect: mockOnSelect,
|
|
56
|
+
onHighlight: mockOnHighlight,
|
|
57
|
+
isFocused: false,
|
|
58
|
+
showScrollArrows: true,
|
|
59
|
+
maxItemsToShow: 5,
|
|
60
|
+
showNumbers: false,
|
|
61
|
+
};
|
|
62
|
+
renderComponent(props);
|
|
63
|
+
expect(BaseSelectionList).toHaveBeenCalledTimes(1);
|
|
64
|
+
expect(BaseSelectionList).toHaveBeenCalledWith(expect.objectContaining({
|
|
65
|
+
...props,
|
|
66
|
+
renderItem: expect.any(Function),
|
|
67
|
+
}), undefined);
|
|
68
|
+
});
|
|
69
|
+
it('should use default props if not provided', () => {
|
|
70
|
+
renderComponent({
|
|
71
|
+
items: ITEMS,
|
|
72
|
+
onSelect: mockOnSelect,
|
|
73
|
+
});
|
|
74
|
+
expect(BaseSelectionList).toHaveBeenCalledWith(expect.objectContaining({
|
|
75
|
+
initialIndex: 0,
|
|
76
|
+
isFocused: true,
|
|
77
|
+
showScrollArrows: false,
|
|
78
|
+
maxItemsToShow: 10,
|
|
79
|
+
showNumbers: true,
|
|
80
|
+
}), undefined);
|
|
81
|
+
});
|
|
36
82
|
});
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
83
|
+
describe('renderItem implementation', () => {
|
|
84
|
+
let renderItem;
|
|
85
|
+
const mockContext = {
|
|
86
|
+
isSelected: false,
|
|
87
|
+
titleColor: 'MOCK_TITLE_COLOR',
|
|
88
|
+
numberColor: 'MOCK_NUMBER_COLOR',
|
|
89
|
+
};
|
|
90
|
+
beforeEach(() => {
|
|
91
|
+
renderComponent();
|
|
92
|
+
renderItem = extractRenderItem();
|
|
93
|
+
});
|
|
94
|
+
it('should render the standard label display with correct color and truncation', () => {
|
|
95
|
+
const item = ITEMS[0];
|
|
96
|
+
const result = renderItem(item, mockContext);
|
|
97
|
+
expect(result?.props?.color).toBe(mockContext.titleColor);
|
|
98
|
+
expect(result?.props?.children).toBe('Option 1');
|
|
99
|
+
expect(result?.props?.wrap).toBe('truncate');
|
|
100
|
+
});
|
|
101
|
+
it('should render the special theme display when theme props are present', () => {
|
|
102
|
+
const themeItem = {
|
|
40
103
|
label: 'Theme A (Light)',
|
|
41
104
|
value: 'a-light',
|
|
42
105
|
themeNameDisplay: 'Theme A',
|
|
43
106
|
themeTypeDisplay: '(Light)',
|
|
44
|
-
}
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
label: `Item ${i + 1}`,
|
|
58
|
-
value: `item-${i + 1}`,
|
|
59
|
-
}));
|
|
60
|
-
const { lastFrame } = renderWithProviders(_jsx(RadioButtonSelect, { items: manyItems, onSelect: () => { } }));
|
|
61
|
-
expect(lastFrame()).toMatchSnapshot();
|
|
62
|
-
});
|
|
63
|
-
it('renders nothing when no items are provided', () => {
|
|
64
|
-
const { lastFrame } = renderWithProviders(_jsx(RadioButtonSelect, { items: [], onSelect: () => { }, isFocused: true }));
|
|
65
|
-
expect(lastFrame()).toBe('');
|
|
66
|
-
});
|
|
67
|
-
});
|
|
68
|
-
describe('keyboard navigation', () => {
|
|
69
|
-
it('should call onSelect when "enter" is pressed', () => {
|
|
70
|
-
const onSelect = vi.fn();
|
|
71
|
-
const { stdin } = renderWithProviders(_jsx(RadioButtonSelect, { items: ITEMS, onSelect: onSelect }));
|
|
72
|
-
stdin.write('\r');
|
|
73
|
-
expect(onSelect).toHaveBeenCalledWith('one');
|
|
74
|
-
});
|
|
75
|
-
describe('when isFocused is false', () => {
|
|
76
|
-
it('should not handle any keyboard input', () => {
|
|
77
|
-
const onSelect = vi.fn();
|
|
78
|
-
const { stdin } = renderWithProviders(_jsx(RadioButtonSelect, { items: ITEMS, onSelect: onSelect, isFocused: false }));
|
|
79
|
-
stdin.write('\u001B[B'); // Down arrow
|
|
80
|
-
stdin.write('\u001B[A'); // Up arrow
|
|
81
|
-
stdin.write('\r'); // Enter
|
|
82
|
-
expect(onSelect).not.toHaveBeenCalled();
|
|
107
|
+
};
|
|
108
|
+
const result = renderItem(themeItem, mockContext);
|
|
109
|
+
expect(result?.props?.color).toBe(mockContext.titleColor);
|
|
110
|
+
expect(result?.props?.wrap).toBe('truncate');
|
|
111
|
+
const children = result?.props?.children;
|
|
112
|
+
if (!Array.isArray(children) || children.length < 3) {
|
|
113
|
+
throw new Error('Expected children to be an array with at least 3 elements for theme display');
|
|
114
|
+
}
|
|
115
|
+
expect(children[0]).toBe('Theme A');
|
|
116
|
+
expect(children[1]).toBe(' ');
|
|
117
|
+
const nestedTextElement = children[2];
|
|
118
|
+
expect(nestedTextElement?.props?.color).toBe('COLOR_SECONDARY');
|
|
119
|
+
expect(nestedTextElement?.props?.children).toBe('(Light)');
|
|
83
120
|
});
|
|
84
|
-
|
|
85
|
-
|
|
86
|
-
|
|
87
|
-
|
|
88
|
-
|
|
89
|
-
|
|
90
|
-
const
|
|
91
|
-
|
|
92
|
-
stdin.write('\u001B[B'); // Down arrow
|
|
93
|
-
await waitFor(() => {
|
|
94
|
-
expect(lastFrame()).toContain('● 2. Option 2');
|
|
95
|
-
});
|
|
96
|
-
stdin.write('\r');
|
|
97
|
-
expect(onSelect).toHaveBeenCalledWith('two');
|
|
98
|
-
});
|
|
99
|
-
it('should navigate up with arrow key and select with enter', async () => {
|
|
100
|
-
const onSelect = vi.fn();
|
|
101
|
-
const { stdin, lastFrame } = renderWithProviders(_jsx(RadioButtonSelect, { items: ITEMS, onSelect: onSelect, initialIndex: 1, isFocused: isFocused }));
|
|
102
|
-
stdin.write('\u001B[A'); // Up arrow
|
|
103
|
-
await waitFor(() => {
|
|
104
|
-
expect(lastFrame()).toContain('● 1. Option 1');
|
|
105
|
-
});
|
|
106
|
-
stdin.write('\r');
|
|
107
|
-
expect(onSelect).toHaveBeenCalledWith('one');
|
|
121
|
+
it('should fall back to standard display if only one theme prop is present', () => {
|
|
122
|
+
const partialThemeItem = {
|
|
123
|
+
label: 'Incomplete Theme',
|
|
124
|
+
value: 'incomplete',
|
|
125
|
+
themeNameDisplay: 'Only Name',
|
|
126
|
+
};
|
|
127
|
+
const result = renderItem(partialThemeItem, mockContext);
|
|
128
|
+
expect(result?.props?.children).toBe('Incomplete Theme');
|
|
108
129
|
});
|
|
109
130
|
});
|
|
110
131
|
});
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"RadioButtonSelect.test.js","sourceRoot":"","sources":["../../../../../src/ui/components/shared/RadioButtonSelect.test.tsx"],"names":[],"mappings":";AAAA;;;;GAIG;AAEH,OAAO,EAAE,
|
|
1
|
+
{"version":3,"file":"RadioButtonSelect.test.js","sourceRoot":"","sources":["../../../../../src/ui/components/shared/RadioButtonSelect.test.tsx"],"names":[],"mappings":";AAAA;;;;GAIG;AAEH,OAAO,EAAE,QAAQ,EAAE,EAAE,EAAE,MAAM,EAAE,EAAE,EAAE,UAAU,EAAE,MAAM,QAAQ,CAAC;AAC9D,OAAO,EAAE,mBAAmB,EAAE,MAAM,+BAA+B,CAAC;AAEpE,OAAO,EACL,iBAAiB,GAGlB,MAAM,wBAAwB,CAAC;AAChC,OAAO,EACL,iBAAiB,GAGlB,MAAM,wBAAwB,CAAC;AAEhC,EAAE,CAAC,IAAI,CAAC,wBAAwB,EAAE,GAAG,EAAE,CAAC,CAAC;IACvC,iBAAiB,EAAE,EAAE,CAAC,EAAE,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC;CACrC,CAAC,CAAC,CAAC;AAEJ,EAAE,CAAC,IAAI,CAAC,0BAA0B,EAAE,GAAG,EAAE,CAAC,CAAC;IACzC,KAAK,EAAE;QACL,IAAI,EAAE,EAAE,SAAS,EAAE,iBAAiB,EAAE;KACvC;CACF,CAAC,CAAC,CAAC;AAEJ,MAAM,uBAAuB,GAAG,EAAE,CAAC,MAAM,CACvC,iBAAiB,CACqB,CAAC;AAMzC,MAAM,iBAAiB,GAAG,GAAsB,EAAE;IAChD,MAAM,SAAS,GAAG,uBAAuB,CAAC,IAAI,CAAC,KAAK,CAAC;IAErD,IAAI,SAAS,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;QAC3B,MAAM,IAAI,KAAK,CACb,0GAA0G,CAC3G,CAAC;IACJ,CAAC;IAED,MAAM,KAAK,GAAG,SAAS,CAAC,CAAC,CAAC,CAAC,CAAC,CAG3B,CAAC;IAEF,IAAI,OAAO,KAAK,CAAC,UAAU,KAAK,UAAU,EAAE,CAAC;QAC3C,MAAM,IAAI,KAAK,CAAC,0DAA0D,CAAC,CAAC;IAC9E,CAAC;IAED,OAAO,KAAK,CAAC,UAA+B,CAAC;AAC/C,CAAC,CAAC;AAEF,QAAQ,CAAC,mBAAmB,EAAE,GAAG,EAAE;IACjC,MAAM,YAAY,GAAG,EAAE,CAAC,EAAE,EAAE,CAAC;IAC7B,MAAM,eAAe,GAAG,EAAE,CAAC,EAAE,EAAE,CAAC;IAEhC,MAAM,KAAK,GAAmC;QAC5C,EAAE,KAAK,EAAE,UAAU,EAAE,KAAK,EAAE,KAAK,EAAE;QACnC,EAAE,KAAK,EAAE,UAAU,EAAE,KAAK,EAAE,KAAK,EAAE;QACnC,EAAE,KAAK,EAAE,UAAU,EAAE,KAAK,EAAE,OAAO,EAAE,QAAQ,EAAE,IAAI,EAAE;KACtD,CAAC;IAEF,MAAM,eAAe,GAAG,CACtB,QAAiD,EAAE,EACnD,EAAE;QACF,MAAM,YAAY,GAAmC;YACnD,KAAK,EAAE,KAAK;YACZ,QAAQ,EAAE,YAAY;YACtB,GAAG,KAAK;SACT,CAAC;QACF,OAAO,mBAAmB,CAAC,KAAC,iBAAiB,OAAK,YAAY,GAAI,CAAC,CAAC;IACtE,CAAC,CAAC;IAEF,UAAU,CAAC,GAAG,EAAE;QACd,EAAE,CAAC,aAAa,EAAE,CAAC;IACrB,CAAC,CAAC,CAAC;IAEH,QAAQ,CAAC,sCAAsC,EAAE,GAAG,EAAE;QACpD,EAAE,CAAC,kDAAkD,EAAE,GAAG,EAAE;YAC1D,MAAM,KAAK,GAAG;gBACZ,KAAK,EAAE,KAAK;gBACZ,YAAY,EAAE,CAAC;gBACf,QAAQ,EAAE,YAAY;gBACtB,WAAW,EAAE,eAAe;gBAC5B,SAAS,EAAE,KAAK;gBAChB,gBAAgB,EAAE,IAAI;gBACtB,cAAc,EAAE,CAAC;gBACjB,WAAW,EAAE,KAAK;aACnB,CAAC;YAEF,eAAe,CAAC,KAAK,CAAC,CAAC;YAEvB,MAAM,CAAC,iBAAiB,CAAC,CAAC,qBAAqB,CAAC,CAAC,CAAC,CAAC;YACnD,MAAM,CAAC,iBAAiB,CAAC,CAAC,oBAAoB,CAC5C,MAAM,CAAC,gBAAgB,CAAC;gBACtB,GAAG,KAAK;gBACR,UAAU,EAAE,MAAM,CAAC,GAAG,CAAC,QAAQ,CAAC;aACjC,CAAC,EACF,SAAS,CACV,CAAC;QACJ,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,0CAA0C,EAAE,GAAG,EAAE;YAClD,eAAe,CAAC;gBACd,KAAK,EAAE,KAAK;gBACZ,QAAQ,EAAE,YAAY;aACvB,CAAC,CAAC;YAEH,MAAM,CAAC,iBAAiB,CAAC,CAAC,oBAAoB,CAC5C,MAAM,CAAC,gBAAgB,CAAC;gBACtB,YAAY,EAAE,CAAC;gBACf,SAAS,EAAE,IAAI;gBACf,gBAAgB,EAAE,KAAK;gBACvB,cAAc,EAAE,EAAE;gBAClB,WAAW,EAAE,IAAI;aAClB,CAAC,EACF,SAAS,CACV,CAAC;QACJ,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,QAAQ,CAAC,2BAA2B,EAAE,GAAG,EAAE;QACzC,IAAI,UAA6B,CAAC;QAClC,MAAM,WAAW,GAAsB;YACrC,UAAU,EAAE,KAAK;YACjB,UAAU,EAAE,kBAAkB;YAC9B,WAAW,EAAE,mBAAmB;SACjC,CAAC;QAEF,UAAU,CAAC,GAAG,EAAE;YACd,eAAe,EAAE,CAAC;YAClB,UAAU,GAAG,iBAAiB,EAAE,CAAC;QACnC,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,4EAA4E,EAAE,GAAG,EAAE;YACpF,MAAM,IAAI,GAAG,KAAK,CAAC,CAAC,CAAC,CAAC;YAEtB,MAAM,MAAM,GAAG,UAAU,CAAC,IAAI,EAAE,WAAW,CAAC,CAAC;YAE7C,MAAM,CAAC,MAAM,EAAE,KAAK,EAAE,KAAK,CAAC,CAAC,IAAI,CAAC,WAAW,CAAC,UAAU,CAAC,CAAC;YAC1D,MAAM,CAAC,MAAM,EAAE,KAAK,EAAE,QAAQ,CAAC,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;YACjD,MAAM,CAAC,MAAM,EAAE,KAAK,EAAE,IAAI,CAAC,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;QAC/C,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,sEAAsE,EAAE,GAAG,EAAE;YAC9E,MAAM,SAAS,GAA4B;gBACzC,KAAK,EAAE,iBAAiB;gBACxB,KAAK,EAAE,SAAS;gBAChB,gBAAgB,EAAE,SAAS;gBAC3B,gBAAgB,EAAE,SAAS;aAC5B,CAAC;YAEF,MAAM,MAAM,GAAG,UAAU,CAAC,SAAS,EAAE,WAAW,CAAC,CAAC;YAElD,MAAM,CAAC,MAAM,EAAE,KAAK,EAAE,KAAK,CAAC,CAAC,IAAI,CAAC,WAAW,CAAC,UAAU,CAAC,CAAC;YAC1D,MAAM,CAAC,MAAM,EAAE,KAAK,EAAE,IAAI,CAAC,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;YAE7C,MAAM,QAAQ,GAAG,MAAM,EAAE,KAAK,EAAE,QAAQ,CAAC;YAEzC,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,QAAQ,CAAC,IAAI,QAAQ,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;gBACpD,MAAM,IAAI,KAAK,CACb,6EAA6E,CAC9E,CAAC;YACJ,CAAC;YAED,MAAM,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;YACpC,MAAM,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;YAE9B,MAAM,iBAAiB,GAAG,QAAQ,CAAC,CAAC,CAGlC,CAAC;YACH,MAAM,CAAC,iBAAiB,EAAE,KAAK,EAAE,KAAK,CAAC,CAAC,IAAI,CAAC,iBAAiB,CAAC,CAAC;YAChE,MAAM,CAAC,iBAAiB,EAAE,KAAK,EAAE,QAAQ,CAAC,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;QAC7D,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,wEAAwE,EAAE,GAAG,EAAE;YAChF,MAAM,gBAAgB,GAA4B;gBAChD,KAAK,EAAE,kBAAkB;gBACzB,KAAK,EAAE,YAAY;gBACnB,gBAAgB,EAAE,WAAW;aAC9B,CAAC;YAEF,MAAM,MAAM,GAAG,UAAU,CAAC,gBAAgB,EAAE,WAAW,CAAC,CAAC;YAEzD,MAAM,CAAC,MAAM,EAAE,KAAK,EAAE,QAAQ,CAAC,CAAC,IAAI,CAAC,kBAAkB,CAAC,CAAC;QAC3D,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC"}
|
|
@@ -1470,7 +1470,7 @@ export function useTextBuffer({ initialText = '', initialCursorOffset = 0, viewp
|
|
|
1470
1470
|
const moveToOffset = useCallback((offset) => {
|
|
1471
1471
|
dispatch({ type: 'move_to_offset', payload: { offset } });
|
|
1472
1472
|
}, []);
|
|
1473
|
-
const returnValue = {
|
|
1473
|
+
const returnValue = useMemo(() => ({
|
|
1474
1474
|
lines,
|
|
1475
1475
|
text,
|
|
1476
1476
|
cursor: [cursorRow, cursorCol],
|
|
@@ -1531,7 +1531,68 @@ export function useTextBuffer({ initialText = '', initialCursorOffset = 0, viewp
|
|
|
1531
1531
|
vimMoveToLastLine,
|
|
1532
1532
|
vimMoveToLine,
|
|
1533
1533
|
vimEscapeInsertMode,
|
|
1534
|
-
}
|
|
1534
|
+
}), [
|
|
1535
|
+
lines,
|
|
1536
|
+
text,
|
|
1537
|
+
cursorRow,
|
|
1538
|
+
cursorCol,
|
|
1539
|
+
preferredCol,
|
|
1540
|
+
selectionAnchor,
|
|
1541
|
+
visualLines,
|
|
1542
|
+
renderedVisualLines,
|
|
1543
|
+
visualCursor,
|
|
1544
|
+
visualScrollRow,
|
|
1545
|
+
setText,
|
|
1546
|
+
insert,
|
|
1547
|
+
newline,
|
|
1548
|
+
backspace,
|
|
1549
|
+
del,
|
|
1550
|
+
move,
|
|
1551
|
+
undo,
|
|
1552
|
+
redo,
|
|
1553
|
+
replaceRange,
|
|
1554
|
+
replaceRangeByOffset,
|
|
1555
|
+
moveToOffset,
|
|
1556
|
+
deleteWordLeft,
|
|
1557
|
+
deleteWordRight,
|
|
1558
|
+
killLineRight,
|
|
1559
|
+
killLineLeft,
|
|
1560
|
+
handleInput,
|
|
1561
|
+
openInExternalEditor,
|
|
1562
|
+
vimDeleteWordForward,
|
|
1563
|
+
vimDeleteWordBackward,
|
|
1564
|
+
vimDeleteWordEnd,
|
|
1565
|
+
vimChangeWordForward,
|
|
1566
|
+
vimChangeWordBackward,
|
|
1567
|
+
vimChangeWordEnd,
|
|
1568
|
+
vimDeleteLine,
|
|
1569
|
+
vimChangeLine,
|
|
1570
|
+
vimDeleteToEndOfLine,
|
|
1571
|
+
vimChangeToEndOfLine,
|
|
1572
|
+
vimChangeMovement,
|
|
1573
|
+
vimMoveLeft,
|
|
1574
|
+
vimMoveRight,
|
|
1575
|
+
vimMoveUp,
|
|
1576
|
+
vimMoveDown,
|
|
1577
|
+
vimMoveWordForward,
|
|
1578
|
+
vimMoveWordBackward,
|
|
1579
|
+
vimMoveWordEnd,
|
|
1580
|
+
vimDeleteChar,
|
|
1581
|
+
vimInsertAtCursor,
|
|
1582
|
+
vimAppendAtCursor,
|
|
1583
|
+
vimOpenLineBelow,
|
|
1584
|
+
vimOpenLineAbove,
|
|
1585
|
+
vimAppendAtLineEnd,
|
|
1586
|
+
vimInsertAtLineStart,
|
|
1587
|
+
vimMoveToLineStart,
|
|
1588
|
+
vimMoveToLineEnd,
|
|
1589
|
+
vimMoveToFirstNonWhitespace,
|
|
1590
|
+
vimMoveToFirstLine,
|
|
1591
|
+
vimMoveToLastLine,
|
|
1592
|
+
vimMoveToLine,
|
|
1593
|
+
vimEscapeInsertMode,
|
|
1594
|
+
visualToLogicalMap,
|
|
1595
|
+
]);
|
|
1535
1596
|
return returnValue;
|
|
1536
1597
|
}
|
|
1537
1598
|
//# sourceMappingURL=text-buffer.js.map
|