mirror of
https://github.com/google-gemini/gemini-cli.git
synced 2026-04-26 13:04:49 -07:00
Move scope settings to new dialog (#7836)
This commit is contained in:
@@ -0,0 +1,98 @@
|
|||||||
|
/**
|
||||||
|
* @license
|
||||||
|
* Copyright 2025 Google LLC
|
||||||
|
* SPDX-License-Identifier: Apache-2.0
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { render } from 'ink-testing-library';
|
||||||
|
import { describe, it, expect, vi, beforeEach, afterEach } from 'vitest';
|
||||||
|
import { ThemeDialog } from './ThemeDialog.js';
|
||||||
|
import { LoadedSettings } from '../../config/settings.js';
|
||||||
|
import { KeypressProvider } from '../contexts/KeypressContext.js';
|
||||||
|
import { SettingsContext } from '../contexts/SettingsContext.js';
|
||||||
|
import { DEFAULT_THEME, themeManager } from '../themes/theme-manager.js';
|
||||||
|
import { act } from 'react';
|
||||||
|
|
||||||
|
const createMockSettings = (
|
||||||
|
userSettings = {},
|
||||||
|
workspaceSettings = {},
|
||||||
|
systemSettings = {},
|
||||||
|
): LoadedSettings =>
|
||||||
|
new LoadedSettings(
|
||||||
|
{
|
||||||
|
settings: { ui: { customThemes: {} }, ...systemSettings },
|
||||||
|
path: '/system/settings.json',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
settings: {},
|
||||||
|
path: '/system/system-defaults.json',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
settings: {
|
||||||
|
ui: { customThemes: {} },
|
||||||
|
...userSettings,
|
||||||
|
},
|
||||||
|
path: '/user/settings.json',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
settings: {
|
||||||
|
ui: { customThemes: {} },
|
||||||
|
...workspaceSettings,
|
||||||
|
},
|
||||||
|
path: '/workspace/settings.json',
|
||||||
|
},
|
||||||
|
true,
|
||||||
|
new Set(),
|
||||||
|
);
|
||||||
|
|
||||||
|
describe('ThemeDialog Snapshots', () => {
|
||||||
|
const baseProps = {
|
||||||
|
onSelect: vi.fn(),
|
||||||
|
onHighlight: vi.fn(),
|
||||||
|
availableTerminalHeight: 40,
|
||||||
|
terminalWidth: 120,
|
||||||
|
};
|
||||||
|
|
||||||
|
beforeEach(() => {
|
||||||
|
// Reset theme manager to a known state
|
||||||
|
themeManager.setActiveTheme(DEFAULT_THEME.name);
|
||||||
|
});
|
||||||
|
|
||||||
|
afterEach(() => {
|
||||||
|
vi.restoreAllMocks();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should render correctly in theme selection mode', () => {
|
||||||
|
const settings = createMockSettings();
|
||||||
|
const { lastFrame } = render(
|
||||||
|
<SettingsContext.Provider value={settings}>
|
||||||
|
<KeypressProvider kittyProtocolEnabled={false}>
|
||||||
|
<ThemeDialog {...baseProps} settings={settings} />
|
||||||
|
</KeypressProvider>
|
||||||
|
</SettingsContext.Provider>,
|
||||||
|
);
|
||||||
|
|
||||||
|
expect(lastFrame()).toMatchSnapshot();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should render correctly in scope selector mode', async () => {
|
||||||
|
const settings = createMockSettings();
|
||||||
|
const { lastFrame, stdin } = render(
|
||||||
|
<SettingsContext.Provider value={settings}>
|
||||||
|
<KeypressProvider kittyProtocolEnabled={false}>
|
||||||
|
<ThemeDialog {...baseProps} settings={settings} />
|
||||||
|
</KeypressProvider>
|
||||||
|
</SettingsContext.Provider>,
|
||||||
|
);
|
||||||
|
|
||||||
|
// Press Tab to switch to scope selector mode
|
||||||
|
act(() => {
|
||||||
|
stdin.write('\t');
|
||||||
|
});
|
||||||
|
|
||||||
|
// Need to wait for the state update to propagate
|
||||||
|
await new Promise((resolve) => setTimeout(resolve, 100));
|
||||||
|
|
||||||
|
expect(lastFrame()).toMatchSnapshot();
|
||||||
|
});
|
||||||
|
});
|
||||||
@@ -14,11 +14,9 @@ import { DiffRenderer } from './messages/DiffRenderer.js';
|
|||||||
import { colorizeCode } from '../utils/CodeColorizer.js';
|
import { colorizeCode } from '../utils/CodeColorizer.js';
|
||||||
import type { LoadedSettings } from '../../config/settings.js';
|
import type { LoadedSettings } from '../../config/settings.js';
|
||||||
import { SettingScope } from '../../config/settings.js';
|
import { SettingScope } from '../../config/settings.js';
|
||||||
import {
|
import { getScopeMessageForSetting } from '../../utils/dialogScopeUtils.js';
|
||||||
getScopeItems,
|
|
||||||
getScopeMessageForSetting,
|
|
||||||
} from '../../utils/dialogScopeUtils.js';
|
|
||||||
import { useKeypress } from '../hooks/useKeypress.js';
|
import { useKeypress } from '../hooks/useKeypress.js';
|
||||||
|
import { ScopeSelector } from './shared/ScopeSelector.js';
|
||||||
|
|
||||||
interface ThemeDialogProps {
|
interface ThemeDialogProps {
|
||||||
/** Callback function when a theme is selected */
|
/** Callback function when a theme is selected */
|
||||||
@@ -73,18 +71,14 @@ export function ThemeDialog({
|
|||||||
themeTypeDisplay: 'Custom',
|
themeTypeDisplay: 'Custom',
|
||||||
})),
|
})),
|
||||||
];
|
];
|
||||||
const [selectInputKey, setSelectInputKey] = useState(Date.now());
|
|
||||||
|
|
||||||
// Find the index of the selected theme, but only if it exists in the list
|
// Find the index of the selected theme, but only if it exists in the list
|
||||||
const selectedThemeName = settings.merged.ui?.theme || DEFAULT_THEME.name;
|
|
||||||
const initialThemeIndex = themeItems.findIndex(
|
const initialThemeIndex = themeItems.findIndex(
|
||||||
(item) => item.value === selectedThemeName,
|
(item) => item.value === highlightedThemeName,
|
||||||
);
|
);
|
||||||
// If not found, fall back to the first theme
|
// If not found, fall back to the first theme
|
||||||
const safeInitialThemeIndex = initialThemeIndex >= 0 ? initialThemeIndex : 0;
|
const safeInitialThemeIndex = initialThemeIndex >= 0 ? initialThemeIndex : 0;
|
||||||
|
|
||||||
const scopeItems = getScopeItems();
|
|
||||||
|
|
||||||
const handleThemeSelect = useCallback(
|
const handleThemeSelect = useCallback(
|
||||||
(themeName: string) => {
|
(themeName: string) => {
|
||||||
onSelect(themeName, selectedScope);
|
onSelect(themeName, selectedScope);
|
||||||
@@ -99,25 +93,21 @@ export function ThemeDialog({
|
|||||||
|
|
||||||
const handleScopeHighlight = useCallback((scope: SettingScope) => {
|
const handleScopeHighlight = useCallback((scope: SettingScope) => {
|
||||||
setSelectedScope(scope);
|
setSelectedScope(scope);
|
||||||
setSelectInputKey(Date.now());
|
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
const handleScopeSelect = useCallback(
|
const handleScopeSelect = useCallback(
|
||||||
(scope: SettingScope) => {
|
(scope: SettingScope) => {
|
||||||
handleScopeHighlight(scope);
|
onSelect(highlightedThemeName, scope);
|
||||||
setFocusedSection('theme'); // Reset focus to theme section
|
|
||||||
},
|
},
|
||||||
[handleScopeHighlight],
|
[onSelect, highlightedThemeName],
|
||||||
);
|
);
|
||||||
|
|
||||||
const [focusedSection, setFocusedSection] = useState<'theme' | 'scope'>(
|
const [mode, setMode] = useState<'theme' | 'scope'>('theme');
|
||||||
'theme',
|
|
||||||
);
|
|
||||||
|
|
||||||
useKeypress(
|
useKeypress(
|
||||||
(key) => {
|
(key) => {
|
||||||
if (key.name === 'tab') {
|
if (key.name === 'tab') {
|
||||||
setFocusedSection((prev) => (prev === 'theme' ? 'scope' : 'theme'));
|
setMode((prev) => (prev === 'theme' ? 'scope' : 'theme'));
|
||||||
}
|
}
|
||||||
if (key.name === 'escape') {
|
if (key.name === 'escape') {
|
||||||
onSelect(undefined, selectedScope);
|
onSelect(undefined, selectedScope);
|
||||||
@@ -152,20 +142,13 @@ export function ThemeDialog({
|
|||||||
|
|
||||||
const DIALOG_PADDING = 2;
|
const DIALOG_PADDING = 2;
|
||||||
const selectThemeHeight = themeItems.length + 1;
|
const selectThemeHeight = themeItems.length + 1;
|
||||||
const SCOPE_SELECTION_HEIGHT = 4; // Height for the scope selection section + margin.
|
|
||||||
const SPACE_BETWEEN_THEME_SELECTION_AND_APPLY_TO = 1;
|
|
||||||
const TAB_TO_SELECT_HEIGHT = 2;
|
const TAB_TO_SELECT_HEIGHT = 2;
|
||||||
availableTerminalHeight = availableTerminalHeight ?? Number.MAX_SAFE_INTEGER;
|
availableTerminalHeight = availableTerminalHeight ?? Number.MAX_SAFE_INTEGER;
|
||||||
availableTerminalHeight -= 2; // Top and bottom borders.
|
availableTerminalHeight -= 2; // Top and bottom borders.
|
||||||
availableTerminalHeight -= TAB_TO_SELECT_HEIGHT;
|
availableTerminalHeight -= TAB_TO_SELECT_HEIGHT;
|
||||||
|
|
||||||
let totalLeftHandSideHeight =
|
let totalLeftHandSideHeight = DIALOG_PADDING + selectThemeHeight;
|
||||||
DIALOG_PADDING +
|
|
||||||
selectThemeHeight +
|
|
||||||
SCOPE_SELECTION_HEIGHT +
|
|
||||||
SPACE_BETWEEN_THEME_SELECTION_AND_APPLY_TO;
|
|
||||||
|
|
||||||
let showScopeSelection = true;
|
|
||||||
let includePadding = true;
|
let includePadding = true;
|
||||||
|
|
||||||
// Remove content from the LHS that can be omitted if it exceeds the available height.
|
// Remove content from the LHS that can be omitted if it exceeds the available height.
|
||||||
@@ -174,15 +157,6 @@ export function ThemeDialog({
|
|||||||
totalLeftHandSideHeight -= DIALOG_PADDING;
|
totalLeftHandSideHeight -= DIALOG_PADDING;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (totalLeftHandSideHeight > availableTerminalHeight) {
|
|
||||||
// First, try hiding the scope selection
|
|
||||||
totalLeftHandSideHeight -= SCOPE_SELECTION_HEIGHT;
|
|
||||||
showScopeSelection = false;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Don't focus the scope selection if it is hidden due to height constraints.
|
|
||||||
const currentFocusedSection = !showScopeSelection ? 'theme' : focusedSection;
|
|
||||||
|
|
||||||
// Vertical space taken by elements other than the two code blocks in the preview pane.
|
// Vertical space taken by elements other than the two code blocks in the preview pane.
|
||||||
// Includes "Preview" title, borders, and margin between blocks.
|
// Includes "Preview" title, borders, and margin between blocks.
|
||||||
const PREVIEW_PANE_FIXED_VERTICAL_SPACE = 8;
|
const PREVIEW_PANE_FIXED_VERTICAL_SPACE = 8;
|
||||||
@@ -217,94 +191,85 @@ export function ThemeDialog({
|
|||||||
paddingRight={1}
|
paddingRight={1}
|
||||||
width="100%"
|
width="100%"
|
||||||
>
|
>
|
||||||
<Box flexDirection="row">
|
{mode === 'theme' ? (
|
||||||
{/* Left Column: Selection */}
|
<Box flexDirection="row">
|
||||||
<Box flexDirection="column" width="45%" paddingRight={2}>
|
{/* Left Column: Selection */}
|
||||||
<Text bold={currentFocusedSection === 'theme'} wrap="truncate">
|
<Box flexDirection="column" width="45%" paddingRight={2}>
|
||||||
{currentFocusedSection === 'theme' ? '> ' : ' '}Select Theme{' '}
|
<Text bold={mode === 'theme'} wrap="truncate">
|
||||||
<Text color={Colors.Gray}>{otherScopeModifiedMessage}</Text>
|
{mode === 'theme' ? '> ' : ' '}Select Theme{' '}
|
||||||
</Text>
|
<Text color={Colors.Gray}>{otherScopeModifiedMessage}</Text>
|
||||||
<RadioButtonSelect
|
</Text>
|
||||||
key={selectInputKey}
|
<RadioButtonSelect
|
||||||
items={themeItems}
|
items={themeItems}
|
||||||
initialIndex={safeInitialThemeIndex}
|
initialIndex={safeInitialThemeIndex}
|
||||||
onSelect={handleThemeSelect}
|
onSelect={handleThemeSelect}
|
||||||
onHighlight={handleThemeHighlight}
|
onHighlight={handleThemeHighlight}
|
||||||
isFocused={currentFocusedSection === 'theme'}
|
isFocused={mode === 'theme'}
|
||||||
maxItemsToShow={8}
|
maxItemsToShow={12}
|
||||||
showScrollArrows={true}
|
showScrollArrows={true}
|
||||||
showNumbers={currentFocusedSection === 'theme'}
|
showNumbers={mode === 'theme'}
|
||||||
/>
|
/>
|
||||||
|
</Box>
|
||||||
|
|
||||||
{/* Scope Selection */}
|
{/* Right Column: Preview */}
|
||||||
{showScopeSelection && (
|
<Box flexDirection="column" width="55%" paddingLeft={2}>
|
||||||
<Box marginTop={1} flexDirection="column">
|
<Text bold>Preview</Text>
|
||||||
<Text bold={currentFocusedSection === 'scope'} wrap="truncate">
|
{/* Get the Theme object for the highlighted theme, fall back to default if not found */}
|
||||||
{currentFocusedSection === 'scope' ? '> ' : ' '}Apply To
|
{(() => {
|
||||||
</Text>
|
const previewTheme =
|
||||||
<RadioButtonSelect
|
themeManager.getTheme(
|
||||||
items={scopeItems}
|
highlightedThemeName || DEFAULT_THEME.name,
|
||||||
initialIndex={0} // Default to User Settings
|
) || DEFAULT_THEME;
|
||||||
onSelect={handleScopeSelect}
|
return (
|
||||||
onHighlight={handleScopeHighlight}
|
<Box
|
||||||
isFocused={currentFocusedSection === 'scope'}
|
borderStyle="single"
|
||||||
showNumbers={currentFocusedSection === 'scope'}
|
borderColor={Colors.Gray}
|
||||||
/>
|
paddingTop={includePadding ? 1 : 0}
|
||||||
</Box>
|
paddingBottom={includePadding ? 1 : 0}
|
||||||
)}
|
paddingLeft={1}
|
||||||
</Box>
|
paddingRight={1}
|
||||||
|
flexDirection="column"
|
||||||
{/* Right Column: Preview */}
|
>
|
||||||
<Box flexDirection="column" width="55%" paddingLeft={2}>
|
{colorizeCode(
|
||||||
<Text bold>Preview</Text>
|
`# function
|
||||||
{/* Get the Theme object for the highlighted theme, fall back to default if not found */}
|
|
||||||
{(() => {
|
|
||||||
const previewTheme =
|
|
||||||
themeManager.getTheme(
|
|
||||||
highlightedThemeName || DEFAULT_THEME.name,
|
|
||||||
) || DEFAULT_THEME;
|
|
||||||
return (
|
|
||||||
<Box
|
|
||||||
borderStyle="single"
|
|
||||||
borderColor={Colors.Gray}
|
|
||||||
paddingTop={includePadding ? 1 : 0}
|
|
||||||
paddingBottom={includePadding ? 1 : 0}
|
|
||||||
paddingLeft={1}
|
|
||||||
paddingRight={1}
|
|
||||||
flexDirection="column"
|
|
||||||
>
|
|
||||||
{colorizeCode(
|
|
||||||
`# function
|
|
||||||
def fibonacci(n):
|
def fibonacci(n):
|
||||||
a, b = 0, 1
|
a, b = 0, 1
|
||||||
for _ in range(n):
|
for _ in range(n):
|
||||||
a, b = b, a + b
|
a, b = b, a + b
|
||||||
return a`,
|
return a`,
|
||||||
'python',
|
'python',
|
||||||
codeBlockHeight,
|
codeBlockHeight,
|
||||||
colorizeCodeWidth,
|
colorizeCodeWidth,
|
||||||
)}
|
)}
|
||||||
<Box marginTop={1} />
|
<Box marginTop={1} />
|
||||||
<DiffRenderer
|
<DiffRenderer
|
||||||
diffContent={`--- a/util.py
|
diffContent={`--- a/util.py
|
||||||
+++ b/util.py
|
+++ b/util.py
|
||||||
@@ -1,2 +1,2 @@
|
@@ -1,2 +1,2 @@
|
||||||
- print("Hello, " + name)
|
- print("Hello, " + name)
|
||||||
+ print(f"Hello, {name}!")
|
+ print(f"Hello, {name}!")
|
||||||
`}
|
`}
|
||||||
availableTerminalHeight={diffHeight}
|
availableTerminalHeight={diffHeight}
|
||||||
terminalWidth={colorizeCodeWidth}
|
terminalWidth={colorizeCodeWidth}
|
||||||
theme={previewTheme}
|
theme={previewTheme}
|
||||||
/>
|
/>
|
||||||
</Box>
|
</Box>
|
||||||
);
|
);
|
||||||
})()}
|
})()}
|
||||||
|
</Box>
|
||||||
</Box>
|
</Box>
|
||||||
</Box>
|
) : (
|
||||||
|
<ScopeSelector
|
||||||
|
onSelect={handleScopeSelect}
|
||||||
|
onHighlight={handleScopeHighlight}
|
||||||
|
isFocused={mode === 'scope'}
|
||||||
|
initialScope={selectedScope}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
<Box marginTop={1}>
|
<Box marginTop={1}>
|
||||||
<Text color={Colors.Gray} wrap="truncate">
|
<Text color={Colors.Gray} wrap="truncate">
|
||||||
(Use Enter to select
|
(Use Enter to {mode === 'theme' ? 'select' : 'apply scope'}, Tab to{' '}
|
||||||
{showScopeSelection ? ', Tab to change focus' : ''})
|
{mode === 'theme' ? 'configure scope' : 'select theme'})
|
||||||
</Text>
|
</Text>
|
||||||
</Box>
|
</Box>
|
||||||
</Box>
|
</Box>
|
||||||
|
|||||||
@@ -0,0 +1,38 @@
|
|||||||
|
// Vitest Snapshot v1, https://vitest.dev/guide/snapshot.html
|
||||||
|
|
||||||
|
exports[`ThemeDialog Snapshots > should render correctly in scope selector mode 1`] = `
|
||||||
|
"╭──────────────────────────────────────────────────────────────────────────────────────────────────╮
|
||||||
|
│ │
|
||||||
|
│ > Apply To │
|
||||||
|
│ ● 1. User Settings │
|
||||||
|
│ 2. Workspace Settings │
|
||||||
|
│ 3. System Settings │
|
||||||
|
│ │
|
||||||
|
│ (Use Enter to apply scope, Tab to select theme) │
|
||||||
|
│ │
|
||||||
|
╰──────────────────────────────────────────────────────────────────────────────────────────────────╯"
|
||||||
|
`;
|
||||||
|
|
||||||
|
exports[`ThemeDialog Snapshots > should render correctly in theme selection mode 1`] = `
|
||||||
|
"╭──────────────────────────────────────────────────────────────────────────────────────────────────╮
|
||||||
|
│ │
|
||||||
|
│ > Select Theme Preview │
|
||||||
|
│ ▲ ┌─────────────────────────────────────────────────┐ │
|
||||||
|
│ 1. ANSI Dark │ │ │
|
||||||
|
│ 2. Atom One Dark │ 1 # function │ │
|
||||||
|
│ 3. Ayu Dark │ 2 def fibonacci(n): │ │
|
||||||
|
│ ● 4. Default Dark │ 3 a, b = 0, 1 │ │
|
||||||
|
│ 5. Dracula Dark │ 4 for _ in range(n): │ │
|
||||||
|
│ 6. GitHub Dark │ 5 a, b = b, a + b │ │
|
||||||
|
│ 7. Shades Of Purple Dark │ 6 return a │ │
|
||||||
|
│ 8. ANSI Light Light │ │ │
|
||||||
|
│ 9. Ayu Light Light │ 1 - print("Hello, " + name) │ │
|
||||||
|
│ 10. Default Light Light │ 1 + print(f"Hello, {name}!") │ │
|
||||||
|
│ 11. GitHub Light Light │ │ │
|
||||||
|
│ 12. Google Code Light └─────────────────────────────────────────────────┘ │
|
||||||
|
│ ▼ │
|
||||||
|
│ │
|
||||||
|
│ (Use Enter to select, Tab to configure scope) │
|
||||||
|
│ │
|
||||||
|
╰──────────────────────────────────────────────────────────────────────────────────────────────────╯"
|
||||||
|
`;
|
||||||
@@ -0,0 +1,52 @@
|
|||||||
|
/**
|
||||||
|
* @license
|
||||||
|
* Copyright 2025 Google LLC
|
||||||
|
* SPDX-License-Identifier: Apache-2.0
|
||||||
|
*/
|
||||||
|
|
||||||
|
import type React from 'react';
|
||||||
|
import { Box, Text } from 'ink';
|
||||||
|
import type { SettingScope } from '../../../config/settings.js';
|
||||||
|
import { getScopeItems } from '../../../utils/dialogScopeUtils.js';
|
||||||
|
import { RadioButtonSelect } from './RadioButtonSelect.js';
|
||||||
|
|
||||||
|
interface ScopeSelectorProps {
|
||||||
|
/** Callback function when a scope is selected */
|
||||||
|
onSelect: (scope: SettingScope) => void;
|
||||||
|
/** Callback function when a scope is highlighted */
|
||||||
|
onHighlight: (scope: SettingScope) => void;
|
||||||
|
/** Whether the component is focused */
|
||||||
|
isFocused: boolean;
|
||||||
|
/** The initial scope to select */
|
||||||
|
initialScope: SettingScope;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function ScopeSelector({
|
||||||
|
onSelect,
|
||||||
|
onHighlight,
|
||||||
|
isFocused,
|
||||||
|
initialScope,
|
||||||
|
}: ScopeSelectorProps): React.JSX.Element {
|
||||||
|
const scopeItems = getScopeItems();
|
||||||
|
|
||||||
|
const initialIndex = scopeItems.findIndex(
|
||||||
|
(item) => item.value === initialScope,
|
||||||
|
);
|
||||||
|
const safeInitialIndex = initialIndex >= 0 ? initialIndex : 0;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Box flexDirection="column">
|
||||||
|
<Text bold={isFocused} wrap="truncate">
|
||||||
|
{isFocused ? '> ' : ' '}Apply To
|
||||||
|
</Text>
|
||||||
|
<RadioButtonSelect
|
||||||
|
items={scopeItems}
|
||||||
|
initialIndex={safeInitialIndex}
|
||||||
|
onSelect={onSelect}
|
||||||
|
onHighlight={onHighlight}
|
||||||
|
isFocused={isFocused}
|
||||||
|
showNumbers={isFocused}
|
||||||
|
/>
|
||||||
|
</Box>
|
||||||
|
);
|
||||||
|
}
|
||||||
Reference in New Issue
Block a user