@google/gemini-cli 0.13.0-preview.3 → 0.14.0-preview.0
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 +2 -2
- package/dist/src/config/config.js +1 -0
- package/dist/src/config/config.js.map +1 -1
- package/dist/src/config/extension-manager.js +19 -7
- package/dist/src/config/extension-manager.js.map +1 -1
- package/dist/src/config/extensions/extensionSettings.js +3 -2
- package/dist/src/config/extensions/extensionSettings.js.map +1 -1
- package/dist/src/config/policy-engine.integration.test.js +44 -44
- package/dist/src/config/policy-engine.integration.test.js.map +1 -1
- package/dist/src/config/settings.d.ts +15 -5
- package/dist/src/config/settings.js +26 -7
- package/dist/src/config/settings.js.map +1 -1
- package/dist/src/config/settingsSchema.d.ts +29 -0
- package/dist/src/config/settingsSchema.js +30 -1
- package/dist/src/config/settingsSchema.js.map +1 -1
- package/dist/src/gemini.js +2 -1
- package/dist/src/gemini.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/nonInteractiveCli.test.js +1 -1
- package/dist/src/nonInteractiveCli.test.js.map +1 -1
- package/dist/src/services/BuiltinCommandLoader.js +1 -1
- package/dist/src/services/BuiltinCommandLoader.js.map +1 -1
- package/dist/src/services/BuiltinCommandLoader.test.js +3 -1
- package/dist/src/services/BuiltinCommandLoader.test.js.map +1 -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.test.js +9 -1
- package/dist/src/ui/App.test.js.map +1 -1
- package/dist/src/ui/AppContainer.js +1 -2
- package/dist/src/ui/AppContainer.js.map +1 -1
- package/dist/src/ui/AppContainer.test.js +2 -2
- package/dist/src/ui/AppContainer.test.js.map +1 -1
- package/dist/src/ui/auth/AuthDialog.js.map +1 -1
- package/dist/src/ui/commands/extensionsCommand.d.ts +7 -2
- package/dist/src/ui/commands/extensionsCommand.js +150 -25
- package/dist/src/ui/commands/extensionsCommand.js.map +1 -1
- package/dist/src/ui/commands/extensionsCommand.test.js +198 -81
- package/dist/src/ui/commands/extensionsCommand.test.js.map +1 -1
- package/dist/src/ui/commands/types.d.ts +1 -1
- package/dist/src/ui/components/CliSpinner.d.ts +0 -1
- package/dist/src/ui/components/CliSpinner.js +3 -4
- package/dist/src/ui/components/CliSpinner.js.map +1 -1
- package/dist/src/ui/components/CliSpinner.test.d.ts +6 -0
- package/dist/src/ui/components/CliSpinner.test.js +23 -0
- package/dist/src/ui/components/CliSpinner.test.js.map +1 -0
- package/dist/src/ui/components/Composer.js +1 -1
- package/dist/src/ui/components/Composer.js.map +1 -1
- package/dist/src/ui/components/DebugProfiler.js +2 -2
- package/dist/src/ui/components/DebugProfiler.js.map +1 -1
- package/dist/src/ui/components/DebugProfiler.test.js +15 -0
- package/dist/src/ui/components/DebugProfiler.test.js.map +1 -1
- package/dist/src/ui/components/DetailedMessagesDisplay.d.ts +1 -0
- package/dist/src/ui/components/DetailedMessagesDisplay.js +47 -27
- package/dist/src/ui/components/DetailedMessagesDisplay.js.map +1 -1
- package/dist/src/ui/components/DialogManager.js +1 -1
- package/dist/src/ui/components/DialogManager.js.map +1 -1
- package/dist/src/ui/components/EditorSettingsDialog.d.ts +2 -3
- package/dist/src/ui/components/EditorSettingsDialog.js.map +1 -1
- package/dist/src/ui/components/HistoryItemDisplay.test.js +9 -0
- package/dist/src/ui/components/HistoryItemDisplay.test.js.map +1 -1
- package/dist/src/ui/components/InputPrompt.test.js +53 -90
- package/dist/src/ui/components/InputPrompt.test.js.map +1 -1
- package/dist/src/ui/components/ProQuotaDialog.d.ts +2 -3
- package/dist/src/ui/components/ProQuotaDialog.js +9 -9
- package/dist/src/ui/components/ProQuotaDialog.js.map +1 -1
- package/dist/src/ui/components/ProQuotaDialog.test.js +12 -12
- package/dist/src/ui/components/ProQuotaDialog.test.js.map +1 -1
- package/dist/src/ui/components/SettingsDialog.js.map +1 -1
- package/dist/src/ui/components/ThemeDialog.d.ts +2 -3
- package/dist/src/ui/components/ThemeDialog.js.map +1 -1
- package/dist/src/ui/components/messages/InfoMessage.js +1 -1
- package/dist/src/ui/components/messages/InfoMessage.js.map +1 -1
- package/dist/src/ui/components/shared/ScopeSelector.d.ts +4 -4
- package/dist/src/ui/components/shared/Scrollable.d.ts +18 -0
- package/dist/src/ui/components/shared/Scrollable.js +79 -0
- package/dist/src/ui/components/shared/Scrollable.js.map +1 -0
- package/dist/src/ui/components/shared/Scrollable.test.d.ts +6 -0
- package/dist/src/ui/components/shared/Scrollable.test.js +36 -0
- package/dist/src/ui/components/shared/Scrollable.test.js.map +1 -0
- package/dist/src/ui/components/shared/ScrollableList.d.ts +26 -0
- package/dist/src/ui/components/shared/ScrollableList.js +62 -0
- package/dist/src/ui/components/shared/ScrollableList.js.map +1 -0
- package/dist/src/ui/components/shared/ScrollableList.test.d.ts +6 -0
- package/dist/src/ui/components/shared/ScrollableList.test.js +125 -0
- package/dist/src/ui/components/shared/ScrollableList.test.js.map +1 -0
- package/dist/src/ui/components/shared/VirtualizedList.d.ts +44 -0
- package/dist/src/ui/components/shared/VirtualizedList.js +305 -0
- package/dist/src/ui/components/shared/VirtualizedList.js.map +1 -0
- package/dist/src/ui/components/shared/VirtualizedList.test.d.ts +6 -0
- package/dist/src/ui/components/shared/VirtualizedList.test.js +148 -0
- package/dist/src/ui/components/shared/VirtualizedList.test.js.map +1 -0
- package/dist/src/ui/contexts/ScrollProvider.d.ts +24 -0
- package/dist/src/ui/contexts/ScrollProvider.js +100 -0
- package/dist/src/ui/contexts/ScrollProvider.js.map +1 -0
- package/dist/src/ui/contexts/UIActionsContext.d.ts +5 -5
- package/dist/src/ui/contexts/UIActionsContext.js.map +1 -1
- package/dist/src/ui/debug.d.ts +8 -0
- package/dist/src/ui/debug.js +11 -0
- package/dist/src/ui/debug.js.map +1 -0
- package/dist/src/ui/hooks/atCommandProcessor.js +1 -1
- package/dist/src/ui/hooks/atCommandProcessor.js.map +1 -1
- package/dist/src/ui/hooks/slashCommandProcessor.js +14 -0
- package/dist/src/ui/hooks/slashCommandProcessor.js.map +1 -1
- package/dist/src/ui/hooks/slashCommandProcessor.test.js +18 -0
- package/dist/src/ui/hooks/slashCommandProcessor.test.js.map +1 -1
- package/dist/src/ui/hooks/useAnimatedScrollbar.d.ts +10 -0
- package/dist/src/ui/hooks/useAnimatedScrollbar.js +88 -0
- package/dist/src/ui/hooks/useAnimatedScrollbar.js.map +1 -0
- package/dist/src/ui/hooks/useAnimatedScrollbar.test.d.ts +6 -0
- package/dist/src/ui/hooks/useAnimatedScrollbar.test.js +63 -0
- package/dist/src/ui/hooks/useAnimatedScrollbar.test.js.map +1 -0
- package/dist/src/ui/hooks/useEditorSettings.d.ts +2 -2
- package/dist/src/ui/hooks/useEditorSettings.js.map +1 -1
- package/dist/src/ui/hooks/useEditorSettings.test.js +4 -1
- package/dist/src/ui/hooks/useEditorSettings.test.js.map +1 -1
- package/dist/src/ui/hooks/useGeminiStream.js +2 -2
- package/dist/src/ui/hooks/useGeminiStream.js.map +1 -1
- package/dist/src/ui/hooks/useQuotaAndFallback.d.ts +3 -5
- package/dist/src/ui/hooks/useQuotaAndFallback.js +35 -63
- package/dist/src/ui/hooks/useQuotaAndFallback.js.map +1 -1
- package/dist/src/ui/hooks/useQuotaAndFallback.test.js +118 -87
- package/dist/src/ui/hooks/useQuotaAndFallback.test.js.map +1 -1
- package/dist/src/ui/hooks/useThemeCommand.d.ts +2 -2
- package/dist/src/ui/hooks/useThemeCommand.js.map +1 -1
- package/dist/src/ui/layouts/DefaultAppLayout.js +3 -1
- package/dist/src/ui/layouts/DefaultAppLayout.js.map +1 -1
- package/dist/src/utils/dialogScopeUtils.d.ts +6 -13
- package/dist/src/utils/dialogScopeUtils.js +4 -2
- package/dist/src/utils/dialogScopeUtils.js.map +1 -1
- package/dist/src/utils/settingsUtils.d.ts +2 -2
- package/dist/src/utils/settingsUtils.js.map +1 -1
- package/dist/src/validateNonInterActiveAuth.test.js +1 -1
- package/dist/src/validateNonInterActiveAuth.test.js.map +1 -1
- package/dist/tsconfig.tsbuildinfo +1 -1
- package/package.json +3 -3
- package/dist/google-gemini-cli-0.13.0-preview.2.tgz +0 -0
- package/dist/src/config/policies/read-only.toml +0 -56
- package/dist/src/config/policies/write.toml +0 -63
- package/dist/src/config/policies/yolo.toml +0 -31
|
@@ -0,0 +1,305 @@
|
|
|
1
|
+
import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
|
|
2
|
+
/**
|
|
3
|
+
* @license
|
|
4
|
+
* Copyright 2025 Google LLC
|
|
5
|
+
* SPDX-License-Identifier: Apache-2.0
|
|
6
|
+
*/
|
|
7
|
+
import { useState, useRef, useLayoutEffect, forwardRef, useImperativeHandle, useEffect, useMemo, useCallback, } from 'react';
|
|
8
|
+
import { theme } from '../../semantic-colors.js';
|
|
9
|
+
import { measureElement, Box } from 'ink';
|
|
10
|
+
export const SCROLL_TO_ITEM_END = Number.MAX_SAFE_INTEGER;
|
|
11
|
+
function findLastIndex(array, predicate) {
|
|
12
|
+
for (let i = array.length - 1; i >= 0; i--) {
|
|
13
|
+
if (predicate(array[i], i, array)) {
|
|
14
|
+
return i;
|
|
15
|
+
}
|
|
16
|
+
}
|
|
17
|
+
return -1;
|
|
18
|
+
}
|
|
19
|
+
function VirtualizedList(props, ref) {
|
|
20
|
+
const { data, renderItem, estimatedItemHeight, keyExtractor, initialScrollIndex, initialScrollOffsetInIndex, } = props;
|
|
21
|
+
const dataRef = useRef(data);
|
|
22
|
+
useEffect(() => {
|
|
23
|
+
dataRef.current = data;
|
|
24
|
+
}, [data]);
|
|
25
|
+
const [scrollAnchor, setScrollAnchor] = useState(() => {
|
|
26
|
+
const scrollToEnd = initialScrollIndex === SCROLL_TO_ITEM_END ||
|
|
27
|
+
(typeof initialScrollIndex === 'number' &&
|
|
28
|
+
initialScrollIndex >= data.length - 1 &&
|
|
29
|
+
initialScrollOffsetInIndex === SCROLL_TO_ITEM_END);
|
|
30
|
+
if (scrollToEnd) {
|
|
31
|
+
return {
|
|
32
|
+
index: data.length > 0 ? data.length - 1 : 0,
|
|
33
|
+
offset: SCROLL_TO_ITEM_END,
|
|
34
|
+
};
|
|
35
|
+
}
|
|
36
|
+
if (typeof initialScrollIndex === 'number') {
|
|
37
|
+
return {
|
|
38
|
+
index: Math.max(0, Math.min(data.length - 1, initialScrollIndex)),
|
|
39
|
+
offset: initialScrollOffsetInIndex ?? 0,
|
|
40
|
+
};
|
|
41
|
+
}
|
|
42
|
+
return { index: 0, offset: 0 };
|
|
43
|
+
});
|
|
44
|
+
const [isStickingToBottom, setIsStickingToBottom] = useState(() => {
|
|
45
|
+
const scrollToEnd = initialScrollIndex === SCROLL_TO_ITEM_END ||
|
|
46
|
+
(typeof initialScrollIndex === 'number' &&
|
|
47
|
+
initialScrollIndex >= data.length - 1 &&
|
|
48
|
+
initialScrollOffsetInIndex === SCROLL_TO_ITEM_END);
|
|
49
|
+
return scrollToEnd;
|
|
50
|
+
});
|
|
51
|
+
const containerRef = useRef(null);
|
|
52
|
+
const [containerHeight, setContainerHeight] = useState(0);
|
|
53
|
+
const itemRefs = useRef([]);
|
|
54
|
+
const [heights, setHeights] = useState([]);
|
|
55
|
+
const isInitialScrollSet = useRef(false);
|
|
56
|
+
const { totalHeight, offsets } = useMemo(() => {
|
|
57
|
+
const offsets = [0];
|
|
58
|
+
let totalHeight = 0;
|
|
59
|
+
for (let i = 0; i < data.length; i++) {
|
|
60
|
+
const height = heights[i] ?? estimatedItemHeight(i);
|
|
61
|
+
totalHeight += height;
|
|
62
|
+
offsets.push(totalHeight);
|
|
63
|
+
}
|
|
64
|
+
return { totalHeight, offsets };
|
|
65
|
+
}, [heights, data, estimatedItemHeight]);
|
|
66
|
+
useEffect(() => {
|
|
67
|
+
setHeights((prevHeights) => {
|
|
68
|
+
if (data.length === prevHeights.length) {
|
|
69
|
+
return prevHeights;
|
|
70
|
+
}
|
|
71
|
+
const newHeights = [...prevHeights];
|
|
72
|
+
if (data.length < prevHeights.length) {
|
|
73
|
+
newHeights.length = data.length;
|
|
74
|
+
}
|
|
75
|
+
else {
|
|
76
|
+
for (let i = prevHeights.length; i < data.length; i++) {
|
|
77
|
+
newHeights[i] = estimatedItemHeight(i);
|
|
78
|
+
}
|
|
79
|
+
}
|
|
80
|
+
return newHeights;
|
|
81
|
+
});
|
|
82
|
+
}, [data, estimatedItemHeight]);
|
|
83
|
+
// This layout effect needs to run on every render to correctly measure the
|
|
84
|
+
// container and ensure we recompute the layout if it has changed.
|
|
85
|
+
// eslint-disable-next-line react-hooks/exhaustive-deps
|
|
86
|
+
useLayoutEffect(() => {
|
|
87
|
+
if (containerRef.current) {
|
|
88
|
+
const height = Math.round(measureElement(containerRef.current).height);
|
|
89
|
+
if (containerHeight !== height) {
|
|
90
|
+
setContainerHeight(height);
|
|
91
|
+
}
|
|
92
|
+
}
|
|
93
|
+
let newHeights = null;
|
|
94
|
+
for (let i = startIndex; i <= endIndex; i++) {
|
|
95
|
+
const itemRef = itemRefs.current[i];
|
|
96
|
+
if (itemRef) {
|
|
97
|
+
const height = Math.round(measureElement(itemRef).height);
|
|
98
|
+
if (height !== heights[i]) {
|
|
99
|
+
if (!newHeights) {
|
|
100
|
+
newHeights = [...heights];
|
|
101
|
+
}
|
|
102
|
+
newHeights[i] = height;
|
|
103
|
+
}
|
|
104
|
+
}
|
|
105
|
+
}
|
|
106
|
+
if (newHeights) {
|
|
107
|
+
setHeights(newHeights);
|
|
108
|
+
}
|
|
109
|
+
});
|
|
110
|
+
const scrollableContainerHeight = containerRef.current
|
|
111
|
+
? Math.round(measureElement(containerRef.current).height)
|
|
112
|
+
: containerHeight;
|
|
113
|
+
const getAnchorForScrollTop = useCallback((scrollTop, offsets) => {
|
|
114
|
+
const index = findLastIndex(offsets, (offset) => offset <= scrollTop);
|
|
115
|
+
if (index === -1) {
|
|
116
|
+
return { index: 0, offset: 0 };
|
|
117
|
+
}
|
|
118
|
+
return { index, offset: scrollTop - offsets[index] };
|
|
119
|
+
}, []);
|
|
120
|
+
const scrollTop = useMemo(() => {
|
|
121
|
+
const offset = offsets[scrollAnchor.index];
|
|
122
|
+
if (typeof offset !== 'number') {
|
|
123
|
+
return 0;
|
|
124
|
+
}
|
|
125
|
+
if (scrollAnchor.offset === SCROLL_TO_ITEM_END) {
|
|
126
|
+
const itemHeight = heights[scrollAnchor.index] ?? 0;
|
|
127
|
+
return offset + itemHeight - scrollableContainerHeight;
|
|
128
|
+
}
|
|
129
|
+
return offset + scrollAnchor.offset;
|
|
130
|
+
}, [scrollAnchor, offsets, heights, scrollableContainerHeight]);
|
|
131
|
+
const prevDataLength = useRef(data.length);
|
|
132
|
+
const prevTotalHeight = useRef(totalHeight);
|
|
133
|
+
const prevScrollTop = useRef(scrollTop);
|
|
134
|
+
const prevContainerHeight = useRef(scrollableContainerHeight);
|
|
135
|
+
useLayoutEffect(() => {
|
|
136
|
+
const contentPreviouslyFit = prevTotalHeight.current <= prevContainerHeight.current;
|
|
137
|
+
const wasScrolledToBottomPixels = prevScrollTop.current >=
|
|
138
|
+
prevTotalHeight.current - prevContainerHeight.current - 1;
|
|
139
|
+
const wasAtBottom = contentPreviouslyFit || wasScrolledToBottomPixels;
|
|
140
|
+
// If the user was at the bottom, they are now sticking. This handles
|
|
141
|
+
// manually scrolling back to the bottom.
|
|
142
|
+
if (wasAtBottom && scrollTop >= prevScrollTop.current) {
|
|
143
|
+
setIsStickingToBottom(true);
|
|
144
|
+
}
|
|
145
|
+
const listGrew = data.length > prevDataLength.current;
|
|
146
|
+
const containerChanged = prevContainerHeight.current !== scrollableContainerHeight;
|
|
147
|
+
// We scroll to the end if:
|
|
148
|
+
// 1. The list grew AND we were already at the bottom (or sticking).
|
|
149
|
+
// 2. We are sticking to the bottom AND the container size changed.
|
|
150
|
+
if ((listGrew && (isStickingToBottom || wasAtBottom)) ||
|
|
151
|
+
(isStickingToBottom && containerChanged)) {
|
|
152
|
+
setScrollAnchor({
|
|
153
|
+
index: data.length > 0 ? data.length - 1 : 0,
|
|
154
|
+
offset: SCROLL_TO_ITEM_END,
|
|
155
|
+
});
|
|
156
|
+
// If we are scrolling to the bottom, we are by definition sticking.
|
|
157
|
+
if (!isStickingToBottom) {
|
|
158
|
+
setIsStickingToBottom(true);
|
|
159
|
+
}
|
|
160
|
+
}
|
|
161
|
+
// Scenario 2: The list has changed (shrunk) in a way that our
|
|
162
|
+
// current scroll position or anchor is invalid. We should adjust to the bottom.
|
|
163
|
+
else if ((scrollAnchor.index >= data.length ||
|
|
164
|
+
scrollTop > totalHeight - scrollableContainerHeight) &&
|
|
165
|
+
data.length > 0) {
|
|
166
|
+
const newScrollTop = Math.max(0, totalHeight - scrollableContainerHeight);
|
|
167
|
+
setScrollAnchor(getAnchorForScrollTop(newScrollTop, offsets));
|
|
168
|
+
}
|
|
169
|
+
else if (data.length === 0) {
|
|
170
|
+
// List is now empty, reset scroll to top.
|
|
171
|
+
setScrollAnchor({ index: 0, offset: 0 });
|
|
172
|
+
}
|
|
173
|
+
// Update refs for the next render cycle.
|
|
174
|
+
prevDataLength.current = data.length;
|
|
175
|
+
prevTotalHeight.current = totalHeight;
|
|
176
|
+
prevScrollTop.current = scrollTop;
|
|
177
|
+
prevContainerHeight.current = scrollableContainerHeight;
|
|
178
|
+
}, [
|
|
179
|
+
data.length,
|
|
180
|
+
totalHeight,
|
|
181
|
+
scrollTop,
|
|
182
|
+
scrollableContainerHeight,
|
|
183
|
+
scrollAnchor.index,
|
|
184
|
+
getAnchorForScrollTop,
|
|
185
|
+
offsets,
|
|
186
|
+
isStickingToBottom,
|
|
187
|
+
]);
|
|
188
|
+
useLayoutEffect(() => {
|
|
189
|
+
if (isInitialScrollSet.current ||
|
|
190
|
+
offsets.length <= 1 ||
|
|
191
|
+
totalHeight <= 0 ||
|
|
192
|
+
containerHeight <= 0) {
|
|
193
|
+
return;
|
|
194
|
+
}
|
|
195
|
+
if (typeof initialScrollIndex === 'number') {
|
|
196
|
+
const scrollToEnd = initialScrollIndex === SCROLL_TO_ITEM_END ||
|
|
197
|
+
(initialScrollIndex >= data.length - 1 &&
|
|
198
|
+
initialScrollOffsetInIndex === SCROLL_TO_ITEM_END);
|
|
199
|
+
if (scrollToEnd) {
|
|
200
|
+
setScrollAnchor({
|
|
201
|
+
index: data.length - 1,
|
|
202
|
+
offset: SCROLL_TO_ITEM_END,
|
|
203
|
+
});
|
|
204
|
+
setIsStickingToBottom(true);
|
|
205
|
+
isInitialScrollSet.current = true;
|
|
206
|
+
return;
|
|
207
|
+
}
|
|
208
|
+
const index = Math.max(0, Math.min(data.length - 1, initialScrollIndex));
|
|
209
|
+
const offset = initialScrollOffsetInIndex ?? 0;
|
|
210
|
+
const newScrollTop = (offsets[index] ?? 0) + offset;
|
|
211
|
+
const clampedScrollTop = Math.max(0, Math.min(totalHeight - scrollableContainerHeight, newScrollTop));
|
|
212
|
+
setScrollAnchor(getAnchorForScrollTop(clampedScrollTop, offsets));
|
|
213
|
+
isInitialScrollSet.current = true;
|
|
214
|
+
}
|
|
215
|
+
}, [
|
|
216
|
+
initialScrollIndex,
|
|
217
|
+
initialScrollOffsetInIndex,
|
|
218
|
+
offsets,
|
|
219
|
+
totalHeight,
|
|
220
|
+
containerHeight,
|
|
221
|
+
getAnchorForScrollTop,
|
|
222
|
+
data.length,
|
|
223
|
+
heights,
|
|
224
|
+
scrollableContainerHeight,
|
|
225
|
+
]);
|
|
226
|
+
const startIndex = Math.max(0, findLastIndex(offsets, (offset) => offset <= scrollTop) - 1);
|
|
227
|
+
const endIndexOffset = offsets.findIndex((offset) => offset > scrollTop + scrollableContainerHeight);
|
|
228
|
+
const endIndex = endIndexOffset === -1
|
|
229
|
+
? data.length - 1
|
|
230
|
+
: Math.min(data.length - 1, endIndexOffset);
|
|
231
|
+
const topSpacerHeight = offsets[startIndex] ?? 0;
|
|
232
|
+
const bottomSpacerHeight = totalHeight - (offsets[endIndex + 1] ?? totalHeight);
|
|
233
|
+
const renderedItems = [];
|
|
234
|
+
for (let i = startIndex; i <= endIndex; i++) {
|
|
235
|
+
const item = data[i];
|
|
236
|
+
if (item) {
|
|
237
|
+
renderedItems.push(_jsx(Box, { width: "100%", ref: (el) => {
|
|
238
|
+
itemRefs.current[i] = el;
|
|
239
|
+
}, children: renderItem({ item, index: i }) }, keyExtractor(item, i)));
|
|
240
|
+
}
|
|
241
|
+
}
|
|
242
|
+
useImperativeHandle(ref, () => ({
|
|
243
|
+
scrollBy: (delta) => {
|
|
244
|
+
if (delta < 0) {
|
|
245
|
+
setIsStickingToBottom(false);
|
|
246
|
+
}
|
|
247
|
+
const currentScrollTop = scrollTop;
|
|
248
|
+
const newScrollTop = Math.max(0, Math.min(totalHeight - scrollableContainerHeight, currentScrollTop + delta));
|
|
249
|
+
setScrollAnchor(getAnchorForScrollTop(newScrollTop, offsets));
|
|
250
|
+
},
|
|
251
|
+
scrollTo: (offset) => {
|
|
252
|
+
setIsStickingToBottom(false);
|
|
253
|
+
const newScrollTop = Math.max(0, Math.min(totalHeight - scrollableContainerHeight, offset));
|
|
254
|
+
setScrollAnchor(getAnchorForScrollTop(newScrollTop, offsets));
|
|
255
|
+
},
|
|
256
|
+
scrollToEnd: () => {
|
|
257
|
+
setIsStickingToBottom(true);
|
|
258
|
+
if (data.length > 0) {
|
|
259
|
+
setScrollAnchor({
|
|
260
|
+
index: data.length - 1,
|
|
261
|
+
offset: SCROLL_TO_ITEM_END,
|
|
262
|
+
});
|
|
263
|
+
}
|
|
264
|
+
},
|
|
265
|
+
scrollToIndex: ({ index, viewOffset = 0, viewPosition = 0, }) => {
|
|
266
|
+
setIsStickingToBottom(false);
|
|
267
|
+
const offset = offsets[index];
|
|
268
|
+
if (offset !== undefined) {
|
|
269
|
+
const newScrollTop = Math.max(0, Math.min(totalHeight - scrollableContainerHeight, offset - viewPosition * scrollableContainerHeight + viewOffset));
|
|
270
|
+
setScrollAnchor(getAnchorForScrollTop(newScrollTop, offsets));
|
|
271
|
+
}
|
|
272
|
+
},
|
|
273
|
+
scrollToItem: ({ item, viewOffset = 0, viewPosition = 0, }) => {
|
|
274
|
+
setIsStickingToBottom(false);
|
|
275
|
+
const index = data.indexOf(item);
|
|
276
|
+
if (index !== -1) {
|
|
277
|
+
const offset = offsets[index];
|
|
278
|
+
if (offset !== undefined) {
|
|
279
|
+
const newScrollTop = Math.max(0, Math.min(totalHeight - scrollableContainerHeight, offset - viewPosition * scrollableContainerHeight + viewOffset));
|
|
280
|
+
setScrollAnchor(getAnchorForScrollTop(newScrollTop, offsets));
|
|
281
|
+
}
|
|
282
|
+
}
|
|
283
|
+
},
|
|
284
|
+
getScrollIndex: () => scrollAnchor.index,
|
|
285
|
+
getScrollState: () => ({
|
|
286
|
+
scrollTop,
|
|
287
|
+
scrollHeight: totalHeight,
|
|
288
|
+
innerHeight: containerHeight,
|
|
289
|
+
}),
|
|
290
|
+
}), [
|
|
291
|
+
offsets,
|
|
292
|
+
scrollAnchor,
|
|
293
|
+
totalHeight,
|
|
294
|
+
getAnchorForScrollTop,
|
|
295
|
+
data,
|
|
296
|
+
scrollableContainerHeight,
|
|
297
|
+
scrollTop,
|
|
298
|
+
containerHeight,
|
|
299
|
+
]);
|
|
300
|
+
return (_jsx(Box, { ref: containerRef, overflowY: "scroll", overflowX: "hidden", scrollTop: scrollTop, scrollbarThumbColor: props.scrollbarThumbColor ?? theme.text.secondary, width: "100%", height: "100%", flexDirection: "column", children: _jsxs(Box, { flexShrink: 0, width: "100%", flexDirection: "column", children: [_jsx(Box, { height: topSpacerHeight, flexShrink: 0 }), renderedItems, _jsx(Box, { height: bottomSpacerHeight, flexShrink: 0 })] }) }));
|
|
301
|
+
}
|
|
302
|
+
const VirtualizedListWithForwardRef = forwardRef(VirtualizedList);
|
|
303
|
+
export { VirtualizedListWithForwardRef as VirtualizedList };
|
|
304
|
+
VirtualizedList.displayName = 'VirtualizedList';
|
|
305
|
+
//# sourceMappingURL=VirtualizedList.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"VirtualizedList.js","sourceRoot":"","sources":["../../../../../src/ui/components/shared/VirtualizedList.tsx"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,148 @@
|
|
|
1
|
+
import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
|
|
2
|
+
/**
|
|
3
|
+
* @license
|
|
4
|
+
* Copyright 2025 Google LLC
|
|
5
|
+
* SPDX-License-Identifier: Apache-2.0
|
|
6
|
+
*/
|
|
7
|
+
import { render } from '../../../test-utils/render.js';
|
|
8
|
+
import { VirtualizedList } from './VirtualizedList.js';
|
|
9
|
+
import { Text, Box } from 'ink';
|
|
10
|
+
import { createRef, act, useEffect, createContext, useContext, useState, } from 'react';
|
|
11
|
+
import { describe, it, expect, vi, beforeEach } from 'vitest';
|
|
12
|
+
const delay = (ms) => new Promise((resolve) => setTimeout(resolve, ms));
|
|
13
|
+
describe('<VirtualizedList />', () => {
|
|
14
|
+
const keyExtractor = (item) => item;
|
|
15
|
+
beforeEach(() => {
|
|
16
|
+
vi.clearAllMocks();
|
|
17
|
+
});
|
|
18
|
+
describe('with 10px height and 100 items', () => {
|
|
19
|
+
const longData = Array.from({ length: 100 }, (_, i) => `Item ${i}`);
|
|
20
|
+
// We use 1px for items. Container is 10px.
|
|
21
|
+
// Viewport shows 10 items. Overscan adds 10 items.
|
|
22
|
+
const itemHeight = 1;
|
|
23
|
+
const renderItem1px = ({ item }) => (_jsx(Box, { height: itemHeight, children: _jsx(Text, { children: item }) }));
|
|
24
|
+
it.each([
|
|
25
|
+
{
|
|
26
|
+
name: 'top',
|
|
27
|
+
initialScrollIndex: undefined,
|
|
28
|
+
visible: ['Item 0', 'Item 7'],
|
|
29
|
+
notVisible: ['Item 8', 'Item 15', 'Item 50', 'Item 99'],
|
|
30
|
+
},
|
|
31
|
+
{
|
|
32
|
+
name: 'scrolled to bottom',
|
|
33
|
+
initialScrollIndex: 99,
|
|
34
|
+
visible: ['Item 99', 'Item 92'],
|
|
35
|
+
notVisible: ['Item 91', 'Item 85', 'Item 50', 'Item 0'],
|
|
36
|
+
},
|
|
37
|
+
])('renders only visible items ($name)', async ({ initialScrollIndex, visible, notVisible }) => {
|
|
38
|
+
const { lastFrame } = render(_jsx(Box, { height: 10, width: 100, borderStyle: "round", children: _jsx(VirtualizedList, { data: longData, renderItem: renderItem1px, keyExtractor: keyExtractor, estimatedItemHeight: () => itemHeight, initialScrollIndex: initialScrollIndex }) }));
|
|
39
|
+
await act(async () => {
|
|
40
|
+
await delay(0);
|
|
41
|
+
});
|
|
42
|
+
const frame = lastFrame();
|
|
43
|
+
visible.forEach((item) => {
|
|
44
|
+
expect(frame).toContain(item);
|
|
45
|
+
});
|
|
46
|
+
notVisible.forEach((item) => {
|
|
47
|
+
expect(frame).not.toContain(item);
|
|
48
|
+
});
|
|
49
|
+
expect(frame).toMatchSnapshot();
|
|
50
|
+
});
|
|
51
|
+
it('sticks to bottom when new items added', async () => {
|
|
52
|
+
const { lastFrame, rerender } = render(_jsx(Box, { height: 10, width: 100, borderStyle: "round", children: _jsx(VirtualizedList, { data: longData, renderItem: renderItem1px, keyExtractor: keyExtractor, estimatedItemHeight: () => itemHeight, initialScrollIndex: 99 }) }));
|
|
53
|
+
await act(async () => {
|
|
54
|
+
await delay(0);
|
|
55
|
+
});
|
|
56
|
+
expect(lastFrame()).toContain('Item 99');
|
|
57
|
+
// Add items
|
|
58
|
+
const newData = [...longData, 'Item 100', 'Item 101'];
|
|
59
|
+
rerender(_jsx(Box, { height: 10, width: 100, borderStyle: "round", children: _jsx(VirtualizedList, { data: newData, renderItem: renderItem1px, keyExtractor: keyExtractor, estimatedItemHeight: () => itemHeight }) }));
|
|
60
|
+
await act(async () => {
|
|
61
|
+
await delay(0);
|
|
62
|
+
});
|
|
63
|
+
const frame = lastFrame();
|
|
64
|
+
expect(frame).toContain('Item 101');
|
|
65
|
+
expect(frame).not.toContain('Item 0');
|
|
66
|
+
});
|
|
67
|
+
it('scrolls down to show new items when requested via ref', async () => {
|
|
68
|
+
const ref = createRef();
|
|
69
|
+
const { lastFrame } = render(_jsx(Box, { height: 10, width: 100, borderStyle: "round", children: _jsx(VirtualizedList, { ref: ref, data: longData, renderItem: renderItem1px, keyExtractor: keyExtractor, estimatedItemHeight: () => itemHeight }) }));
|
|
70
|
+
await act(async () => {
|
|
71
|
+
await delay(0);
|
|
72
|
+
});
|
|
73
|
+
expect(lastFrame()).toContain('Item 0');
|
|
74
|
+
// Scroll to bottom via ref
|
|
75
|
+
await act(async () => {
|
|
76
|
+
ref.current?.scrollToEnd();
|
|
77
|
+
await delay(0);
|
|
78
|
+
});
|
|
79
|
+
const frame = lastFrame();
|
|
80
|
+
expect(frame).toContain('Item 99');
|
|
81
|
+
});
|
|
82
|
+
it.each([
|
|
83
|
+
{ initialScrollIndex: 0, expectedMountedCount: 5 },
|
|
84
|
+
{ initialScrollIndex: 500, expectedMountedCount: 6 },
|
|
85
|
+
{ initialScrollIndex: 999, expectedMountedCount: 5 },
|
|
86
|
+
])('mounts only visible items with 1000 items and 10px height (scroll: $initialScrollIndex)', async ({ initialScrollIndex, expectedMountedCount }) => {
|
|
87
|
+
let mountedCount = 0;
|
|
88
|
+
const tallItemHeight = 5;
|
|
89
|
+
const ItemWithEffect = ({ item }) => {
|
|
90
|
+
useEffect(() => {
|
|
91
|
+
mountedCount++;
|
|
92
|
+
return () => {
|
|
93
|
+
mountedCount--;
|
|
94
|
+
};
|
|
95
|
+
}, []);
|
|
96
|
+
return (_jsx(Box, { height: tallItemHeight, children: _jsx(Text, { children: item }) }));
|
|
97
|
+
};
|
|
98
|
+
const veryLongData = Array.from({ length: 1000 }, (_, i) => `Item ${i}`);
|
|
99
|
+
const { lastFrame } = render(_jsx(Box, { height: 20, width: 100, borderStyle: "round", children: _jsx(VirtualizedList, { data: veryLongData, renderItem: ({ item }) => (_jsx(ItemWithEffect, { item: item }, item)), keyExtractor: keyExtractor, estimatedItemHeight: () => tallItemHeight, initialScrollIndex: initialScrollIndex }) }));
|
|
100
|
+
await act(async () => {
|
|
101
|
+
await delay(0);
|
|
102
|
+
});
|
|
103
|
+
const frame = lastFrame();
|
|
104
|
+
expect(mountedCount).toBe(expectedMountedCount);
|
|
105
|
+
expect(frame).toMatchSnapshot();
|
|
106
|
+
});
|
|
107
|
+
});
|
|
108
|
+
it('renders more items when a visible item shrinks via context update', async () => {
|
|
109
|
+
const SizeContext = createContext({
|
|
110
|
+
firstItemHeight: 10,
|
|
111
|
+
setFirstItemHeight: () => { },
|
|
112
|
+
});
|
|
113
|
+
const items = Array.from({ length: 20 }, (_, i) => ({
|
|
114
|
+
id: `Item ${i}`,
|
|
115
|
+
}));
|
|
116
|
+
const ItemWithContext = ({ item, index, }) => {
|
|
117
|
+
const { firstItemHeight } = useContext(SizeContext);
|
|
118
|
+
const height = index === 0 ? firstItemHeight : 1;
|
|
119
|
+
return (_jsx(Box, { height: height, children: _jsx(Text, { children: item.id }) }));
|
|
120
|
+
};
|
|
121
|
+
const TestComponent = () => {
|
|
122
|
+
const [firstItemHeight, setFirstItemHeight] = useState(10);
|
|
123
|
+
return (_jsxs(SizeContext.Provider, { value: { firstItemHeight, setFirstItemHeight }, children: [_jsx(Box, { height: 10, width: 100, children: _jsx(VirtualizedList, { data: items, renderItem: ({ item, index }) => (_jsx(ItemWithContext, { item: item, index: index })), keyExtractor: (item) => item.id, estimatedItemHeight: () => 1 }) }), _jsx(TestControl, { setFirstItemHeight: setFirstItemHeight })] }));
|
|
124
|
+
};
|
|
125
|
+
let setHeightFn = () => { };
|
|
126
|
+
const TestControl = ({ setFirstItemHeight, }) => {
|
|
127
|
+
setHeightFn = setFirstItemHeight;
|
|
128
|
+
return null;
|
|
129
|
+
};
|
|
130
|
+
const { lastFrame } = render(_jsx(TestComponent, {}));
|
|
131
|
+
await act(async () => {
|
|
132
|
+
await delay(0);
|
|
133
|
+
});
|
|
134
|
+
// Initially, only Item 0 (height 10) fills the 10px viewport
|
|
135
|
+
expect(lastFrame()).toContain('Item 0');
|
|
136
|
+
expect(lastFrame()).not.toContain('Item 1');
|
|
137
|
+
// Shrink Item 0 to 1px via context
|
|
138
|
+
await act(async () => {
|
|
139
|
+
setHeightFn(1);
|
|
140
|
+
await delay(0);
|
|
141
|
+
});
|
|
142
|
+
// Now Item 0 is 1px, so Items 1-9 should also be visible to fill 10px
|
|
143
|
+
expect(lastFrame()).toContain('Item 0');
|
|
144
|
+
expect(lastFrame()).toContain('Item 1');
|
|
145
|
+
expect(lastFrame()).toContain('Item 9');
|
|
146
|
+
});
|
|
147
|
+
});
|
|
148
|
+
//# sourceMappingURL=VirtualizedList.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"VirtualizedList.test.js","sourceRoot":"","sources":["../../../../../src/ui/components/shared/VirtualizedList.test.tsx"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,24 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @license
|
|
3
|
+
* Copyright 2025 Google LLC
|
|
4
|
+
* SPDX-License-Identifier: Apache-2.0
|
|
5
|
+
*/
|
|
6
|
+
import type React from 'react';
|
|
7
|
+
import { type DOMElement } from 'ink';
|
|
8
|
+
export interface ScrollState {
|
|
9
|
+
scrollTop: number;
|
|
10
|
+
scrollHeight: number;
|
|
11
|
+
innerHeight: number;
|
|
12
|
+
}
|
|
13
|
+
export interface ScrollableEntry {
|
|
14
|
+
id: string;
|
|
15
|
+
ref: React.RefObject<DOMElement>;
|
|
16
|
+
getScrollState: () => ScrollState;
|
|
17
|
+
scrollBy: (delta: number) => void;
|
|
18
|
+
hasFocus: () => boolean;
|
|
19
|
+
flashScrollbar: () => void;
|
|
20
|
+
}
|
|
21
|
+
export declare const ScrollProvider: React.FC<{
|
|
22
|
+
children: React.ReactNode;
|
|
23
|
+
}>;
|
|
24
|
+
export declare const useScrollable: (entry: Omit<ScrollableEntry, "id">, isActive: boolean) => void;
|
|
@@ -0,0 +1,100 @@
|
|
|
1
|
+
import { jsx as _jsx } from "react/jsx-runtime";
|
|
2
|
+
import { createContext, useCallback, useContext, useEffect, useMemo, useRef, useState, } from 'react';
|
|
3
|
+
import { getBoundingBox } from 'ink';
|
|
4
|
+
import { useMouse } from '../hooks/useMouse.js';
|
|
5
|
+
const ScrollContext = createContext(null);
|
|
6
|
+
const findScrollableCandidates = (mouseEvent, scrollables) => {
|
|
7
|
+
const candidates = [];
|
|
8
|
+
for (const entry of scrollables.values()) {
|
|
9
|
+
if (!entry.ref.current || !entry.hasFocus()) {
|
|
10
|
+
continue;
|
|
11
|
+
}
|
|
12
|
+
const boundingBox = getBoundingBox(entry.ref.current);
|
|
13
|
+
if (!boundingBox)
|
|
14
|
+
continue;
|
|
15
|
+
const { x, y, width, height } = boundingBox;
|
|
16
|
+
const isInside = mouseEvent.col >= x &&
|
|
17
|
+
mouseEvent.col < x + width + 1 && // Intentionally add one to width to include scrollbar.
|
|
18
|
+
mouseEvent.row >= y &&
|
|
19
|
+
mouseEvent.row < y + height;
|
|
20
|
+
if (isInside) {
|
|
21
|
+
candidates.push({ ...entry, area: width * height });
|
|
22
|
+
}
|
|
23
|
+
}
|
|
24
|
+
// Sort by smallest area first
|
|
25
|
+
candidates.sort((a, b) => a.area - b.area);
|
|
26
|
+
return candidates;
|
|
27
|
+
};
|
|
28
|
+
export const ScrollProvider = ({ children, }) => {
|
|
29
|
+
const [scrollables, setScrollables] = useState(new Map());
|
|
30
|
+
const register = useCallback((entry) => {
|
|
31
|
+
setScrollables((prev) => new Map(prev).set(entry.id, entry));
|
|
32
|
+
}, []);
|
|
33
|
+
const unregister = useCallback((id) => {
|
|
34
|
+
setScrollables((prev) => {
|
|
35
|
+
const next = new Map(prev);
|
|
36
|
+
next.delete(id);
|
|
37
|
+
return next;
|
|
38
|
+
});
|
|
39
|
+
}, []);
|
|
40
|
+
const scrollablesRef = useRef(scrollables);
|
|
41
|
+
useEffect(() => {
|
|
42
|
+
scrollablesRef.current = scrollables;
|
|
43
|
+
}, [scrollables]);
|
|
44
|
+
const handleScroll = (direction, mouseEvent) => {
|
|
45
|
+
const delta = direction === 'up' ? -1 : 1;
|
|
46
|
+
const candidates = findScrollableCandidates(mouseEvent, scrollablesRef.current);
|
|
47
|
+
for (const candidate of candidates) {
|
|
48
|
+
const { scrollTop, scrollHeight, innerHeight } = candidate.getScrollState();
|
|
49
|
+
// Epsilon to handle floating point inaccuracies.
|
|
50
|
+
const canScrollUp = scrollTop > 0.001;
|
|
51
|
+
const canScrollDown = scrollTop < scrollHeight - innerHeight - 0.001;
|
|
52
|
+
if (direction === 'up' && canScrollUp) {
|
|
53
|
+
candidate.scrollBy(delta);
|
|
54
|
+
return;
|
|
55
|
+
}
|
|
56
|
+
if (direction === 'down' && canScrollDown) {
|
|
57
|
+
candidate.scrollBy(delta);
|
|
58
|
+
return;
|
|
59
|
+
}
|
|
60
|
+
}
|
|
61
|
+
};
|
|
62
|
+
const handleClick = (mouseEvent) => {
|
|
63
|
+
const candidates = findScrollableCandidates(mouseEvent, scrollablesRef.current);
|
|
64
|
+
if (candidates.length > 0) {
|
|
65
|
+
// The first candidate is the innermost one.
|
|
66
|
+
candidates[0].flashScrollbar();
|
|
67
|
+
}
|
|
68
|
+
};
|
|
69
|
+
useMouse((event) => {
|
|
70
|
+
if (event.name === 'scroll-up') {
|
|
71
|
+
handleScroll('up', event);
|
|
72
|
+
}
|
|
73
|
+
else if (event.name === 'scroll-down') {
|
|
74
|
+
handleScroll('down', event);
|
|
75
|
+
}
|
|
76
|
+
else if (event.name === 'left-press') {
|
|
77
|
+
handleClick(event);
|
|
78
|
+
}
|
|
79
|
+
}, { isActive: true });
|
|
80
|
+
const contextValue = useMemo(() => ({ register, unregister }), [register, unregister]);
|
|
81
|
+
return (_jsx(ScrollContext.Provider, { value: contextValue, children: children }));
|
|
82
|
+
};
|
|
83
|
+
let nextId = 0;
|
|
84
|
+
export const useScrollable = (entry, isActive) => {
|
|
85
|
+
const context = useContext(ScrollContext);
|
|
86
|
+
if (!context) {
|
|
87
|
+
throw new Error('useScrollable must be used within a ScrollProvider');
|
|
88
|
+
}
|
|
89
|
+
const [id] = useState(() => `scrollable-${nextId++}`);
|
|
90
|
+
useEffect(() => {
|
|
91
|
+
if (isActive) {
|
|
92
|
+
context.register({ ...entry, id });
|
|
93
|
+
return () => {
|
|
94
|
+
context.unregister(id);
|
|
95
|
+
};
|
|
96
|
+
}
|
|
97
|
+
return;
|
|
98
|
+
}, [context, entry, id, isActive]);
|
|
99
|
+
};
|
|
100
|
+
//# sourceMappingURL=ScrollProvider.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"ScrollProvider.js","sourceRoot":"","sources":["../../../../src/ui/contexts/ScrollProvider.tsx"],"names":[],"mappings":"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"}
|