@google/gemini-cli 0.0.8999999 → 0.0.77777773

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.
Files changed (465) hide show
  1. package/README.md +105 -62
  2. package/dist/package.json +13 -5
  3. package/dist/src/commands/extensions/examples/context/GEMINI.md +9 -3
  4. package/dist/src/commands/extensions/examples/context/gemini-extension.json +1 -2
  5. package/dist/src/commands/extensions/examples/mcp-server/gemini-extension.json +3 -2
  6. package/dist/src/commands/extensions/examples/mcp-server/package.json +18 -0
  7. package/dist/src/commands/extensions/examples/mcp-server/tsconfig.json +13 -0
  8. package/dist/src/commands/extensions/install.d.ts +2 -2
  9. package/dist/src/commands/extensions/install.js +35 -36
  10. package/dist/src/commands/extensions/install.js.map +1 -1
  11. package/dist/src/commands/extensions/install.test.js +25 -25
  12. package/dist/src/commands/extensions/install.test.js.map +1 -1
  13. package/dist/src/commands/extensions/link.js +2 -2
  14. package/dist/src/commands/extensions/link.js.map +1 -1
  15. package/dist/src/commands/extensions/list.js +1 -1
  16. package/dist/src/commands/extensions/list.js.map +1 -1
  17. package/dist/src/commands/extensions/new.js +28 -8
  18. package/dist/src/commands/extensions/new.js.map +1 -1
  19. package/dist/src/commands/extensions/new.test.js +14 -5
  20. package/dist/src/commands/extensions/new.test.js.map +1 -1
  21. package/dist/src/commands/extensions/uninstall.js +1 -1
  22. package/dist/src/commands/extensions/uninstall.js.map +1 -1
  23. package/dist/src/commands/extensions/uninstall.test.js +4 -1
  24. package/dist/src/commands/extensions/uninstall.test.js.map +1 -1
  25. package/dist/src/commands/extensions/update.js +35 -24
  26. package/dist/src/commands/extensions/update.js.map +1 -1
  27. package/dist/src/commands/mcp/add.js +6 -1
  28. package/dist/src/commands/mcp/add.js.map +1 -1
  29. package/dist/src/commands/mcp/list.js +5 -4
  30. package/dist/src/commands/mcp/list.js.map +1 -1
  31. package/dist/src/config/config.d.ts +7 -4
  32. package/dist/src/config/config.js +125 -39
  33. package/dist/src/config/config.js.map +1 -1
  34. package/dist/src/config/extension.d.ts +44 -18
  35. package/dist/src/config/extension.js +250 -143
  36. package/dist/src/config/extension.js.map +1 -1
  37. package/dist/src/config/extensions/extensionEnablement.d.ts +12 -9
  38. package/dist/src/config/extensions/extensionEnablement.js +36 -9
  39. package/dist/src/config/extensions/extensionEnablement.js.map +1 -1
  40. package/dist/src/config/extensions/extensionEnablement.test.js +74 -1
  41. package/dist/src/config/extensions/extensionEnablement.test.js.map +1 -1
  42. package/dist/src/config/extensions/github.d.ts +14 -2
  43. package/dist/src/config/extensions/github.js +111 -89
  44. package/dist/src/config/extensions/github.js.map +1 -1
  45. package/dist/src/config/extensions/github.test.js +124 -13
  46. package/dist/src/config/extensions/github.test.js.map +1 -1
  47. package/dist/src/config/extensions/github_fetch.d.ts +7 -0
  48. package/dist/src/config/extensions/github_fetch.js +34 -0
  49. package/dist/src/config/extensions/github_fetch.js.map +1 -0
  50. package/dist/src/config/extensions/update.d.ts +4 -5
  51. package/dist/src/config/extensions/update.js +64 -42
  52. package/dist/src/config/extensions/update.js.map +1 -1
  53. package/dist/src/config/extensions/update.test.js +121 -71
  54. package/dist/src/config/extensions/update.test.js.map +1 -1
  55. package/dist/src/config/keyBindings.js +1 -1
  56. package/dist/src/config/keyBindings.js.map +1 -1
  57. package/dist/src/config/policy.js +2 -2
  58. package/dist/src/config/policy.js.map +1 -1
  59. package/dist/src/config/settings.d.ts +10 -1
  60. package/dist/src/config/settings.js +21 -6
  61. package/dist/src/config/settings.js.map +1 -1
  62. package/dist/src/config/settingsSchema.d.ts +97 -5
  63. package/dist/src/config/settingsSchema.js +96 -4
  64. package/dist/src/config/settingsSchema.js.map +1 -1
  65. package/dist/src/config/settingsSchema.test.js +8 -0
  66. package/dist/src/config/settingsSchema.test.js.map +1 -1
  67. package/dist/src/config/trustedFolders.d.ts +10 -2
  68. package/dist/src/config/trustedFolders.js +41 -16
  69. package/dist/src/config/trustedFolders.js.map +1 -1
  70. package/dist/src/config/trustedFolders.test.js +95 -14
  71. package/dist/src/config/trustedFolders.test.js.map +1 -1
  72. package/dist/src/gemini.js +114 -132
  73. package/dist/src/gemini.js.map +1 -1
  74. package/dist/src/gemini.test.js +95 -14
  75. package/dist/src/gemini.test.js.map +1 -1
  76. package/dist/src/generated/git-commit.d.ts +2 -2
  77. package/dist/src/generated/git-commit.js +2 -2
  78. package/dist/src/generated/git-commit.js.map +1 -1
  79. package/dist/src/services/BuiltinCommandLoader.js +7 -0
  80. package/dist/src/services/BuiltinCommandLoader.js.map +1 -1
  81. package/dist/src/services/BuiltinCommandLoader.test.js +87 -1
  82. package/dist/src/services/BuiltinCommandLoader.test.js.map +1 -1
  83. package/dist/src/services/FileCommandLoader.d.ts +1 -1
  84. package/dist/src/services/FileCommandLoader.js +4 -4
  85. package/dist/src/services/FileCommandLoader.js.map +1 -1
  86. package/dist/src/services/prompt-processors/shellProcessor.js +1 -1
  87. package/dist/src/services/prompt-processors/shellProcessor.js.map +1 -1
  88. package/dist/src/test-utils/render.d.ts +12 -1
  89. package/dist/src/test-utils/render.js +56 -1
  90. package/dist/src/test-utils/render.js.map +1 -1
  91. package/dist/src/ui/App.js +7 -9
  92. package/dist/src/ui/App.js.map +1 -1
  93. package/dist/src/ui/AppContainer.js +99 -27
  94. package/dist/src/ui/AppContainer.js.map +1 -1
  95. package/dist/src/ui/AppContainer.test.js +300 -7
  96. package/dist/src/ui/AppContainer.test.js.map +1 -1
  97. package/dist/src/ui/IdeIntegrationNudge.js +3 -0
  98. package/dist/src/ui/IdeIntegrationNudge.js.map +1 -1
  99. package/dist/src/ui/auth/AuthDialog.js +8 -1
  100. package/dist/src/ui/auth/AuthDialog.js.map +1 -1
  101. package/dist/src/ui/auth/AuthDialog.test.js +1 -0
  102. package/dist/src/ui/auth/AuthDialog.test.js.map +1 -1
  103. package/dist/src/ui/auth/AuthInProgress.js +2 -2
  104. package/dist/src/ui/auth/AuthInProgress.js.map +1 -1
  105. package/dist/src/ui/commands/chatCommand.js +21 -27
  106. package/dist/src/ui/commands/chatCommand.js.map +1 -1
  107. package/dist/src/ui/commands/extensionsCommand.js +39 -34
  108. package/dist/src/ui/commands/extensionsCommand.js.map +1 -1
  109. package/dist/src/ui/commands/mcpCommand.js +78 -260
  110. package/dist/src/ui/commands/mcpCommand.js.map +1 -1
  111. package/dist/src/ui/commands/memoryCommand.js +23 -5
  112. package/dist/src/ui/commands/memoryCommand.js.map +1 -1
  113. package/dist/src/ui/commands/modelCommand.d.ts +7 -0
  114. package/dist/src/ui/commands/modelCommand.js +16 -0
  115. package/dist/src/ui/commands/modelCommand.js.map +1 -0
  116. package/dist/src/ui/commands/modelCommand.test.js +30 -0
  117. package/dist/src/ui/commands/modelCommand.test.js.map +1 -0
  118. package/dist/src/ui/commands/permissionsCommand.d.ts +7 -0
  119. package/dist/src/ui/commands/permissionsCommand.js +16 -0
  120. package/dist/src/ui/commands/permissionsCommand.js.map +1 -0
  121. package/dist/src/ui/commands/permissionsCommand.test.d.ts +6 -0
  122. package/dist/src/ui/commands/permissionsCommand.test.js +30 -0
  123. package/dist/src/ui/commands/permissionsCommand.test.js.map +1 -0
  124. package/dist/src/ui/commands/profileCommand.d.ts +7 -0
  125. package/dist/src/ui/commands/profileCommand.js +23 -0
  126. package/dist/src/ui/commands/profileCommand.js.map +1 -0
  127. package/dist/src/ui/commands/setupGithubCommand.test.js +2 -1
  128. package/dist/src/ui/commands/setupGithubCommand.test.js.map +1 -1
  129. package/dist/src/ui/commands/toolsCommand.js +10 -24
  130. package/dist/src/ui/commands/toolsCommand.js.map +1 -1
  131. package/dist/src/ui/commands/types.d.ts +8 -6
  132. package/dist/src/ui/commands/types.js.map +1 -1
  133. package/dist/src/ui/components/AnsiOutput.d.ts +1 -0
  134. package/dist/src/ui/components/AnsiOutput.js +5 -5
  135. package/dist/src/ui/components/AnsiOutput.js.map +1 -1
  136. package/dist/src/ui/components/AnsiOutput.test.js +6 -6
  137. package/dist/src/ui/components/AnsiOutput.test.js.map +1 -1
  138. package/dist/src/ui/components/AppHeader.js +2 -5
  139. package/dist/src/ui/components/AppHeader.js.map +1 -1
  140. package/dist/src/ui/components/CliSpinner.d.ts +10 -0
  141. package/dist/src/ui/components/CliSpinner.js +20 -0
  142. package/dist/src/ui/components/CliSpinner.js.map +1 -0
  143. package/dist/src/ui/components/Composer.js +7 -29
  144. package/dist/src/ui/components/Composer.js.map +1 -1
  145. package/dist/src/ui/components/ConsentPrompt.d.ts +13 -0
  146. package/dist/src/ui/components/ConsentPrompt.js +19 -0
  147. package/dist/src/ui/components/ConsentPrompt.js.map +1 -0
  148. package/dist/src/ui/components/ConsentPrompt.test.d.ts +6 -0
  149. package/dist/src/ui/components/ConsentPrompt.test.js +67 -0
  150. package/dist/src/ui/components/ConsentPrompt.test.js.map +1 -0
  151. package/dist/src/ui/components/ContextSummaryDisplay.js +2 -2
  152. package/dist/src/ui/components/ContextSummaryDisplay.js.map +1 -1
  153. package/dist/src/ui/components/ContextUsageDisplay.d.ts +2 -1
  154. package/dist/src/ui/components/ContextUsageDisplay.js +4 -2
  155. package/dist/src/ui/components/ContextUsageDisplay.js.map +1 -1
  156. package/dist/src/ui/components/DebugProfiler.d.ts +18 -0
  157. package/dist/src/ui/components/DebugProfiler.js +158 -12
  158. package/dist/src/ui/components/DebugProfiler.js.map +1 -1
  159. package/dist/src/ui/components/DebugProfiler.test.d.ts +6 -0
  160. package/dist/src/ui/components/DebugProfiler.test.js +140 -0
  161. package/dist/src/ui/components/DebugProfiler.test.js.map +1 -0
  162. package/dist/src/ui/components/DialogManager.d.ts +7 -1
  163. package/dist/src/ui/components/DialogManager.js +18 -9
  164. package/dist/src/ui/components/DialogManager.js.map +1 -1
  165. package/dist/src/ui/components/EditorSettingsDialog.js +11 -2
  166. package/dist/src/ui/components/EditorSettingsDialog.js.map +1 -1
  167. package/dist/src/ui/components/ExitWarning.d.ts +7 -0
  168. package/dist/src/ui/components/ExitWarning.js +9 -0
  169. package/dist/src/ui/components/ExitWarning.js.map +1 -0
  170. package/dist/src/ui/components/FolderTrustDialog.js +3 -0
  171. package/dist/src/ui/components/FolderTrustDialog.js.map +1 -1
  172. package/dist/src/ui/components/FolderTrustDialog.test.js +2 -2
  173. package/dist/src/ui/components/FolderTrustDialog.test.js.map +1 -1
  174. package/dist/src/ui/components/Footer.d.ts +1 -19
  175. package/dist/src/ui/components/Footer.js +35 -17
  176. package/dist/src/ui/components/Footer.js.map +1 -1
  177. package/dist/src/ui/components/GeminiRespondingSpinner.js +2 -2
  178. package/dist/src/ui/components/GeminiRespondingSpinner.js.map +1 -1
  179. package/dist/src/ui/components/Help.js +1 -1
  180. package/dist/src/ui/components/Help.js.map +1 -1
  181. package/dist/src/ui/components/HistoryItemDisplay.d.ts +2 -1
  182. package/dist/src/ui/components/HistoryItemDisplay.js +10 -1
  183. package/dist/src/ui/components/HistoryItemDisplay.js.map +1 -1
  184. package/dist/src/ui/components/HistoryItemDisplay.test.js +90 -9
  185. package/dist/src/ui/components/HistoryItemDisplay.test.js.map +1 -1
  186. package/dist/src/ui/components/IdeTrustChangeDialog.d.ts +11 -0
  187. package/dist/src/ui/components/IdeTrustChangeDialog.js +32 -0
  188. package/dist/src/ui/components/IdeTrustChangeDialog.js.map +1 -0
  189. package/dist/src/ui/components/IdeTrustChangeDialog.test.d.ts +6 -0
  190. package/dist/src/ui/components/IdeTrustChangeDialog.test.js +57 -0
  191. package/dist/src/ui/components/IdeTrustChangeDialog.test.js.map +1 -0
  192. package/dist/src/ui/components/InputPrompt.d.ts +8 -2
  193. package/dist/src/ui/components/InputPrompt.js +66 -32
  194. package/dist/src/ui/components/InputPrompt.js.map +1 -1
  195. package/dist/src/ui/components/LoadingIndicator.js +1 -1
  196. package/dist/src/ui/components/LoadingIndicator.js.map +1 -1
  197. package/dist/src/ui/components/LoadingIndicator.test.js +4 -0
  198. package/dist/src/ui/components/LoadingIndicator.test.js.map +1 -1
  199. package/dist/src/ui/components/LoopDetectionConfirmation.js +2 -0
  200. package/dist/src/ui/components/LoopDetectionConfirmation.js.map +1 -1
  201. package/dist/src/ui/components/MainContent.js +7 -2
  202. package/dist/src/ui/components/MainContent.js.map +1 -1
  203. package/dist/src/ui/components/ModelDialog.d.ts +11 -0
  204. package/dist/src/ui/components/ModelDialog.js +57 -0
  205. package/dist/src/ui/components/ModelDialog.js.map +1 -0
  206. package/dist/src/ui/components/ModelDialog.test.d.ts +6 -0
  207. package/dist/src/ui/components/ModelDialog.test.js +153 -0
  208. package/dist/src/ui/components/ModelDialog.test.js.map +1 -0
  209. package/dist/src/ui/components/Notifications.js +12 -4
  210. package/dist/src/ui/components/Notifications.js.map +1 -1
  211. package/dist/src/ui/components/PermissionsModifyTrustDialog.d.ts +13 -0
  212. package/dist/src/ui/components/PermissionsModifyTrustDialog.js +48 -0
  213. package/dist/src/ui/components/PermissionsModifyTrustDialog.js.map +1 -0
  214. package/dist/src/ui/components/PermissionsModifyTrustDialog.test.d.ts +6 -0
  215. package/dist/src/ui/components/PermissionsModifyTrustDialog.test.js +154 -0
  216. package/dist/src/ui/components/PermissionsModifyTrustDialog.test.js.map +1 -0
  217. package/dist/src/ui/components/ProQuotaDialog.js +2 -0
  218. package/dist/src/ui/components/ProQuotaDialog.js.map +1 -1
  219. package/dist/src/ui/components/ProQuotaDialog.test.js +2 -0
  220. package/dist/src/ui/components/ProQuotaDialog.test.js.map +1 -1
  221. package/dist/src/ui/components/SettingsDialog.js +6 -3
  222. package/dist/src/ui/components/SettingsDialog.js.map +1 -1
  223. package/dist/src/ui/components/SettingsDialog.test.js +47 -13
  224. package/dist/src/ui/components/SettingsDialog.test.js.map +1 -1
  225. package/dist/src/ui/components/ShellConfirmationDialog.js +3 -0
  226. package/dist/src/ui/components/ShellConfirmationDialog.js.map +1 -1
  227. package/dist/src/ui/components/ThemeDialog.js +2 -0
  228. package/dist/src/ui/components/ThemeDialog.js.map +1 -1
  229. package/dist/src/ui/components/WorkspaceMigrationDialog.d.ts +2 -2
  230. package/dist/src/ui/components/WorkspaceMigrationDialog.js +7 -5
  231. package/dist/src/ui/components/WorkspaceMigrationDialog.js.map +1 -1
  232. package/dist/src/ui/components/messages/CompressionMessage.js +2 -2
  233. package/dist/src/ui/components/messages/CompressionMessage.js.map +1 -1
  234. package/dist/src/ui/components/messages/ToolConfirmationMessage.js +15 -1
  235. package/dist/src/ui/components/messages/ToolConfirmationMessage.js.map +1 -1
  236. package/dist/src/ui/components/messages/ToolGroupMessage.d.ts +1 -1
  237. package/dist/src/ui/components/messages/ToolGroupMessage.js +5 -5
  238. package/dist/src/ui/components/messages/ToolGroupMessage.js.map +1 -1
  239. package/dist/src/ui/components/messages/ToolMessage.d.ts +1 -1
  240. package/dist/src/ui/components/messages/ToolMessage.js +28 -5
  241. package/dist/src/ui/components/messages/ToolMessage.js.map +1 -1
  242. package/dist/src/ui/components/messages/UserMessage.js +1 -2
  243. package/dist/src/ui/components/messages/UserMessage.js.map +1 -1
  244. package/dist/src/ui/components/shared/BaseSelectionList.d.ts +38 -0
  245. package/dist/src/ui/components/shared/BaseSelectionList.js +72 -0
  246. package/dist/src/ui/components/shared/BaseSelectionList.js.map +1 -0
  247. package/dist/src/ui/components/shared/BaseSelectionList.test.d.ts +6 -0
  248. package/dist/src/ui/components/shared/BaseSelectionList.test.js +376 -0
  249. package/dist/src/ui/components/shared/BaseSelectionList.test.js.map +1 -0
  250. package/dist/src/ui/components/shared/DescriptiveRadioButtonSelect.d.ts +35 -0
  251. package/dist/src/ui/components/shared/DescriptiveRadioButtonSelect.js +13 -0
  252. package/dist/src/ui/components/shared/DescriptiveRadioButtonSelect.js.map +1 -0
  253. package/dist/src/ui/components/shared/DescriptiveRadioButtonSelect.test.d.ts +6 -0
  254. package/dist/src/ui/components/shared/DescriptiveRadioButtonSelect.test.js +79 -0
  255. package/dist/src/ui/components/shared/DescriptiveRadioButtonSelect.test.js.map +1 -0
  256. package/dist/src/ui/components/shared/RadioButtonSelect.d.ts +2 -3
  257. package/dist/src/ui/components/shared/RadioButtonSelect.js +9 -104
  258. package/dist/src/ui/components/shared/RadioButtonSelect.js.map +1 -1
  259. package/dist/src/ui/components/shared/RadioButtonSelect.test.js +115 -92
  260. package/dist/src/ui/components/shared/RadioButtonSelect.test.js.map +1 -1
  261. package/dist/src/ui/components/shared/ScopeSelector.js +4 -1
  262. package/dist/src/ui/components/shared/ScopeSelector.js.map +1 -1
  263. package/dist/src/ui/components/views/ChatList.d.ts +12 -0
  264. package/dist/src/ui/components/views/ChatList.js +17 -0
  265. package/dist/src/ui/components/views/ChatList.js.map +1 -0
  266. package/dist/src/ui/components/views/ChatList.test.d.ts +6 -0
  267. package/dist/src/ui/components/views/ChatList.test.js +42 -0
  268. package/dist/src/ui/components/views/ChatList.test.js.map +1 -0
  269. package/dist/src/ui/components/views/ExtensionsList.test.js +1 -1
  270. package/dist/src/ui/components/views/ExtensionsList.test.js.map +1 -1
  271. package/dist/src/ui/components/views/McpStatus.d.ts +27 -0
  272. package/dist/src/ui/components/views/McpStatus.js +77 -0
  273. package/dist/src/ui/components/views/McpStatus.js.map +1 -0
  274. package/dist/src/ui/components/views/McpStatus.test.d.ts +6 -0
  275. package/dist/src/ui/components/views/McpStatus.test.js +117 -0
  276. package/dist/src/ui/components/views/McpStatus.test.js.map +1 -0
  277. package/dist/src/ui/components/views/ToolsList.d.ts +14 -0
  278. package/dist/src/ui/components/views/ToolsList.js +7 -0
  279. package/dist/src/ui/components/views/ToolsList.js.map +1 -0
  280. package/dist/src/ui/components/views/ToolsList.test.d.ts +6 -0
  281. package/dist/src/ui/components/views/ToolsList.test.js +45 -0
  282. package/dist/src/ui/components/views/ToolsList.test.js.map +1 -0
  283. package/dist/src/ui/contexts/KeypressContext.js +3 -0
  284. package/dist/src/ui/contexts/KeypressContext.js.map +1 -1
  285. package/dist/src/ui/contexts/ShellFocusContext.d.ts +7 -0
  286. package/dist/src/ui/contexts/ShellFocusContext.js +9 -0
  287. package/dist/src/ui/contexts/ShellFocusContext.js.map +1 -0
  288. package/dist/src/ui/contexts/UIActionsContext.d.ts +2 -0
  289. package/dist/src/ui/contexts/UIActionsContext.js.map +1 -1
  290. package/dist/src/ui/contexts/UIStateContext.d.ts +11 -2
  291. package/dist/src/ui/contexts/UIStateContext.js +2 -0
  292. package/dist/src/ui/contexts/UIStateContext.js.map +1 -1
  293. package/dist/src/ui/hooks/shellCommandProcessor.js +5 -6
  294. package/dist/src/ui/hooks/shellCommandProcessor.js.map +1 -1
  295. package/dist/src/ui/hooks/shellCommandProcessor.test.js +19 -32
  296. package/dist/src/ui/hooks/shellCommandProcessor.test.js.map +1 -1
  297. package/dist/src/ui/hooks/slashCommandProcessor.d.ts +8 -5
  298. package/dist/src/ui/hooks/slashCommandProcessor.js +11 -2
  299. package/dist/src/ui/hooks/slashCommandProcessor.js.map +1 -1
  300. package/dist/src/ui/hooks/useExtensionUpdates.d.ts +10 -1
  301. package/dist/src/ui/hooks/useExtensionUpdates.js +140 -38
  302. package/dist/src/ui/hooks/useExtensionUpdates.js.map +1 -1
  303. package/dist/src/ui/hooks/useExtensionUpdates.test.js +163 -84
  304. package/dist/src/ui/hooks/useExtensionUpdates.test.js.map +1 -1
  305. package/dist/src/ui/hooks/useFlickerDetector.d.ts +14 -0
  306. package/dist/src/ui/hooks/useFlickerDetector.js +37 -0
  307. package/dist/src/ui/hooks/useFlickerDetector.js.map +1 -0
  308. package/dist/src/ui/hooks/useFlickerDetector.test.d.ts +6 -0
  309. package/dist/src/ui/hooks/useFlickerDetector.test.js +102 -0
  310. package/dist/src/ui/hooks/useFlickerDetector.test.js.map +1 -0
  311. package/dist/src/ui/hooks/useFocus.js +10 -0
  312. package/dist/src/ui/hooks/useFocus.js.map +1 -1
  313. package/dist/src/ui/hooks/useFolderTrust.d.ts +2 -1
  314. package/dist/src/ui/hooks/useFolderTrust.js +13 -9
  315. package/dist/src/ui/hooks/useFolderTrust.js.map +1 -1
  316. package/dist/src/ui/hooks/useGeminiStream.js +38 -3
  317. package/dist/src/ui/hooks/useGeminiStream.js.map +1 -1
  318. package/dist/src/ui/hooks/useGitBranchName.test.js.map +1 -1
  319. package/dist/src/ui/hooks/useHistoryManager.js +3 -3
  320. package/dist/src/ui/hooks/useHistoryManager.js.map +1 -1
  321. package/dist/src/ui/hooks/useIdeTrustListener.d.ts +4 -2
  322. package/dist/src/ui/hooks/useIdeTrustListener.js +40 -14
  323. package/dist/src/ui/hooks/useIdeTrustListener.js.map +1 -1
  324. package/dist/src/ui/hooks/useIdeTrustListener.test.d.ts +6 -0
  325. package/dist/src/ui/hooks/useIdeTrustListener.test.js +183 -0
  326. package/dist/src/ui/hooks/useIdeTrustListener.test.js.map +1 -0
  327. package/dist/src/ui/hooks/useModelCommand.d.ts +12 -0
  328. package/dist/src/ui/hooks/useModelCommand.js +21 -0
  329. package/dist/src/ui/hooks/useModelCommand.js.map +1 -0
  330. package/dist/src/ui/hooks/useModelCommand.test.d.ts +6 -0
  331. package/dist/src/ui/hooks/useModelCommand.test.js +35 -0
  332. package/dist/src/ui/hooks/useModelCommand.test.js.map +1 -0
  333. package/dist/src/ui/hooks/usePermissionsModifyTrust.d.ts +17 -0
  334. package/dist/src/ui/hooks/usePermissionsModifyTrust.js +78 -0
  335. package/dist/src/ui/hooks/usePermissionsModifyTrust.js.map +1 -0
  336. package/dist/src/ui/hooks/usePermissionsModifyTrust.test.d.ts +6 -0
  337. package/dist/src/ui/hooks/usePermissionsModifyTrust.test.js +182 -0
  338. package/dist/src/ui/hooks/usePermissionsModifyTrust.test.js.map +1 -0
  339. package/dist/src/ui/hooks/usePhraseCycler.js +1 -0
  340. package/dist/src/ui/hooks/usePhraseCycler.js.map +1 -1
  341. package/dist/src/ui/hooks/useQuotaAndFallback.js +3 -17
  342. package/dist/src/ui/hooks/useQuotaAndFallback.js.map +1 -1
  343. package/dist/src/ui/hooks/useQuotaAndFallback.test.js +13 -43
  344. package/dist/src/ui/hooks/useQuotaAndFallback.test.js.map +1 -1
  345. package/dist/src/ui/hooks/useSelectionList.d.ts +34 -0
  346. package/dist/src/ui/hooks/useSelectionList.js +272 -0
  347. package/dist/src/ui/hooks/useSelectionList.js.map +1 -0
  348. package/dist/src/ui/hooks/useSelectionList.test.d.ts +6 -0
  349. package/dist/src/ui/hooks/useSelectionList.test.js +725 -0
  350. package/dist/src/ui/hooks/useSelectionList.test.js.map +1 -0
  351. package/dist/src/ui/hooks/useShellHistory.test.js +12 -8
  352. package/dist/src/ui/hooks/useShellHistory.test.js.map +1 -1
  353. package/dist/src/ui/hooks/useSlashCompletion.js +7 -2
  354. package/dist/src/ui/hooks/useSlashCompletion.js.map +1 -1
  355. package/dist/src/ui/hooks/useSlashCompletion.test.js +33 -0
  356. package/dist/src/ui/hooks/useSlashCompletion.test.js.map +1 -1
  357. package/dist/src/ui/hooks/useTerminalSize.js +2 -3
  358. package/dist/src/ui/hooks/useTerminalSize.js.map +1 -1
  359. package/dist/src/ui/hooks/useToolScheduler.test.js +2 -2
  360. package/dist/src/ui/hooks/useToolScheduler.test.js.map +1 -1
  361. package/dist/src/ui/hooks/useWorkspaceMigration.d.ts +2 -2
  362. package/dist/src/ui/hooks/useWorkspaceMigration.js +13 -8
  363. package/dist/src/ui/hooks/useWorkspaceMigration.js.map +1 -1
  364. package/dist/src/ui/layouts/DefaultAppLayout.d.ts +7 -0
  365. package/dist/src/ui/layouts/DefaultAppLayout.js +16 -0
  366. package/dist/src/ui/layouts/DefaultAppLayout.js.map +1 -0
  367. package/dist/src/ui/layouts/ScreenReaderAppLayout.d.ts +7 -0
  368. package/dist/src/ui/layouts/ScreenReaderAppLayout.js +17 -0
  369. package/dist/src/ui/layouts/ScreenReaderAppLayout.js.map +1 -0
  370. package/dist/src/ui/noninteractive/nonInteractiveUi.js +3 -1
  371. package/dist/src/ui/noninteractive/nonInteractiveUi.js.map +1 -1
  372. package/dist/src/ui/privacy/CloudFreePrivacyNotice.js +5 -5
  373. package/dist/src/ui/privacy/CloudFreePrivacyNotice.js.map +1 -1
  374. package/dist/src/ui/state/extensions.d.ts +47 -1
  375. package/dist/src/ui/state/extensions.js +68 -1
  376. package/dist/src/ui/state/extensions.js.map +1 -1
  377. package/dist/src/ui/themes/theme.js +2 -2
  378. package/dist/src/ui/themes/theme.js.map +1 -1
  379. package/dist/src/ui/types.d.ts +54 -3
  380. package/dist/src/ui/types.js +3 -0
  381. package/dist/src/ui/types.js.map +1 -1
  382. package/dist/src/ui/utils/MarkdownDisplay.js +1 -2
  383. package/dist/src/ui/utils/MarkdownDisplay.js.map +1 -1
  384. package/dist/src/ui/utils/MarkdownDisplay.test.js +94 -91
  385. package/dist/src/ui/utils/MarkdownDisplay.test.js.map +1 -1
  386. package/dist/src/ui/utils/displayUtils.d.ts +1 -0
  387. package/dist/src/ui/utils/displayUtils.js +4 -1
  388. package/dist/src/ui/utils/displayUtils.js.map +1 -1
  389. package/dist/src/ui/utils/displayUtils.test.js +36 -17
  390. package/dist/src/ui/utils/displayUtils.test.js.map +1 -1
  391. package/dist/src/ui/utils/textUtils.d.ts +1 -0
  392. package/dist/src/ui/utils/textUtils.js +56 -0
  393. package/dist/src/ui/utils/textUtils.js.map +1 -1
  394. package/dist/src/ui/utils/textUtils.test.d.ts +6 -0
  395. package/dist/src/ui/utils/textUtils.test.js +132 -0
  396. package/dist/src/ui/utils/textUtils.test.js.map +1 -0
  397. package/dist/src/ui/utils/ui-sizing.d.ts +7 -0
  398. package/dist/src/ui/utils/ui-sizing.js +23 -0
  399. package/dist/src/ui/utils/ui-sizing.js.map +1 -0
  400. package/dist/src/utils/commentJson.js +95 -13
  401. package/dist/src/utils/commentJson.js.map +1 -1
  402. package/dist/src/utils/commentJson.test.js +161 -0
  403. package/dist/src/utils/commentJson.test.js.map +1 -1
  404. package/dist/src/utils/deepMerge.d.ts +2 -3
  405. package/dist/src/utils/errors.d.ts +8 -3
  406. package/dist/src/utils/errors.js +23 -13
  407. package/dist/src/utils/errors.js.map +1 -1
  408. package/dist/src/utils/errors.test.js +40 -46
  409. package/dist/src/utils/errors.test.js.map +1 -1
  410. package/dist/src/utils/events.d.ts +2 -1
  411. package/dist/src/utils/events.js +1 -0
  412. package/dist/src/utils/events.js.map +1 -1
  413. package/dist/src/utils/handleAutoUpdate.js +4 -1
  414. package/dist/src/utils/handleAutoUpdate.js.map +1 -1
  415. package/dist/src/utils/installationInfo.d.ts +1 -0
  416. package/dist/src/utils/installationInfo.js +2 -0
  417. package/dist/src/utils/installationInfo.js.map +1 -1
  418. package/dist/src/utils/math.d.ts +13 -0
  419. package/dist/src/utils/math.js +14 -0
  420. package/dist/src/utils/math.js.map +1 -0
  421. package/dist/src/utils/relaunch.d.ts +7 -0
  422. package/dist/src/utils/relaunch.js +57 -0
  423. package/dist/src/utils/relaunch.js.map +1 -0
  424. package/dist/src/utils/relaunch.test.d.ts +6 -0
  425. package/dist/src/utils/relaunch.test.js +273 -0
  426. package/dist/src/utils/relaunch.test.js.map +1 -0
  427. package/dist/src/utils/sandbox.js +31 -17
  428. package/dist/src/utils/sandbox.js.map +1 -1
  429. package/dist/src/utils/sessionCleanup.d.ts +22 -0
  430. package/dist/src/utils/sessionCleanup.integration.test.d.ts +6 -0
  431. package/dist/src/utils/sessionCleanup.integration.test.js +182 -0
  432. package/dist/src/utils/sessionCleanup.integration.test.js.map +1 -0
  433. package/dist/src/utils/sessionCleanup.js +214 -0
  434. package/dist/src/utils/sessionCleanup.js.map +1 -0
  435. package/dist/src/utils/sessionCleanup.test.d.ts +6 -0
  436. package/dist/src/utils/sessionCleanup.test.js +1232 -0
  437. package/dist/src/utils/sessionCleanup.test.js.map +1 -0
  438. package/dist/src/utils/sessionUtils.d.ts +37 -0
  439. package/dist/src/utils/sessionUtils.js +71 -0
  440. package/dist/src/utils/sessionUtils.js.map +1 -0
  441. package/dist/src/utils/windowTitle.d.ts +12 -0
  442. package/dist/src/utils/windowTitle.js +19 -0
  443. package/dist/src/utils/windowTitle.js.map +1 -0
  444. package/dist/src/utils/windowTitle.test.d.ts +6 -0
  445. package/dist/src/utils/windowTitle.test.js +49 -0
  446. package/dist/src/utils/windowTitle.test.js.map +1 -0
  447. package/dist/src/validateNonInterActiveAuth.js +6 -7
  448. package/dist/src/validateNonInterActiveAuth.js.map +1 -1
  449. package/dist/src/zed-integration/acp.js +1 -2
  450. package/dist/src/zed-integration/acp.js.map +1 -1
  451. package/dist/src/zed-integration/fileSystemService.d.ts +1 -0
  452. package/dist/src/zed-integration/fileSystemService.js +3 -0
  453. package/dist/src/zed-integration/fileSystemService.js.map +1 -1
  454. package/dist/src/zed-integration/schema.d.ts +70 -70
  455. package/dist/src/zed-integration/zedIntegration.d.ts +9 -3
  456. package/dist/src/zed-integration/zedIntegration.js +23 -28
  457. package/dist/src/zed-integration/zedIntegration.js.map +1 -1
  458. package/dist/tsconfig.tsbuildinfo +1 -1
  459. package/package.json +13 -5
  460. package/dist/src/commands/extensions/examples/mcp-server/example.js +0 -46
  461. package/dist/src/commands/extensions/examples/mcp-server/example.js.map +0 -1
  462. package/dist/src/ui/contexts/FocusContext.d.ts +0 -7
  463. package/dist/src/ui/contexts/FocusContext.js +0 -9
  464. package/dist/src/ui/contexts/FocusContext.js.map +0 -1
  465. /package/dist/src/{commands/extensions/examples/mcp-server/example.d.ts → ui/commands/modelCommand.test.d.ts} +0 -0
@@ -0,0 +1,725 @@
1
+ /**
2
+ * @license
3
+ * Copyright 2025 Google LLC
4
+ * SPDX-License-Identifier: Apache-2.0
5
+ */
6
+ import { describe, it, expect, vi, beforeEach, afterEach } from 'vitest';
7
+ import { renderHook, act } from '@testing-library/react';
8
+ import { useSelectionList, } from './useSelectionList.js';
9
+ import { useKeypress } from './useKeypress.js';
10
+ vi.mock('./useKeypress.js');
11
+ let activeKeypressHandler = null;
12
+ describe('useSelectionList', () => {
13
+ const mockOnSelect = vi.fn();
14
+ const mockOnHighlight = vi.fn();
15
+ const items = [
16
+ { value: 'A', key: 'A' },
17
+ { value: 'B', disabled: true, key: 'B' },
18
+ { value: 'C', key: 'C' },
19
+ { value: 'D', key: 'D' },
20
+ ];
21
+ beforeEach(() => {
22
+ activeKeypressHandler = null;
23
+ vi.mocked(useKeypress).mockImplementation((handler, options) => {
24
+ if (options?.isActive) {
25
+ activeKeypressHandler = handler;
26
+ }
27
+ else {
28
+ activeKeypressHandler = null;
29
+ }
30
+ });
31
+ mockOnSelect.mockClear();
32
+ mockOnHighlight.mockClear();
33
+ });
34
+ const pressKey = (name, sequence = name) => {
35
+ act(() => {
36
+ if (activeKeypressHandler) {
37
+ const key = {
38
+ name,
39
+ sequence,
40
+ ctrl: false,
41
+ meta: false,
42
+ shift: false,
43
+ paste: false,
44
+ };
45
+ activeKeypressHandler(key);
46
+ }
47
+ else {
48
+ throw new Error(`Test attempted to press key (${name}) but the keypress handler is not active. Ensure the hook is focused (isFocused=true) and the list is not empty.`);
49
+ }
50
+ });
51
+ };
52
+ describe('Initialization', () => {
53
+ it('should initialize with the default index (0) if enabled', () => {
54
+ const { result } = renderHook(() => useSelectionList({ items, onSelect: mockOnSelect }));
55
+ expect(result.current.activeIndex).toBe(0);
56
+ });
57
+ it('should initialize with the provided initialIndex if enabled', () => {
58
+ const { result } = renderHook(() => useSelectionList({
59
+ items,
60
+ initialIndex: 2,
61
+ onSelect: mockOnSelect,
62
+ }));
63
+ expect(result.current.activeIndex).toBe(2);
64
+ });
65
+ it('should handle an empty list gracefully', () => {
66
+ const { result } = renderHook(() => useSelectionList({ items: [], onSelect: mockOnSelect }));
67
+ expect(result.current.activeIndex).toBe(0);
68
+ });
69
+ it('should find the next enabled item (downwards) if initialIndex is disabled', () => {
70
+ const { result } = renderHook(() => useSelectionList({
71
+ items,
72
+ initialIndex: 1,
73
+ onSelect: mockOnSelect,
74
+ }));
75
+ expect(result.current.activeIndex).toBe(2);
76
+ });
77
+ it('should wrap around to find the next enabled item if initialIndex is disabled', () => {
78
+ const wrappingItems = [
79
+ { value: 'A', key: 'A' },
80
+ { value: 'B', disabled: true, key: 'B' },
81
+ { value: 'C', disabled: true, key: 'C' },
82
+ ];
83
+ const { result } = renderHook(() => useSelectionList({
84
+ items: wrappingItems,
85
+ initialIndex: 2,
86
+ onSelect: mockOnSelect,
87
+ }));
88
+ expect(result.current.activeIndex).toBe(0);
89
+ });
90
+ it('should default to 0 if initialIndex is out of bounds', () => {
91
+ const { result } = renderHook(() => useSelectionList({
92
+ items,
93
+ initialIndex: 10,
94
+ onSelect: mockOnSelect,
95
+ }));
96
+ expect(result.current.activeIndex).toBe(0);
97
+ const { result: resultNeg } = renderHook(() => useSelectionList({
98
+ items,
99
+ initialIndex: -1,
100
+ onSelect: mockOnSelect,
101
+ }));
102
+ expect(resultNeg.current.activeIndex).toBe(0);
103
+ });
104
+ it('should stick to the initial index if all items are disabled', () => {
105
+ const allDisabled = [
106
+ { value: 'A', disabled: true, key: 'A' },
107
+ { value: 'B', disabled: true, key: 'B' },
108
+ ];
109
+ const { result } = renderHook(() => useSelectionList({
110
+ items: allDisabled,
111
+ initialIndex: 1,
112
+ onSelect: mockOnSelect,
113
+ }));
114
+ expect(result.current.activeIndex).toBe(1);
115
+ });
116
+ });
117
+ describe('Keyboard Navigation (Up/Down/J/K)', () => {
118
+ it('should move down with "j" and "down" keys, skipping disabled items', () => {
119
+ const { result } = renderHook(() => useSelectionList({ items, onSelect: mockOnSelect }));
120
+ expect(result.current.activeIndex).toBe(0);
121
+ pressKey('j');
122
+ expect(result.current.activeIndex).toBe(2);
123
+ pressKey('down');
124
+ expect(result.current.activeIndex).toBe(3);
125
+ });
126
+ it('should move up with "k" and "up" keys, skipping disabled items', () => {
127
+ const { result } = renderHook(() => useSelectionList({ items, initialIndex: 3, onSelect: mockOnSelect }));
128
+ expect(result.current.activeIndex).toBe(3);
129
+ pressKey('k');
130
+ expect(result.current.activeIndex).toBe(2);
131
+ pressKey('up');
132
+ expect(result.current.activeIndex).toBe(0);
133
+ });
134
+ it('should wrap navigation correctly', () => {
135
+ const { result } = renderHook(() => useSelectionList({
136
+ items,
137
+ initialIndex: items.length - 1,
138
+ onSelect: mockOnSelect,
139
+ }));
140
+ expect(result.current.activeIndex).toBe(3);
141
+ pressKey('down');
142
+ expect(result.current.activeIndex).toBe(0);
143
+ pressKey('up');
144
+ expect(result.current.activeIndex).toBe(3);
145
+ });
146
+ it('should call onHighlight when index changes', () => {
147
+ renderHook(() => useSelectionList({
148
+ items,
149
+ onSelect: mockOnSelect,
150
+ onHighlight: mockOnHighlight,
151
+ }));
152
+ pressKey('down');
153
+ expect(mockOnHighlight).toHaveBeenCalledTimes(1);
154
+ expect(mockOnHighlight).toHaveBeenCalledWith('C');
155
+ });
156
+ it('should not move or call onHighlight if navigation results in the same index (e.g., single item)', () => {
157
+ const singleItem = [{ value: 'A', key: 'A' }];
158
+ const { result } = renderHook(() => useSelectionList({
159
+ items: singleItem,
160
+ onSelect: mockOnSelect,
161
+ onHighlight: mockOnHighlight,
162
+ }));
163
+ pressKey('down');
164
+ expect(result.current.activeIndex).toBe(0);
165
+ expect(mockOnHighlight).not.toHaveBeenCalled();
166
+ });
167
+ it('should not move or call onHighlight if all items are disabled', () => {
168
+ const allDisabled = [
169
+ { value: 'A', disabled: true, key: 'A' },
170
+ { value: 'B', disabled: true, key: 'B' },
171
+ ];
172
+ const { result } = renderHook(() => useSelectionList({
173
+ items: allDisabled,
174
+ onSelect: mockOnSelect,
175
+ onHighlight: mockOnHighlight,
176
+ }));
177
+ const initialIndex = result.current.activeIndex;
178
+ pressKey('down');
179
+ expect(result.current.activeIndex).toBe(initialIndex);
180
+ expect(mockOnHighlight).not.toHaveBeenCalled();
181
+ });
182
+ });
183
+ describe('Selection (Enter)', () => {
184
+ it('should call onSelect when "return" is pressed on enabled item', () => {
185
+ renderHook(() => useSelectionList({
186
+ items,
187
+ initialIndex: 2,
188
+ onSelect: mockOnSelect,
189
+ }));
190
+ pressKey('return');
191
+ expect(mockOnSelect).toHaveBeenCalledTimes(1);
192
+ expect(mockOnSelect).toHaveBeenCalledWith('C');
193
+ });
194
+ it('should not call onSelect if the active item is disabled', () => {
195
+ const { result } = renderHook(() => useSelectionList({
196
+ items,
197
+ onSelect: mockOnSelect,
198
+ }));
199
+ act(() => result.current.setActiveIndex(1));
200
+ pressKey('return');
201
+ expect(mockOnSelect).not.toHaveBeenCalled();
202
+ });
203
+ });
204
+ describe('Keyboard Navigation Robustness (Rapid Input)', () => {
205
+ it('should handle rapid navigation and selection robustly (avoiding stale state)', () => {
206
+ const { result } = renderHook(() => useSelectionList({
207
+ items, // A, B(disabled), C, D. Initial index 0 (A).
208
+ onSelect: mockOnSelect,
209
+ onHighlight: mockOnHighlight,
210
+ }));
211
+ // Simulate rapid inputs with separate act blocks to allow effects to run
212
+ if (!activeKeypressHandler)
213
+ throw new Error('Handler not active');
214
+ const handler = activeKeypressHandler;
215
+ const press = (name) => {
216
+ const key = {
217
+ name,
218
+ sequence: name,
219
+ ctrl: false,
220
+ meta: false,
221
+ shift: false,
222
+ paste: false,
223
+ };
224
+ handler(key);
225
+ };
226
+ // 1. Press Down. Should move 0 (A) -> 2 (C).
227
+ act(() => {
228
+ press('down');
229
+ });
230
+ // 2. Press Down again. Should move 2 (C) -> 3 (D).
231
+ act(() => {
232
+ press('down');
233
+ });
234
+ // 3. Press Enter. Should select D.
235
+ act(() => {
236
+ press('return');
237
+ });
238
+ expect(result.current.activeIndex).toBe(3);
239
+ expect(mockOnHighlight).toHaveBeenCalledTimes(2);
240
+ expect(mockOnHighlight).toHaveBeenNthCalledWith(1, 'C');
241
+ expect(mockOnHighlight).toHaveBeenNthCalledWith(2, 'D');
242
+ expect(mockOnSelect).toHaveBeenCalledTimes(1);
243
+ expect(mockOnSelect).toHaveBeenCalledWith('D');
244
+ expect(mockOnSelect).not.toHaveBeenCalledWith('A');
245
+ });
246
+ it('should handle ultra-rapid input (multiple presses in single act) without stale state', () => {
247
+ const { result } = renderHook(() => useSelectionList({
248
+ items, // A, B(disabled), C, D. Initial index 0 (A).
249
+ onSelect: mockOnSelect,
250
+ onHighlight: mockOnHighlight,
251
+ }));
252
+ // Simulate ultra-rapid inputs where all keypresses happen faster than React can re-render
253
+ act(() => {
254
+ if (!activeKeypressHandler)
255
+ throw new Error('Handler not active');
256
+ const handler = activeKeypressHandler;
257
+ const press = (name) => {
258
+ const key = {
259
+ name,
260
+ sequence: name,
261
+ ctrl: false,
262
+ meta: false,
263
+ shift: false,
264
+ paste: false,
265
+ };
266
+ handler(key);
267
+ };
268
+ // All presses happen in same render cycle - React batches the state updates
269
+ press('down'); // Should move 0 (A) -> 2 (C)
270
+ press('down'); // Should move 2 (C) -> 3 (D)
271
+ press('return'); // Should select D
272
+ });
273
+ expect(result.current.activeIndex).toBe(3);
274
+ expect(mockOnHighlight).toHaveBeenCalledWith('D');
275
+ expect(mockOnSelect).toHaveBeenCalledTimes(1);
276
+ expect(mockOnSelect).toHaveBeenCalledWith('D');
277
+ });
278
+ });
279
+ describe('Focus Management (isFocused)', () => {
280
+ it('should activate the keypress handler when focused (default) and items exist', () => {
281
+ const { result } = renderHook(() => useSelectionList({ items, onSelect: mockOnSelect }));
282
+ expect(activeKeypressHandler).not.toBeNull();
283
+ pressKey('down');
284
+ expect(result.current.activeIndex).toBe(2);
285
+ });
286
+ it('should not activate the keypress handler when isFocused is false', () => {
287
+ renderHook(() => useSelectionList({ items, onSelect: mockOnSelect, isFocused: false }));
288
+ expect(activeKeypressHandler).toBeNull();
289
+ expect(() => pressKey('down')).toThrow(/keypress handler is not active/);
290
+ });
291
+ it('should not activate the keypress handler when items list is empty', () => {
292
+ renderHook(() => useSelectionList({
293
+ items: [],
294
+ onSelect: mockOnSelect,
295
+ isFocused: true,
296
+ }));
297
+ expect(activeKeypressHandler).toBeNull();
298
+ expect(() => pressKey('down')).toThrow(/keypress handler is not active/);
299
+ });
300
+ it('should activate/deactivate when isFocused prop changes', () => {
301
+ const { result, rerender } = renderHook((props) => useSelectionList({ items, onSelect: mockOnSelect, ...props }), { initialProps: { isFocused: false } });
302
+ expect(activeKeypressHandler).toBeNull();
303
+ rerender({ isFocused: true });
304
+ expect(activeKeypressHandler).not.toBeNull();
305
+ pressKey('down');
306
+ expect(result.current.activeIndex).toBe(2);
307
+ rerender({ isFocused: false });
308
+ expect(activeKeypressHandler).toBeNull();
309
+ expect(() => pressKey('down')).toThrow(/keypress handler is not active/);
310
+ });
311
+ });
312
+ describe('Numeric Quick Selection (showNumbers=true)', () => {
313
+ beforeEach(() => {
314
+ vi.useFakeTimers();
315
+ });
316
+ afterEach(() => {
317
+ vi.useRealTimers();
318
+ });
319
+ const shortList = items;
320
+ const longList = Array.from({ length: 15 }, (_, i) => ({ value: `Item ${i + 1}`, key: `Item ${i + 1}` }));
321
+ const pressNumber = (num) => pressKey(num, num);
322
+ it('should not respond to numbers if showNumbers is false (default)', () => {
323
+ const { result } = renderHook(() => useSelectionList({ items: shortList, onSelect: mockOnSelect }));
324
+ pressNumber('1');
325
+ expect(result.current.activeIndex).toBe(0);
326
+ expect(mockOnSelect).not.toHaveBeenCalled();
327
+ });
328
+ it('should select item immediately if the number cannot be extended (unambiguous)', () => {
329
+ const { result } = renderHook(() => useSelectionList({
330
+ items: shortList,
331
+ onSelect: mockOnSelect,
332
+ onHighlight: mockOnHighlight,
333
+ showNumbers: true,
334
+ }));
335
+ pressNumber('3');
336
+ expect(result.current.activeIndex).toBe(2);
337
+ expect(mockOnHighlight).toHaveBeenCalledWith('C');
338
+ expect(mockOnSelect).toHaveBeenCalledTimes(1);
339
+ expect(mockOnSelect).toHaveBeenCalledWith('C');
340
+ expect(vi.getTimerCount()).toBe(0);
341
+ });
342
+ it('should highlight and wait for timeout if the number can be extended (ambiguous)', () => {
343
+ const { result } = renderHook(() => useSelectionList({
344
+ items: longList,
345
+ initialIndex: 1, // Start at index 1 so pressing "1" (index 0) causes a change
346
+ onSelect: mockOnSelect,
347
+ onHighlight: mockOnHighlight,
348
+ showNumbers: true,
349
+ }));
350
+ pressNumber('1');
351
+ expect(result.current.activeIndex).toBe(0);
352
+ expect(mockOnHighlight).toHaveBeenCalledWith('Item 1');
353
+ expect(mockOnSelect).not.toHaveBeenCalled();
354
+ expect(vi.getTimerCount()).toBe(1);
355
+ act(() => {
356
+ vi.advanceTimersByTime(1000);
357
+ });
358
+ expect(mockOnSelect).toHaveBeenCalledTimes(1);
359
+ expect(mockOnSelect).toHaveBeenCalledWith('Item 1');
360
+ });
361
+ it('should handle multi-digit input correctly', () => {
362
+ const { result } = renderHook(() => useSelectionList({
363
+ items: longList,
364
+ onSelect: mockOnSelect,
365
+ showNumbers: true,
366
+ }));
367
+ pressNumber('1');
368
+ expect(mockOnSelect).not.toHaveBeenCalled();
369
+ pressNumber('2');
370
+ expect(result.current.activeIndex).toBe(11);
371
+ expect(mockOnSelect).toHaveBeenCalledTimes(1);
372
+ expect(mockOnSelect).toHaveBeenCalledWith('Item 12');
373
+ });
374
+ it('should reset buffer if input becomes invalid (out of bounds)', () => {
375
+ const { result } = renderHook(() => useSelectionList({
376
+ items: shortList,
377
+ onSelect: mockOnSelect,
378
+ showNumbers: true,
379
+ }));
380
+ pressNumber('5');
381
+ expect(result.current.activeIndex).toBe(0);
382
+ expect(mockOnSelect).not.toHaveBeenCalled();
383
+ pressNumber('3');
384
+ expect(result.current.activeIndex).toBe(2);
385
+ expect(mockOnSelect).toHaveBeenCalledWith('C');
386
+ });
387
+ it('should allow "0" as subsequent digit, but ignore as first digit', () => {
388
+ const { result } = renderHook(() => useSelectionList({
389
+ items: longList,
390
+ onSelect: mockOnSelect,
391
+ showNumbers: true,
392
+ }));
393
+ pressNumber('0');
394
+ expect(result.current.activeIndex).toBe(0);
395
+ expect(mockOnSelect).not.toHaveBeenCalled();
396
+ // Timer should be running to clear the '0' input buffer
397
+ expect(vi.getTimerCount()).toBe(1);
398
+ // Press '1', then '0' (Item 10, index 9)
399
+ pressNumber('1');
400
+ pressNumber('0');
401
+ expect(result.current.activeIndex).toBe(9);
402
+ expect(mockOnSelect).toHaveBeenCalledWith('Item 10');
403
+ });
404
+ it('should clear the initial "0" input after timeout', () => {
405
+ renderHook(() => useSelectionList({
406
+ items: longList,
407
+ onSelect: mockOnSelect,
408
+ showNumbers: true,
409
+ }));
410
+ pressNumber('0');
411
+ act(() => vi.advanceTimersByTime(1000)); // Timeout the '0' input
412
+ pressNumber('1');
413
+ expect(mockOnSelect).not.toHaveBeenCalled(); // Should be waiting for second digit
414
+ act(() => vi.advanceTimersByTime(1000)); // Timeout '1'
415
+ expect(mockOnSelect).toHaveBeenCalledWith('Item 1');
416
+ });
417
+ it('should highlight but not select a disabled item (immediate selection case)', () => {
418
+ const { result } = renderHook(() => useSelectionList({
419
+ items: shortList, // B (index 1, number 2) is disabled
420
+ onSelect: mockOnSelect,
421
+ onHighlight: mockOnHighlight,
422
+ showNumbers: true,
423
+ }));
424
+ pressNumber('2');
425
+ expect(result.current.activeIndex).toBe(1);
426
+ expect(mockOnHighlight).toHaveBeenCalledWith('B');
427
+ // Should not select immediately, even though 20 > 4
428
+ expect(mockOnSelect).not.toHaveBeenCalled();
429
+ });
430
+ it('should highlight but not select a disabled item (timeout case)', () => {
431
+ // Create a list where the ambiguous prefix points to a disabled item
432
+ const disabledAmbiguousList = [
433
+ { value: 'Item 1 Disabled', disabled: true, key: 'Item 1 Disabled' },
434
+ ...longList.slice(1),
435
+ ];
436
+ const { result } = renderHook(() => useSelectionList({
437
+ items: disabledAmbiguousList,
438
+ onSelect: mockOnSelect,
439
+ showNumbers: true,
440
+ }));
441
+ pressNumber('1');
442
+ expect(result.current.activeIndex).toBe(0);
443
+ expect(vi.getTimerCount()).toBe(1);
444
+ act(() => {
445
+ vi.advanceTimersByTime(1000);
446
+ });
447
+ // Should not select after timeout
448
+ expect(mockOnSelect).not.toHaveBeenCalled();
449
+ });
450
+ it('should clear the number buffer if a non-numeric key (e.g., navigation) is pressed', () => {
451
+ const { result } = renderHook(() => useSelectionList({
452
+ items: longList,
453
+ onSelect: mockOnSelect,
454
+ showNumbers: true,
455
+ }));
456
+ pressNumber('1');
457
+ expect(vi.getTimerCount()).toBe(1);
458
+ pressKey('down');
459
+ expect(result.current.activeIndex).toBe(1);
460
+ expect(vi.getTimerCount()).toBe(0);
461
+ pressNumber('3');
462
+ // Should select '3', not '13'
463
+ expect(result.current.activeIndex).toBe(2);
464
+ });
465
+ it('should clear the number buffer if "return" is pressed', () => {
466
+ renderHook(() => useSelectionList({
467
+ items: longList,
468
+ onSelect: mockOnSelect,
469
+ showNumbers: true,
470
+ }));
471
+ pressNumber('1');
472
+ pressKey('return');
473
+ expect(mockOnSelect).toHaveBeenCalledTimes(1);
474
+ expect(vi.getTimerCount()).toBe(0);
475
+ act(() => {
476
+ vi.advanceTimersByTime(1000);
477
+ });
478
+ expect(mockOnSelect).toHaveBeenCalledTimes(1);
479
+ });
480
+ });
481
+ describe('Reactivity (Dynamic Updates)', () => {
482
+ it('should update activeIndex when initialIndex prop changes', () => {
483
+ const { result, rerender } = renderHook(({ initialIndex }) => useSelectionList({
484
+ items,
485
+ onSelect: mockOnSelect,
486
+ initialIndex,
487
+ }), { initialProps: { initialIndex: 0 } });
488
+ rerender({ initialIndex: 2 });
489
+ expect(result.current.activeIndex).toBe(2);
490
+ });
491
+ it('should respect a new initialIndex even after user interaction', () => {
492
+ const { result, rerender } = renderHook(({ initialIndex }) => useSelectionList({
493
+ items,
494
+ onSelect: mockOnSelect,
495
+ initialIndex,
496
+ }), { initialProps: { initialIndex: 0 } });
497
+ // User navigates, changing the active index
498
+ pressKey('down');
499
+ expect(result.current.activeIndex).toBe(2);
500
+ // The component re-renders with a new initial index
501
+ rerender({ initialIndex: 3 });
502
+ // The hook should now respect the new initial index
503
+ expect(result.current.activeIndex).toBe(3);
504
+ });
505
+ it('should validate index when initialIndex prop changes to a disabled item', () => {
506
+ const { result, rerender } = renderHook(({ initialIndex }) => useSelectionList({
507
+ items,
508
+ onSelect: mockOnSelect,
509
+ initialIndex,
510
+ }), { initialProps: { initialIndex: 0 } });
511
+ rerender({ initialIndex: 1 });
512
+ expect(result.current.activeIndex).toBe(2);
513
+ });
514
+ it('should adjust activeIndex if items change and the initialIndex is now out of bounds', () => {
515
+ const { result, rerender } = renderHook(({ items: testItems }) => useSelectionList({
516
+ onSelect: mockOnSelect,
517
+ initialIndex: 3,
518
+ items: testItems,
519
+ }), { initialProps: { items } });
520
+ expect(result.current.activeIndex).toBe(3);
521
+ const shorterItems = [
522
+ { value: 'X', key: 'X' },
523
+ { value: 'Y', key: 'Y' },
524
+ ];
525
+ rerender({ items: shorterItems }); // Length 2
526
+ // The useEffect syncs based on the initialIndex (3) which is now out of bounds. It defaults to 0.
527
+ expect(result.current.activeIndex).toBe(0);
528
+ });
529
+ it('should adjust activeIndex if items change and the initialIndex becomes disabled', () => {
530
+ const initialItems = [
531
+ { value: 'A', key: 'A' },
532
+ { value: 'B', key: 'B' },
533
+ { value: 'C', key: 'C' },
534
+ ];
535
+ const { result, rerender } = renderHook(({ items: testItems }) => useSelectionList({
536
+ onSelect: mockOnSelect,
537
+ initialIndex: 1,
538
+ items: testItems,
539
+ }), { initialProps: { items: initialItems } });
540
+ expect(result.current.activeIndex).toBe(1);
541
+ const newItems = [
542
+ { value: 'A', key: 'A' },
543
+ { value: 'B', disabled: true, key: 'B' },
544
+ { value: 'C', key: 'C' },
545
+ ];
546
+ rerender({ items: newItems });
547
+ expect(result.current.activeIndex).toBe(2);
548
+ });
549
+ it('should reset to 0 if items change to an empty list', () => {
550
+ const { result, rerender } = renderHook(({ items: testItems }) => useSelectionList({
551
+ onSelect: mockOnSelect,
552
+ initialIndex: 2,
553
+ items: testItems,
554
+ }), { initialProps: { items } });
555
+ rerender({ items: [] });
556
+ expect(result.current.activeIndex).toBe(0);
557
+ });
558
+ it('should not reset activeIndex when items are deeply equal', () => {
559
+ const initialItems = [
560
+ { value: 'A', key: 'A' },
561
+ { value: 'B', disabled: true, key: 'B' },
562
+ { value: 'C', key: 'C' },
563
+ { value: 'D', key: 'D' },
564
+ ];
565
+ const { result, rerender } = renderHook(({ items: testItems }) => useSelectionList({
566
+ onSelect: mockOnSelect,
567
+ onHighlight: mockOnHighlight,
568
+ initialIndex: 2,
569
+ items: testItems,
570
+ }), { initialProps: { items: initialItems } });
571
+ expect(result.current.activeIndex).toBe(2);
572
+ act(() => {
573
+ result.current.setActiveIndex(3);
574
+ });
575
+ expect(result.current.activeIndex).toBe(3);
576
+ mockOnHighlight.mockClear();
577
+ // Create new array with same content (deeply equal but not identical)
578
+ const newItems = [
579
+ { value: 'A', key: 'A' },
580
+ { value: 'B', disabled: true, key: 'B' },
581
+ { value: 'C', key: 'C' },
582
+ { value: 'D', key: 'D' },
583
+ ];
584
+ rerender({ items: newItems });
585
+ // Active index should remain the same since items are deeply equal
586
+ expect(result.current.activeIndex).toBe(3);
587
+ // onHighlight should NOT be called since the index didn't change
588
+ expect(mockOnHighlight).not.toHaveBeenCalled();
589
+ });
590
+ it('should update activeIndex when items change structurally', () => {
591
+ const initialItems = [
592
+ { value: 'A', key: 'A' },
593
+ { value: 'B', disabled: true, key: 'B' },
594
+ { value: 'C', key: 'C' },
595
+ { value: 'D', key: 'D' },
596
+ ];
597
+ const { result, rerender } = renderHook(({ items: testItems }) => useSelectionList({
598
+ onSelect: mockOnSelect,
599
+ onHighlight: mockOnHighlight,
600
+ initialIndex: 3,
601
+ items: testItems,
602
+ }), { initialProps: { items: initialItems } });
603
+ expect(result.current.activeIndex).toBe(3);
604
+ mockOnHighlight.mockClear();
605
+ // Change item values (not deeply equal)
606
+ const newItems = [
607
+ { value: 'X', key: 'X' },
608
+ { value: 'Y', key: 'Y' },
609
+ { value: 'Z', key: 'Z' },
610
+ ];
611
+ rerender({ items: newItems });
612
+ // Active index should update based on initialIndex and new items
613
+ expect(result.current.activeIndex).toBe(0);
614
+ });
615
+ it('should handle partial changes in items array', () => {
616
+ const initialItems = [
617
+ { value: 'A', key: 'A' },
618
+ { value: 'B', key: 'B' },
619
+ { value: 'C', key: 'C' },
620
+ ];
621
+ const { result, rerender } = renderHook(({ items: testItems }) => useSelectionList({
622
+ onSelect: mockOnSelect,
623
+ initialIndex: 1,
624
+ items: testItems,
625
+ }), { initialProps: { items: initialItems } });
626
+ expect(result.current.activeIndex).toBe(1);
627
+ // Change only one item's disabled status
628
+ const newItems = [
629
+ { value: 'A', key: 'A' },
630
+ { value: 'B', disabled: true, key: 'B' },
631
+ { value: 'C', key: 'C' },
632
+ ];
633
+ rerender({ items: newItems });
634
+ // Should find next valid index since current became disabled
635
+ expect(result.current.activeIndex).toBe(2);
636
+ });
637
+ it('should update selection when a new item is added to the start of the list', () => {
638
+ const initialItems = [
639
+ { value: 'A', key: 'A' },
640
+ { value: 'B', key: 'B' },
641
+ { value: 'C', key: 'C' },
642
+ ];
643
+ const { result, rerender } = renderHook(({ items: testItems }) => useSelectionList({
644
+ onSelect: mockOnSelect,
645
+ items: testItems,
646
+ }), { initialProps: { items: initialItems } });
647
+ pressKey('down');
648
+ expect(result.current.activeIndex).toBe(1);
649
+ const newItems = [
650
+ { value: 'D', key: 'D' },
651
+ { value: 'A', key: 'A' },
652
+ { value: 'B', key: 'B' },
653
+ { value: 'C', key: 'C' },
654
+ ];
655
+ rerender({ items: newItems });
656
+ expect(result.current.activeIndex).toBe(2);
657
+ });
658
+ it('should not re-initialize when items have identical keys but are different objects', () => {
659
+ const initialItems = [
660
+ { value: 'A', key: 'A' },
661
+ { value: 'B', key: 'B' },
662
+ ];
663
+ let renderCount = 0;
664
+ const { rerender } = renderHook(({ items: testItems }) => {
665
+ renderCount++;
666
+ return useSelectionList({
667
+ onSelect: mockOnSelect,
668
+ onHighlight: mockOnHighlight,
669
+ items: testItems,
670
+ });
671
+ }, { initialProps: { items: initialItems } });
672
+ // Initial render
673
+ expect(renderCount).toBe(1);
674
+ // Create new items with the same keys but different object references
675
+ const newItems = [
676
+ { value: 'A', key: 'A' },
677
+ { value: 'B', key: 'B' },
678
+ ];
679
+ rerender({ items: newItems });
680
+ expect(renderCount).toBe(2);
681
+ });
682
+ });
683
+ describe('Manual Control', () => {
684
+ it('should allow manual setting of active index via setActiveIndex', () => {
685
+ const { result } = renderHook(() => useSelectionList({ items, onSelect: mockOnSelect }));
686
+ act(() => {
687
+ result.current.setActiveIndex(3);
688
+ });
689
+ expect(result.current.activeIndex).toBe(3);
690
+ act(() => {
691
+ result.current.setActiveIndex(1);
692
+ });
693
+ expect(result.current.activeIndex).toBe(1);
694
+ });
695
+ });
696
+ describe('Cleanup', () => {
697
+ beforeEach(() => {
698
+ vi.useFakeTimers();
699
+ });
700
+ afterEach(() => {
701
+ vi.useRealTimers();
702
+ });
703
+ it('should clear timeout on unmount when timer is active', () => {
704
+ const longList = Array.from({ length: 15 }, (_, i) => ({ value: `Item ${i + 1}`, key: `Item ${i + 1}` }));
705
+ const { unmount } = renderHook(() => useSelectionList({
706
+ items: longList,
707
+ onSelect: mockOnSelect,
708
+ showNumbers: true,
709
+ }));
710
+ pressKey('1', '1');
711
+ expect(vi.getTimerCount()).toBe(1);
712
+ act(() => {
713
+ vi.advanceTimersByTime(500);
714
+ });
715
+ expect(mockOnSelect).not.toHaveBeenCalled();
716
+ unmount();
717
+ expect(vi.getTimerCount()).toBe(0);
718
+ act(() => {
719
+ vi.advanceTimersByTime(1000);
720
+ });
721
+ expect(mockOnSelect).not.toHaveBeenCalled();
722
+ });
723
+ });
724
+ });
725
+ //# sourceMappingURL=useSelectionList.test.js.map