fss-link 1.0.49 → 1.0.51

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 (419) hide show
  1. package/dist/index.js +0 -0
  2. package/dist/package.json +2 -2
  3. package/dist/src/config/auth.js +8 -5
  4. package/dist/src/config/auth.js.map +1 -1
  5. package/dist/src/config/database.d.ts +103 -11
  6. package/dist/src/config/database.js +301 -59
  7. package/dist/src/config/database.js.map +1 -1
  8. package/dist/src/config/databaseBackup.d.ts +114 -0
  9. package/dist/src/config/databaseBackup.js +334 -0
  10. package/dist/src/config/databaseBackup.js.map +1 -0
  11. package/dist/src/config/databaseMigrations.d.ts +63 -0
  12. package/dist/src/config/databaseMigrations.js +379 -0
  13. package/dist/src/config/databaseMigrations.js.map +1 -0
  14. package/dist/src/config/databasePool.d.ts +70 -0
  15. package/dist/src/config/databasePool.js +193 -0
  16. package/dist/src/config/databasePool.js.map +1 -0
  17. package/dist/src/config/queryOptimizer.d.ts +127 -0
  18. package/dist/src/config/queryOptimizer.js +309 -0
  19. package/dist/src/config/queryOptimizer.js.map +1 -0
  20. package/dist/src/utils/sandbox.js +2 -8
  21. package/dist/src/utils/sandbox.js.map +1 -1
  22. package/dist/src/validateNonInterActiveAuth.js +3 -7
  23. package/dist/src/validateNonInterActiveAuth.js.map +1 -1
  24. package/dist/tsconfig.tsbuildinfo +1 -1
  25. package/package.json +2 -2
  26. package/dist/commands/mcp/add.test.ts +0 -122
  27. package/dist/commands/mcp/add.ts +0 -222
  28. package/dist/commands/mcp/list.test.ts +0 -154
  29. package/dist/commands/mcp/list.ts +0 -139
  30. package/dist/commands/mcp/remove.test.ts +0 -69
  31. package/dist/commands/mcp/remove.ts +0 -60
  32. package/dist/commands/mcp.test.ts +0 -55
  33. package/dist/commands/mcp.ts +0 -27
  34. package/dist/config/apiValidation.test.ts +0 -118
  35. package/dist/config/auth.test.ts +0 -79
  36. package/dist/config/auth.ts +0 -100
  37. package/dist/config/config.integration.test.ts +0 -407
  38. package/dist/config/config.test.ts +0 -1952
  39. package/dist/config/config.ts +0 -690
  40. package/dist/config/database.test.ts +0 -96
  41. package/dist/config/database.ts +0 -824
  42. package/dist/config/extension.test.ts +0 -236
  43. package/dist/config/extension.ts +0 -180
  44. package/dist/config/keyBindings.test.ts +0 -62
  45. package/dist/config/keyBindings.ts +0 -184
  46. package/dist/config/modelManager.ts +0 -326
  47. package/dist/config/providerManager.ts +0 -244
  48. package/dist/config/providerPersistence.test.ts +0 -377
  49. package/dist/config/providerPersistence.ts +0 -105
  50. package/dist/config/sandboxConfig.ts +0 -107
  51. package/dist/config/settings.test.ts +0 -1424
  52. package/dist/config/settings.ts +0 -517
  53. package/dist/config/settingsSchema.test.ts +0 -252
  54. package/dist/config/settingsSchema.ts +0 -728
  55. package/dist/config/trustedFolders.test.ts +0 -208
  56. package/dist/config/trustedFolders.ts +0 -167
  57. package/dist/gemini.test.tsx +0 -252
  58. package/dist/gemini.tsx +0 -357
  59. package/dist/generated/git-commit.ts +0 -10
  60. package/dist/index.ts +0 -21
  61. package/dist/nonInteractiveCli.test.ts +0 -276
  62. package/dist/nonInteractiveCli.ts +0 -143
  63. package/dist/patches/is-in-ci.ts +0 -17
  64. package/dist/services/BuiltinCommandLoader.test.ts +0 -127
  65. package/dist/services/BuiltinCommandLoader.ts +0 -95
  66. package/dist/services/CommandService.test.ts +0 -352
  67. package/dist/services/CommandService.ts +0 -103
  68. package/dist/services/FileCommandLoader.test.ts +0 -1002
  69. package/dist/services/FileCommandLoader.ts +0 -289
  70. package/dist/services/McpPromptLoader.ts +0 -231
  71. package/dist/services/SearchEngineConfigProvider.ts +0 -100
  72. package/dist/services/prompt-processors/argumentProcessor.test.ts +0 -41
  73. package/dist/services/prompt-processors/argumentProcessor.ts +0 -23
  74. package/dist/services/prompt-processors/shellProcessor.test.ts +0 -709
  75. package/dist/services/prompt-processors/shellProcessor.ts +0 -248
  76. package/dist/services/prompt-processors/types.ts +0 -44
  77. package/dist/services/types.ts +0 -24
  78. package/dist/src/config/apiValidation.test.d.ts +0 -6
  79. package/dist/src/config/apiValidation.test.js +0 -99
  80. package/dist/src/config/apiValidation.test.js.map +0 -1
  81. package/dist/src/config/database.test.d.ts +0 -6
  82. package/dist/src/config/database.test.js +0 -80
  83. package/dist/src/config/database.test.js.map +0 -1
  84. package/dist/src/config/providerManager.d.ts +0 -74
  85. package/dist/src/config/providerManager.js +0 -203
  86. package/dist/src/config/providerManager.js.map +0 -1
  87. package/dist/src/config/providerPersistence.test.d.ts +0 -6
  88. package/dist/src/config/providerPersistence.test.js +0 -283
  89. package/dist/src/config/providerPersistence.test.js.map +0 -1
  90. package/dist/src/ui/components/GeminiKeyDialog.d.ts +0 -11
  91. package/dist/src/ui/components/GeminiKeyDialog.js +0 -156
  92. package/dist/src/ui/components/GeminiKeyDialog.js.map +0 -1
  93. package/dist/src/ui/components/OpenAIEndpointDialog.d.ts +0 -19
  94. package/dist/src/ui/components/OpenAIEndpointDialog.js +0 -163
  95. package/dist/src/ui/components/OpenAIEndpointDialog.js.map +0 -1
  96. package/dist/test-setup.ts +0 -12
  97. package/dist/test-utils/customMatchers.ts +0 -65
  98. package/dist/test-utils/mockCommandContext.test.ts +0 -62
  99. package/dist/test-utils/mockCommandContext.ts +0 -105
  100. package/dist/test-utils/render.tsx +0 -18
  101. package/dist/ui/App.test.tsx +0 -2181
  102. package/dist/ui/App.tsx +0 -1344
  103. package/dist/ui/IdeIntegrationNudge.tsx +0 -98
  104. package/dist/ui/__snapshots__/App.test.tsx.snap +0 -124
  105. package/dist/ui/colors.ts +0 -56
  106. package/dist/ui/commands/aboutCommand.test.ts +0 -153
  107. package/dist/ui/commands/aboutCommand.ts +0 -49
  108. package/dist/ui/commands/authCommand.test.ts +0 -36
  109. package/dist/ui/commands/authCommand.ts +0 -17
  110. package/dist/ui/commands/bugCommand.test.ts +0 -114
  111. package/dist/ui/commands/bugCommand.ts +0 -92
  112. package/dist/ui/commands/chatCommand.test.ts +0 -414
  113. package/dist/ui/commands/chatCommand.ts +0 -280
  114. package/dist/ui/commands/clearCommand.test.ts +0 -100
  115. package/dist/ui/commands/clearCommand.ts +0 -29
  116. package/dist/ui/commands/compressCommand.test.ts +0 -129
  117. package/dist/ui/commands/compressCommand.ts +0 -78
  118. package/dist/ui/commands/contextCommand.ts +0 -132
  119. package/dist/ui/commands/copyCommand.test.ts +0 -296
  120. package/dist/ui/commands/copyCommand.ts +0 -67
  121. package/dist/ui/commands/corgiCommand.test.ts +0 -34
  122. package/dist/ui/commands/corgiCommand.ts +0 -16
  123. package/dist/ui/commands/directoryCommand.test.tsx +0 -185
  124. package/dist/ui/commands/directoryCommand.tsx +0 -179
  125. package/dist/ui/commands/docsCommand.test.ts +0 -99
  126. package/dist/ui/commands/docsCommand.ts +0 -42
  127. package/dist/ui/commands/editorCommand.test.ts +0 -30
  128. package/dist/ui/commands/editorCommand.ts +0 -21
  129. package/dist/ui/commands/extensionsCommand.test.ts +0 -67
  130. package/dist/ui/commands/extensionsCommand.ts +0 -46
  131. package/dist/ui/commands/helpCommand.test.ts +0 -52
  132. package/dist/ui/commands/helpCommand.ts +0 -23
  133. package/dist/ui/commands/ideCommand.test.ts +0 -255
  134. package/dist/ui/commands/ideCommand.ts +0 -283
  135. package/dist/ui/commands/initCommand.test.ts +0 -127
  136. package/dist/ui/commands/initCommand.ts +0 -117
  137. package/dist/ui/commands/mcpCommand.test.ts +0 -1057
  138. package/dist/ui/commands/mcpCommand.ts +0 -531
  139. package/dist/ui/commands/memoryCommand.test.ts +0 -344
  140. package/dist/ui/commands/memoryCommand.ts +0 -305
  141. package/dist/ui/commands/privacyCommand.test.ts +0 -38
  142. package/dist/ui/commands/privacyCommand.ts +0 -17
  143. package/dist/ui/commands/quitCommand.test.ts +0 -55
  144. package/dist/ui/commands/quitCommand.ts +0 -36
  145. package/dist/ui/commands/restoreCommand.test.ts +0 -250
  146. package/dist/ui/commands/restoreCommand.ts +0 -157
  147. package/dist/ui/commands/searchEngineSetupCommand.ts +0 -18
  148. package/dist/ui/commands/settingsCommand.test.ts +0 -36
  149. package/dist/ui/commands/settingsCommand.ts +0 -17
  150. package/dist/ui/commands/setupGithubCommand.test.ts +0 -238
  151. package/dist/ui/commands/setupGithubCommand.ts +0 -212
  152. package/dist/ui/commands/speakCommand.ts +0 -175
  153. package/dist/ui/commands/statsCommand.test.ts +0 -78
  154. package/dist/ui/commands/statsCommand.ts +0 -70
  155. package/dist/ui/commands/terminalSetupCommand.test.ts +0 -85
  156. package/dist/ui/commands/terminalSetupCommand.ts +0 -45
  157. package/dist/ui/commands/themeCommand.test.ts +0 -38
  158. package/dist/ui/commands/themeCommand.ts +0 -17
  159. package/dist/ui/commands/toolsCommand.test.ts +0 -105
  160. package/dist/ui/commands/toolsCommand.ts +0 -71
  161. package/dist/ui/commands/ttsCommand.ts +0 -143
  162. package/dist/ui/commands/types.ts +0 -204
  163. package/dist/ui/commands/vimCommand.ts +0 -25
  164. package/dist/ui/commands/voiceCommand.ts +0 -125
  165. package/dist/ui/components/AboutBox.tsx +0 -133
  166. package/dist/ui/components/AsciiArt.ts +0 -54
  167. package/dist/ui/components/AuthDialog.test.tsx +0 -334
  168. package/dist/ui/components/AuthDialog.tsx +0 -289
  169. package/dist/ui/components/AuthInProgress.tsx +0 -62
  170. package/dist/ui/components/AutoAcceptIndicator.tsx +0 -47
  171. package/dist/ui/components/ConsoleSummaryDisplay.tsx +0 -35
  172. package/dist/ui/components/ContextSummaryDisplay.test.tsx +0 -85
  173. package/dist/ui/components/ContextSummaryDisplay.tsx +0 -120
  174. package/dist/ui/components/ContextUsageDisplay.tsx +0 -77
  175. package/dist/ui/components/DebugProfiler.tsx +0 -36
  176. package/dist/ui/components/DetailedMessagesDisplay.tsx +0 -82
  177. package/dist/ui/components/EditorSettingsDialog.tsx +0 -172
  178. package/dist/ui/components/FolderTrustDialog.test.tsx +0 -36
  179. package/dist/ui/components/FolderTrustDialog.tsx +0 -74
  180. package/dist/ui/components/Footer.test.tsx +0 -159
  181. package/dist/ui/components/Footer.tsx +0 -158
  182. package/dist/ui/components/GeminiKeyDialog.tsx +0 -252
  183. package/dist/ui/components/GeminiRespondingSpinner.tsx +0 -34
  184. package/dist/ui/components/Header.test.tsx +0 -44
  185. package/dist/ui/components/Header.tsx +0 -70
  186. package/dist/ui/components/Help.tsx +0 -174
  187. package/dist/ui/components/HistoryItemDisplay.test.tsx +0 -125
  188. package/dist/ui/components/HistoryItemDisplay.tsx +0 -98
  189. package/dist/ui/components/InputPrompt.test.tsx +0 -1467
  190. package/dist/ui/components/InputPrompt.tsx +0 -641
  191. package/dist/ui/components/LMStudioModelPrompt.tsx +0 -215
  192. package/dist/ui/components/LoadingIndicator.test.tsx +0 -296
  193. package/dist/ui/components/LoadingIndicator.tsx +0 -82
  194. package/dist/ui/components/MemoryUsageDisplay.tsx +0 -36
  195. package/dist/ui/components/ModelStatsDisplay.test.tsx +0 -252
  196. package/dist/ui/components/ModelStatsDisplay.tsx +0 -197
  197. package/dist/ui/components/OllamaModelPrompt.tsx +0 -206
  198. package/dist/ui/components/OpenAIEndpointDialog.tsx +0 -261
  199. package/dist/ui/components/OpenAIKeyPrompt.test.tsx +0 -64
  200. package/dist/ui/components/OpenAIKeyPrompt.tsx +0 -197
  201. package/dist/ui/components/PrepareLabel.tsx +0 -48
  202. package/dist/ui/components/SearchEngineConfigDialog.tsx +0 -280
  203. package/dist/ui/components/SessionSummaryDisplay.test.tsx +0 -75
  204. package/dist/ui/components/SessionSummaryDisplay.tsx +0 -18
  205. package/dist/ui/components/SettingsDialog.test.tsx +0 -865
  206. package/dist/ui/components/SettingsDialog.tsx +0 -753
  207. package/dist/ui/components/ShellConfirmationDialog.test.tsx +0 -53
  208. package/dist/ui/components/ShellConfirmationDialog.tsx +0 -103
  209. package/dist/ui/components/ShellModeIndicator.tsx +0 -18
  210. package/dist/ui/components/ShowMoreLines.tsx +0 -40
  211. package/dist/ui/components/StatsDisplay.test.tsx +0 -401
  212. package/dist/ui/components/StatsDisplay.tsx +0 -273
  213. package/dist/ui/components/SuggestionsDisplay.tsx +0 -102
  214. package/dist/ui/components/ThemeDialog.tsx +0 -310
  215. package/dist/ui/components/Tips.tsx +0 -45
  216. package/dist/ui/components/TodoDisplay.test.tsx +0 -97
  217. package/dist/ui/components/TodoDisplay.tsx +0 -72
  218. package/dist/ui/components/ToolStatsDisplay.test.tsx +0 -180
  219. package/dist/ui/components/ToolStatsDisplay.tsx +0 -208
  220. package/dist/ui/components/UpdateNotification.tsx +0 -23
  221. package/dist/ui/components/WelcomeBackDialog.tsx +0 -290
  222. package/dist/ui/components/__snapshots__/IDEContextDetailDisplay.test.tsx.snap +0 -24
  223. package/dist/ui/components/__snapshots__/ModelStatsDisplay.test.tsx.snap +0 -121
  224. package/dist/ui/components/__snapshots__/SessionSummaryDisplay.test.tsx.snap +0 -30
  225. package/dist/ui/components/__snapshots__/ShellConfirmationDialog.test.tsx.snap +0 -21
  226. package/dist/ui/components/__snapshots__/StatsDisplay.test.tsx.snap +0 -264
  227. package/dist/ui/components/__snapshots__/ToolStatsDisplay.test.tsx.snap +0 -91
  228. package/dist/ui/components/messages/CompressionMessage.tsx +0 -49
  229. package/dist/ui/components/messages/DiffRenderer.test.tsx +0 -365
  230. package/dist/ui/components/messages/DiffRenderer.tsx +0 -358
  231. package/dist/ui/components/messages/ErrorMessage.tsx +0 -31
  232. package/dist/ui/components/messages/GeminiMessage.tsx +0 -43
  233. package/dist/ui/components/messages/GeminiMessageContent.tsx +0 -43
  234. package/dist/ui/components/messages/InfoMessage.tsx +0 -32
  235. package/dist/ui/components/messages/ToolConfirmationMessage.test.tsx +0 -58
  236. package/dist/ui/components/messages/ToolConfirmationMessage.tsx +0 -297
  237. package/dist/ui/components/messages/ToolGroupMessage.tsx +0 -126
  238. package/dist/ui/components/messages/ToolMessage.test.tsx +0 -183
  239. package/dist/ui/components/messages/ToolMessage.tsx +0 -296
  240. package/dist/ui/components/messages/UserMessage.tsx +0 -43
  241. package/dist/ui/components/messages/UserShellMessage.tsx +0 -25
  242. package/dist/ui/components/shared/MaxSizedBox.test.tsx +0 -425
  243. package/dist/ui/components/shared/MaxSizedBox.tsx +0 -624
  244. package/dist/ui/components/shared/RadioButtonSelect.test.tsx +0 -181
  245. package/dist/ui/components/shared/RadioButtonSelect.tsx +0 -234
  246. package/dist/ui/components/shared/__snapshots__/RadioButtonSelect.test.tsx.snap +0 -47
  247. package/dist/ui/components/shared/text-buffer.test.ts +0 -1728
  248. package/dist/ui/components/shared/text-buffer.ts +0 -2227
  249. package/dist/ui/components/shared/vim-buffer-actions.test.ts +0 -1119
  250. package/dist/ui/components/shared/vim-buffer-actions.ts +0 -814
  251. package/dist/ui/constants.ts +0 -17
  252. package/dist/ui/contexts/KeypressContext.test.tsx +0 -391
  253. package/dist/ui/contexts/KeypressContext.tsx +0 -440
  254. package/dist/ui/contexts/OverflowContext.tsx +0 -87
  255. package/dist/ui/contexts/SessionContext.test.tsx +0 -132
  256. package/dist/ui/contexts/SessionContext.tsx +0 -143
  257. package/dist/ui/contexts/SettingsContext.tsx +0 -20
  258. package/dist/ui/contexts/StreamingContext.tsx +0 -22
  259. package/dist/ui/contexts/VimModeContext.tsx +0 -79
  260. package/dist/ui/editors/editorSettingsManager.ts +0 -66
  261. package/dist/ui/hooks/atCommandProcessor.test.ts +0 -1102
  262. package/dist/ui/hooks/atCommandProcessor.ts +0 -485
  263. package/dist/ui/hooks/shellCommandProcessor.test.ts +0 -481
  264. package/dist/ui/hooks/shellCommandProcessor.ts +0 -314
  265. package/dist/ui/hooks/slashCommandProcessor.test.ts +0 -1044
  266. package/dist/ui/hooks/slashCommandProcessor.ts +0 -595
  267. package/dist/ui/hooks/useAtCompletion.test.ts +0 -497
  268. package/dist/ui/hooks/useAtCompletion.ts +0 -244
  269. package/dist/ui/hooks/useAuthCommand.ts +0 -129
  270. package/dist/ui/hooks/useAutoAcceptIndicator.test.ts +0 -300
  271. package/dist/ui/hooks/useAutoAcceptIndicator.ts +0 -52
  272. package/dist/ui/hooks/useBracketedPaste.ts +0 -37
  273. package/dist/ui/hooks/useCommandCompletion.test.ts +0 -518
  274. package/dist/ui/hooks/useCommandCompletion.tsx +0 -238
  275. package/dist/ui/hooks/useCompletion.ts +0 -128
  276. package/dist/ui/hooks/useConsoleMessages.test.ts +0 -147
  277. package/dist/ui/hooks/useConsoleMessages.ts +0 -110
  278. package/dist/ui/hooks/useEditorSettings.test.ts +0 -283
  279. package/dist/ui/hooks/useEditorSettings.ts +0 -75
  280. package/dist/ui/hooks/useFocus.test.ts +0 -119
  281. package/dist/ui/hooks/useFocus.ts +0 -48
  282. package/dist/ui/hooks/useFolderTrust.test.ts +0 -159
  283. package/dist/ui/hooks/useFolderTrust.ts +0 -72
  284. package/dist/ui/hooks/useGeminiStream.test.tsx +0 -1998
  285. package/dist/ui/hooks/useGeminiStream.ts +0 -1017
  286. package/dist/ui/hooks/useGitBranchName.test.ts +0 -280
  287. package/dist/ui/hooks/useGitBranchName.ts +0 -79
  288. package/dist/ui/hooks/useHistoryManager.test.ts +0 -202
  289. package/dist/ui/hooks/useHistoryManager.ts +0 -111
  290. package/dist/ui/hooks/useInputHistory.test.ts +0 -261
  291. package/dist/ui/hooks/useInputHistory.ts +0 -111
  292. package/dist/ui/hooks/useKeypress.test.ts +0 -280
  293. package/dist/ui/hooks/useKeypress.ts +0 -39
  294. package/dist/ui/hooks/useKittyKeyboardProtocol.ts +0 -31
  295. package/dist/ui/hooks/useLoadingIndicator.test.ts +0 -139
  296. package/dist/ui/hooks/useLoadingIndicator.ts +0 -57
  297. package/dist/ui/hooks/useLogger.ts +0 -32
  298. package/dist/ui/hooks/useMessageQueue.test.ts +0 -226
  299. package/dist/ui/hooks/useMessageQueue.ts +0 -69
  300. package/dist/ui/hooks/usePhraseCycler.test.ts +0 -145
  301. package/dist/ui/hooks/usePhraseCycler.ts +0 -198
  302. package/dist/ui/hooks/usePrivacySettings.test.ts +0 -242
  303. package/dist/ui/hooks/usePrivacySettings.ts +0 -150
  304. package/dist/ui/hooks/useReactToolScheduler.ts +0 -309
  305. package/dist/ui/hooks/useRefreshMemoryCommand.ts +0 -7
  306. package/dist/ui/hooks/useReverseSearchCompletion.test.tsx +0 -260
  307. package/dist/ui/hooks/useReverseSearchCompletion.tsx +0 -95
  308. package/dist/ui/hooks/useSettingsCommand.ts +0 -25
  309. package/dist/ui/hooks/useShellHistory.test.ts +0 -219
  310. package/dist/ui/hooks/useShellHistory.ts +0 -133
  311. package/dist/ui/hooks/useShowMemoryCommand.ts +0 -75
  312. package/dist/ui/hooks/useSlashCompletion.test.ts +0 -434
  313. package/dist/ui/hooks/useSlashCompletion.ts +0 -187
  314. package/dist/ui/hooks/useStateAndRef.ts +0 -36
  315. package/dist/ui/hooks/useTerminalSize.ts +0 -32
  316. package/dist/ui/hooks/useThemeCommand.ts +0 -110
  317. package/dist/ui/hooks/useTimer.test.ts +0 -120
  318. package/dist/ui/hooks/useTimer.ts +0 -65
  319. package/dist/ui/hooks/useToolScheduler.test.ts +0 -1123
  320. package/dist/ui/hooks/useWelcomeBack.ts +0 -253
  321. package/dist/ui/hooks/vim.test.ts +0 -1691
  322. package/dist/ui/hooks/vim.ts +0 -784
  323. package/dist/ui/keyMatchers.test.ts +0 -337
  324. package/dist/ui/keyMatchers.ts +0 -105
  325. package/dist/ui/privacy/CloudFreePrivacyNotice.tsx +0 -117
  326. package/dist/ui/privacy/CloudPaidPrivacyNotice.tsx +0 -59
  327. package/dist/ui/privacy/GeminiPrivacyNotice.tsx +0 -62
  328. package/dist/ui/privacy/PrivacyNotice.tsx +0 -42
  329. package/dist/ui/semantic-colors.ts +0 -26
  330. package/dist/ui/themes/ansi-light.ts +0 -150
  331. package/dist/ui/themes/ansi.ts +0 -159
  332. package/dist/ui/themes/atom-one-dark.ts +0 -147
  333. package/dist/ui/themes/ayu-light.ts +0 -139
  334. package/dist/ui/themes/ayu.ts +0 -113
  335. package/dist/ui/themes/color-utils.test.ts +0 -221
  336. package/dist/ui/themes/color-utils.ts +0 -231
  337. package/dist/ui/themes/default-light.ts +0 -108
  338. package/dist/ui/themes/default.ts +0 -151
  339. package/dist/ui/themes/dracula.ts +0 -124
  340. package/dist/ui/themes/fss-code-dark.ts +0 -156
  341. package/dist/ui/themes/fss-dark.ts +0 -113
  342. package/dist/ui/themes/fss-light.ts +0 -139
  343. package/dist/ui/themes/github-dark.ts +0 -147
  344. package/dist/ui/themes/github-light.ts +0 -149
  345. package/dist/ui/themes/googlecode.ts +0 -146
  346. package/dist/ui/themes/no-color.ts +0 -125
  347. package/dist/ui/themes/qwen-dark.ts +0 -118
  348. package/dist/ui/themes/qwen-light.ts +0 -144
  349. package/dist/ui/themes/semantic-tokens.ts +0 -127
  350. package/dist/ui/themes/shades-of-purple.ts +0 -352
  351. package/dist/ui/themes/theme-manager.test.ts +0 -99
  352. package/dist/ui/themes/theme-manager.ts +0 -257
  353. package/dist/ui/themes/theme.test.ts +0 -97
  354. package/dist/ui/themes/theme.ts +0 -451
  355. package/dist/ui/themes/xcode.ts +0 -154
  356. package/dist/ui/types.ts +0 -255
  357. package/dist/ui/utils/CodeColorizer.tsx +0 -217
  358. package/dist/ui/utils/ConsolePatcher.ts +0 -71
  359. package/dist/ui/utils/InlineMarkdownRenderer.tsx +0 -173
  360. package/dist/ui/utils/MarkdownDisplay.test.tsx +0 -244
  361. package/dist/ui/utils/MarkdownDisplay.tsx +0 -415
  362. package/dist/ui/utils/TableRenderer.tsx +0 -159
  363. package/dist/ui/utils/__snapshots__/MarkdownDisplay.test.tsx.snap +0 -93
  364. package/dist/ui/utils/clipboardUtils.test.ts +0 -76
  365. package/dist/ui/utils/clipboardUtils.ts +0 -149
  366. package/dist/ui/utils/commandUtils.test.ts +0 -384
  367. package/dist/ui/utils/commandUtils.ts +0 -106
  368. package/dist/ui/utils/computeStats.test.ts +0 -292
  369. package/dist/ui/utils/computeStats.ts +0 -86
  370. package/dist/ui/utils/displayUtils.test.ts +0 -58
  371. package/dist/ui/utils/displayUtils.ts +0 -32
  372. package/dist/ui/utils/formatters.test.ts +0 -72
  373. package/dist/ui/utils/formatters.ts +0 -63
  374. package/dist/ui/utils/isNarrowWidth.ts +0 -9
  375. package/dist/ui/utils/kittyProtocolDetector.ts +0 -105
  376. package/dist/ui/utils/markdownUtilities.test.ts +0 -50
  377. package/dist/ui/utils/markdownUtilities.ts +0 -125
  378. package/dist/ui/utils/platformConstants.ts +0 -52
  379. package/dist/ui/utils/terminalSetup.ts +0 -342
  380. package/dist/ui/utils/textUtils.ts +0 -40
  381. package/dist/ui/utils/updateCheck.test.ts +0 -163
  382. package/dist/ui/utils/updateCheck.ts +0 -100
  383. package/dist/utils/checks.ts +0 -28
  384. package/dist/utils/cleanup.test.ts +0 -68
  385. package/dist/utils/cleanup.ts +0 -36
  386. package/dist/utils/dialogScopeUtils.ts +0 -64
  387. package/dist/utils/events.ts +0 -14
  388. package/dist/utils/gitUtils.test.ts +0 -149
  389. package/dist/utils/gitUtils.ts +0 -116
  390. package/dist/utils/handleAutoUpdate.test.ts +0 -272
  391. package/dist/utils/handleAutoUpdate.ts +0 -145
  392. package/dist/utils/installationInfo.test.ts +0 -315
  393. package/dist/utils/installationInfo.ts +0 -176
  394. package/dist/utils/package.ts +0 -38
  395. package/dist/utils/readStdin.ts +0 -51
  396. package/dist/utils/resolvePath.ts +0 -21
  397. package/dist/utils/sandbox-macos-permissive-closed.sb +0 -32
  398. package/dist/utils/sandbox-macos-permissive-open.sb +0 -25
  399. package/dist/utils/sandbox-macos-permissive-proxied.sb +0 -37
  400. package/dist/utils/sandbox-macos-restrictive-closed.sb +0 -93
  401. package/dist/utils/sandbox-macos-restrictive-open.sb +0 -96
  402. package/dist/utils/sandbox-macos-restrictive-proxied.sb +0 -98
  403. package/dist/utils/sandbox.ts +0 -962
  404. package/dist/utils/settingsUtils.test.ts +0 -797
  405. package/dist/utils/settingsUtils.ts +0 -489
  406. package/dist/utils/spawnWrapper.ts +0 -9
  407. package/dist/utils/startupWarnings.test.ts +0 -83
  408. package/dist/utils/startupWarnings.ts +0 -40
  409. package/dist/utils/updateEventEmitter.ts +0 -13
  410. package/dist/utils/userStartupWarnings.test.ts +0 -87
  411. package/dist/utils/userStartupWarnings.ts +0 -69
  412. package/dist/utils/version.ts +0 -12
  413. package/dist/validateNonInterActiveAuth.test.ts +0 -260
  414. package/dist/validateNonInterActiveAuth.ts +0 -51
  415. package/dist/vitest.config.ts +0 -37
  416. package/dist/zed-integration/acp.ts +0 -366
  417. package/dist/zed-integration/fileSystemService.ts +0 -47
  418. package/dist/zed-integration/schema.ts +0 -466
  419. package/dist/zed-integration/zedIntegration.ts +0 -944
@@ -1,1119 +0,0 @@
1
- /**
2
- * @license
3
- * Copyright 2025 Google LLC
4
- * SPDX-License-Identifier: Apache-2.0
5
- */
6
-
7
- import { describe, it, expect } from 'vitest';
8
- import { handleVimAction } from './vim-buffer-actions.js';
9
- import type { TextBufferState } from './text-buffer.js';
10
-
11
- // Helper to create test state
12
- const createTestState = (
13
- lines: string[] = ['hello world'],
14
- cursorRow = 0,
15
- cursorCol = 0,
16
- ): TextBufferState => ({
17
- lines,
18
- cursorRow,
19
- cursorCol,
20
- preferredCol: null,
21
- undoStack: [],
22
- redoStack: [],
23
- clipboard: null,
24
- selectionAnchor: null,
25
- viewportWidth: 80,
26
- });
27
-
28
- describe('vim-buffer-actions', () => {
29
- describe('Movement commands', () => {
30
- describe('vim_move_left', () => {
31
- it('should move cursor left by count', () => {
32
- const state = createTestState(['hello world'], 0, 5);
33
- const action = {
34
- type: 'vim_move_left' as const,
35
- payload: { count: 3 },
36
- };
37
-
38
- const result = handleVimAction(state, action);
39
- expect(result).toHaveOnlyValidCharacters();
40
- expect(result.cursorCol).toBe(2);
41
- expect(result.preferredCol).toBeNull();
42
- });
43
-
44
- it('should not move past beginning of line', () => {
45
- const state = createTestState(['hello'], 0, 2);
46
- const action = {
47
- type: 'vim_move_left' as const,
48
- payload: { count: 5 },
49
- };
50
-
51
- const result = handleVimAction(state, action);
52
- expect(result).toHaveOnlyValidCharacters();
53
- expect(result.cursorCol).toBe(0);
54
- });
55
-
56
- it('should wrap to previous line when at beginning', () => {
57
- const state = createTestState(['line1', 'line2'], 1, 0);
58
- const action = {
59
- type: 'vim_move_left' as const,
60
- payload: { count: 1 },
61
- };
62
-
63
- const result = handleVimAction(state, action);
64
- expect(result).toHaveOnlyValidCharacters();
65
- expect(result.cursorRow).toBe(0);
66
- expect(result.cursorCol).toBe(4); // On last character '1' of 'line1'
67
- });
68
-
69
- it('should handle multiple line wrapping', () => {
70
- const state = createTestState(['abc', 'def', 'ghi'], 2, 0);
71
- const action = {
72
- type: 'vim_move_left' as const,
73
- payload: { count: 5 },
74
- };
75
-
76
- const result = handleVimAction(state, action);
77
- expect(result).toHaveOnlyValidCharacters();
78
- expect(result.cursorRow).toBe(0);
79
- expect(result.cursorCol).toBe(1); // On 'b' after 5 left movements
80
- });
81
-
82
- it('should correctly handle h/l movement between lines', () => {
83
- // Start at end of first line at 'd' (position 10)
84
- let state = createTestState(['hello world', 'foo bar'], 0, 10);
85
-
86
- // Move right - should go to beginning of next line
87
- state = handleVimAction(state, {
88
- type: 'vim_move_right' as const,
89
- payload: { count: 1 },
90
- });
91
- expect(state).toHaveOnlyValidCharacters();
92
- expect(state.cursorRow).toBe(1);
93
- expect(state.cursorCol).toBe(0); // Should be on 'f'
94
-
95
- // Move left - should go back to end of previous line on 'd'
96
- state = handleVimAction(state, {
97
- type: 'vim_move_left' as const,
98
- payload: { count: 1 },
99
- });
100
- expect(state).toHaveOnlyValidCharacters();
101
- expect(state.cursorRow).toBe(0);
102
- expect(state.cursorCol).toBe(10); // Should be on 'd', not past it
103
- });
104
- });
105
-
106
- describe('vim_move_right', () => {
107
- it('should move cursor right by count', () => {
108
- const state = createTestState(['hello world'], 0, 2);
109
- const action = {
110
- type: 'vim_move_right' as const,
111
- payload: { count: 3 },
112
- };
113
-
114
- const result = handleVimAction(state, action);
115
- expect(result).toHaveOnlyValidCharacters();
116
- expect(result.cursorCol).toBe(5);
117
- });
118
-
119
- it('should not move past last character of line', () => {
120
- const state = createTestState(['hello'], 0, 3);
121
- const action = {
122
- type: 'vim_move_right' as const,
123
- payload: { count: 5 },
124
- };
125
-
126
- const result = handleVimAction(state, action);
127
- expect(result).toHaveOnlyValidCharacters();
128
- expect(result.cursorCol).toBe(4); // Last character of 'hello'
129
- });
130
-
131
- it('should wrap to next line when at end', () => {
132
- const state = createTestState(['line1', 'line2'], 0, 4); // At end of 'line1'
133
- const action = {
134
- type: 'vim_move_right' as const,
135
- payload: { count: 1 },
136
- };
137
-
138
- const result = handleVimAction(state, action);
139
- expect(result).toHaveOnlyValidCharacters();
140
- expect(result.cursorRow).toBe(1);
141
- expect(result.cursorCol).toBe(0);
142
- });
143
-
144
- it('should skip over combining marks to avoid cursor disappearing', () => {
145
- // Test case for combining character cursor disappearing bug
146
- // "café test" where é is represented as e + combining acute accent
147
- const state = createTestState(['cafe\u0301 test'], 0, 2); // Start at 'f'
148
- const action = {
149
- type: 'vim_move_right' as const,
150
- payload: { count: 1 },
151
- };
152
-
153
- const result = handleVimAction(state, action);
154
- expect(result).toHaveOnlyValidCharacters();
155
- expect(result.cursorCol).toBe(3); // Should be on 'e' of 'café'
156
-
157
- // Move right again - should skip combining mark and land on space
158
- const result2 = handleVimAction(result, action);
159
- expect(result2).toHaveOnlyValidCharacters();
160
- expect(result2.cursorCol).toBe(5); // Should be on space after 'café'
161
- });
162
- });
163
-
164
- describe('vim_move_up', () => {
165
- it('should move cursor up by count', () => {
166
- const state = createTestState(['line1', 'line2', 'line3'], 2, 3);
167
- const action = { type: 'vim_move_up' as const, payload: { count: 2 } };
168
-
169
- const result = handleVimAction(state, action);
170
- expect(result).toHaveOnlyValidCharacters();
171
- expect(result.cursorRow).toBe(0);
172
- expect(result.cursorCol).toBe(3);
173
- });
174
-
175
- it('should not move past first line', () => {
176
- const state = createTestState(['line1', 'line2'], 1, 3);
177
- const action = { type: 'vim_move_up' as const, payload: { count: 5 } };
178
-
179
- const result = handleVimAction(state, action);
180
- expect(result).toHaveOnlyValidCharacters();
181
- expect(result.cursorRow).toBe(0);
182
- });
183
-
184
- it('should adjust column for shorter lines', () => {
185
- const state = createTestState(['short', 'very long line'], 1, 10);
186
- const action = { type: 'vim_move_up' as const, payload: { count: 1 } };
187
-
188
- const result = handleVimAction(state, action);
189
- expect(result).toHaveOnlyValidCharacters();
190
- expect(result.cursorRow).toBe(0);
191
- expect(result.cursorCol).toBe(4); // Last character 't' of 'short', not past it
192
- });
193
- });
194
-
195
- describe('vim_move_down', () => {
196
- it('should move cursor down by count', () => {
197
- const state = createTestState(['line1', 'line2', 'line3'], 0, 2);
198
- const action = {
199
- type: 'vim_move_down' as const,
200
- payload: { count: 2 },
201
- };
202
-
203
- const result = handleVimAction(state, action);
204
- expect(result).toHaveOnlyValidCharacters();
205
- expect(result.cursorRow).toBe(2);
206
- expect(result.cursorCol).toBe(2);
207
- });
208
-
209
- it('should not move past last line', () => {
210
- const state = createTestState(['line1', 'line2'], 0, 2);
211
- const action = {
212
- type: 'vim_move_down' as const,
213
- payload: { count: 5 },
214
- };
215
-
216
- const result = handleVimAction(state, action);
217
- expect(result).toHaveOnlyValidCharacters();
218
- expect(result.cursorRow).toBe(1);
219
- });
220
- });
221
-
222
- describe('vim_move_word_forward', () => {
223
- it('should move to start of next word', () => {
224
- const state = createTestState(['hello world test'], 0, 0);
225
- const action = {
226
- type: 'vim_move_word_forward' as const,
227
- payload: { count: 1 },
228
- };
229
-
230
- const result = handleVimAction(state, action);
231
- expect(result).toHaveOnlyValidCharacters();
232
- expect(result.cursorCol).toBe(6); // Start of 'world'
233
- });
234
-
235
- it('should handle multiple words', () => {
236
- const state = createTestState(['hello world test'], 0, 0);
237
- const action = {
238
- type: 'vim_move_word_forward' as const,
239
- payload: { count: 2 },
240
- };
241
-
242
- const result = handleVimAction(state, action);
243
- expect(result).toHaveOnlyValidCharacters();
244
- expect(result.cursorCol).toBe(12); // Start of 'test'
245
- });
246
-
247
- it('should handle punctuation correctly', () => {
248
- const state = createTestState(['hello, world!'], 0, 0);
249
- const action = {
250
- type: 'vim_move_word_forward' as const,
251
- payload: { count: 1 },
252
- };
253
-
254
- const result = handleVimAction(state, action);
255
- expect(result).toHaveOnlyValidCharacters();
256
- expect(result.cursorCol).toBe(5); // Start of ','
257
- });
258
-
259
- it('should move across empty lines when starting from within a word', () => {
260
- // Testing the exact scenario: cursor on 'w' of 'hello world', w should move to next line
261
- const state = createTestState(['hello world', ''], 0, 6); // At 'w' of 'world'
262
- const action = {
263
- type: 'vim_move_word_forward' as const,
264
- payload: { count: 1 },
265
- };
266
-
267
- const result = handleVimAction(state, action);
268
- expect(result).toHaveOnlyValidCharacters();
269
- expect(result.cursorRow).toBe(1); // Should move to empty line
270
- expect(result.cursorCol).toBe(0); // Beginning of empty line
271
- });
272
- });
273
-
274
- describe('vim_move_word_backward', () => {
275
- it('should move to start of previous word', () => {
276
- const state = createTestState(['hello world test'], 0, 12);
277
- const action = {
278
- type: 'vim_move_word_backward' as const,
279
- payload: { count: 1 },
280
- };
281
-
282
- const result = handleVimAction(state, action);
283
- expect(result).toHaveOnlyValidCharacters();
284
- expect(result.cursorCol).toBe(6); // Start of 'world'
285
- });
286
-
287
- it('should handle multiple words', () => {
288
- const state = createTestState(['hello world test'], 0, 12);
289
- const action = {
290
- type: 'vim_move_word_backward' as const,
291
- payload: { count: 2 },
292
- };
293
-
294
- const result = handleVimAction(state, action);
295
- expect(result).toHaveOnlyValidCharacters();
296
- expect(result.cursorCol).toBe(0); // Start of 'hello'
297
- });
298
- });
299
-
300
- describe('vim_move_word_end', () => {
301
- it('should move to end of current word', () => {
302
- const state = createTestState(['hello world'], 0, 0);
303
- const action = {
304
- type: 'vim_move_word_end' as const,
305
- payload: { count: 1 },
306
- };
307
-
308
- const result = handleVimAction(state, action);
309
- expect(result).toHaveOnlyValidCharacters();
310
- expect(result.cursorCol).toBe(4); // End of 'hello'
311
- });
312
-
313
- it('should move to end of next word if already at word end', () => {
314
- const state = createTestState(['hello world'], 0, 4);
315
- const action = {
316
- type: 'vim_move_word_end' as const,
317
- payload: { count: 1 },
318
- };
319
-
320
- const result = handleVimAction(state, action);
321
- expect(result).toHaveOnlyValidCharacters();
322
- expect(result.cursorCol).toBe(10); // End of 'world'
323
- });
324
-
325
- it('should move across empty lines when at word end', () => {
326
- const state = createTestState(['hello world', '', 'test'], 0, 10); // At 'd' of 'world'
327
- const action = {
328
- type: 'vim_move_word_end' as const,
329
- payload: { count: 1 },
330
- };
331
-
332
- const result = handleVimAction(state, action);
333
- expect(result).toHaveOnlyValidCharacters();
334
- expect(result.cursorRow).toBe(2); // Should move to line with 'test'
335
- expect(result.cursorCol).toBe(3); // Should be at 't' (end of 'test')
336
- });
337
-
338
- it('should handle consecutive word-end movements across empty lines', () => {
339
- // Testing the exact scenario: cursor on 'w' of world, press 'e' twice
340
- const state = createTestState(['hello world', ''], 0, 6); // At 'w' of 'world'
341
-
342
- // First 'e' should move to 'd' of 'world'
343
- let result = handleVimAction(state, {
344
- type: 'vim_move_word_end' as const,
345
- payload: { count: 1 },
346
- });
347
- expect(result).toHaveOnlyValidCharacters();
348
- expect(result.cursorRow).toBe(0);
349
- expect(result.cursorCol).toBe(10); // At 'd' of 'world'
350
-
351
- // Second 'e' should move to the empty line (end of file in this case)
352
- result = handleVimAction(result, {
353
- type: 'vim_move_word_end' as const,
354
- payload: { count: 1 },
355
- });
356
- expect(result).toHaveOnlyValidCharacters();
357
- expect(result.cursorRow).toBe(1); // Should move to empty line
358
- expect(result.cursorCol).toBe(0); // Empty line has col 0
359
- });
360
-
361
- it('should handle combining characters - advance from end of base character', () => {
362
- // Test case for combining character word end bug
363
- // "café test" where é is represented as e + combining acute accent
364
- const state = createTestState(['cafe\u0301 test'], 0, 0); // Start at 'c'
365
-
366
- // First 'e' command should move to the 'e' (position 3)
367
- let result = handleVimAction(state, {
368
- type: 'vim_move_word_end' as const,
369
- payload: { count: 1 },
370
- });
371
- expect(result).toHaveOnlyValidCharacters();
372
- expect(result.cursorCol).toBe(3); // At 'e' of café
373
-
374
- // Second 'e' command should advance to end of "test" (position 9), not stay stuck
375
- result = handleVimAction(result, {
376
- type: 'vim_move_word_end' as const,
377
- payload: { count: 1 },
378
- });
379
- expect(result).toHaveOnlyValidCharacters();
380
- expect(result.cursorCol).toBe(9); // At 't' of "test"
381
- });
382
-
383
- it('should handle precomposed characters with diacritics', () => {
384
- // Test case with precomposed é for comparison
385
- const state = createTestState(['café test'], 0, 0); // Start at 'c'
386
-
387
- // First 'e' command should move to the 'é' (position 3)
388
- let result = handleVimAction(state, {
389
- type: 'vim_move_word_end' as const,
390
- payload: { count: 1 },
391
- });
392
- expect(result).toHaveOnlyValidCharacters();
393
- expect(result.cursorCol).toBe(3); // At 'é' of café
394
-
395
- // Second 'e' command should advance to end of "test" (position 8)
396
- result = handleVimAction(result, {
397
- type: 'vim_move_word_end' as const,
398
- payload: { count: 1 },
399
- });
400
- expect(result).toHaveOnlyValidCharacters();
401
- expect(result.cursorCol).toBe(8); // At 't' of "test"
402
- });
403
- });
404
-
405
- describe('Position commands', () => {
406
- it('vim_move_to_line_start should move to column 0', () => {
407
- const state = createTestState(['hello world'], 0, 5);
408
- const action = { type: 'vim_move_to_line_start' as const };
409
-
410
- const result = handleVimAction(state, action);
411
- expect(result).toHaveOnlyValidCharacters();
412
- expect(result.cursorCol).toBe(0);
413
- });
414
-
415
- it('vim_move_to_line_end should move to last character', () => {
416
- const state = createTestState(['hello world'], 0, 0);
417
- const action = { type: 'vim_move_to_line_end' as const };
418
-
419
- const result = handleVimAction(state, action);
420
- expect(result).toHaveOnlyValidCharacters();
421
- expect(result.cursorCol).toBe(10); // Last character of 'hello world'
422
- });
423
-
424
- it('vim_move_to_first_nonwhitespace should skip leading whitespace', () => {
425
- const state = createTestState([' hello world'], 0, 0);
426
- const action = { type: 'vim_move_to_first_nonwhitespace' as const };
427
-
428
- const result = handleVimAction(state, action);
429
- expect(result).toHaveOnlyValidCharacters();
430
- expect(result.cursorCol).toBe(3); // Position of 'h'
431
- });
432
-
433
- it('vim_move_to_first_line should move to row 0', () => {
434
- const state = createTestState(['line1', 'line2', 'line3'], 2, 5);
435
- const action = { type: 'vim_move_to_first_line' as const };
436
-
437
- const result = handleVimAction(state, action);
438
- expect(result).toHaveOnlyValidCharacters();
439
- expect(result.cursorRow).toBe(0);
440
- expect(result.cursorCol).toBe(0);
441
- });
442
-
443
- it('vim_move_to_last_line should move to last row', () => {
444
- const state = createTestState(['line1', 'line2', 'line3'], 0, 5);
445
- const action = { type: 'vim_move_to_last_line' as const };
446
-
447
- const result = handleVimAction(state, action);
448
- expect(result).toHaveOnlyValidCharacters();
449
- expect(result.cursorRow).toBe(2);
450
- expect(result.cursorCol).toBe(0);
451
- });
452
-
453
- it('vim_move_to_line should move to specific line', () => {
454
- const state = createTestState(['line1', 'line2', 'line3'], 0, 5);
455
- const action = {
456
- type: 'vim_move_to_line' as const,
457
- payload: { lineNumber: 2 },
458
- };
459
-
460
- const result = handleVimAction(state, action);
461
- expect(result).toHaveOnlyValidCharacters();
462
- expect(result.cursorRow).toBe(1); // 0-indexed
463
- expect(result.cursorCol).toBe(0);
464
- });
465
-
466
- it('vim_move_to_line should clamp to valid range', () => {
467
- const state = createTestState(['line1', 'line2'], 0, 0);
468
- const action = {
469
- type: 'vim_move_to_line' as const,
470
- payload: { lineNumber: 10 },
471
- };
472
-
473
- const result = handleVimAction(state, action);
474
- expect(result).toHaveOnlyValidCharacters();
475
- expect(result.cursorRow).toBe(1); // Last line
476
- });
477
- });
478
- });
479
-
480
- describe('Edit commands', () => {
481
- describe('vim_delete_char', () => {
482
- it('should delete single character', () => {
483
- const state = createTestState(['hello'], 0, 1);
484
- const action = {
485
- type: 'vim_delete_char' as const,
486
- payload: { count: 1 },
487
- };
488
-
489
- const result = handleVimAction(state, action);
490
- expect(result).toHaveOnlyValidCharacters();
491
- expect(result.lines[0]).toBe('hllo');
492
- expect(result.cursorCol).toBe(1);
493
- });
494
-
495
- it('should delete multiple characters', () => {
496
- const state = createTestState(['hello'], 0, 1);
497
- const action = {
498
- type: 'vim_delete_char' as const,
499
- payload: { count: 3 },
500
- };
501
-
502
- const result = handleVimAction(state, action);
503
- expect(result).toHaveOnlyValidCharacters();
504
- expect(result.lines[0]).toBe('ho');
505
- expect(result.cursorCol).toBe(1);
506
- });
507
-
508
- it('should not delete past end of line', () => {
509
- const state = createTestState(['hello'], 0, 3);
510
- const action = {
511
- type: 'vim_delete_char' as const,
512
- payload: { count: 5 },
513
- };
514
-
515
- const result = handleVimAction(state, action);
516
- expect(result).toHaveOnlyValidCharacters();
517
- expect(result.lines[0]).toBe('hel');
518
- expect(result.cursorCol).toBe(3);
519
- });
520
-
521
- it('should do nothing at end of line', () => {
522
- const state = createTestState(['hello'], 0, 5);
523
- const action = {
524
- type: 'vim_delete_char' as const,
525
- payload: { count: 1 },
526
- };
527
-
528
- const result = handleVimAction(state, action);
529
- expect(result).toHaveOnlyValidCharacters();
530
- expect(result.lines[0]).toBe('hello');
531
- expect(result.cursorCol).toBe(5);
532
- });
533
- });
534
-
535
- describe('vim_delete_word_forward', () => {
536
- it('should delete from cursor to next word start', () => {
537
- const state = createTestState(['hello world test'], 0, 0);
538
- const action = {
539
- type: 'vim_delete_word_forward' as const,
540
- payload: { count: 1 },
541
- };
542
-
543
- const result = handleVimAction(state, action);
544
- expect(result).toHaveOnlyValidCharacters();
545
- expect(result.lines[0]).toBe('world test');
546
- expect(result.cursorCol).toBe(0);
547
- });
548
-
549
- it('should delete multiple words', () => {
550
- const state = createTestState(['hello world test'], 0, 0);
551
- const action = {
552
- type: 'vim_delete_word_forward' as const,
553
- payload: { count: 2 },
554
- };
555
-
556
- const result = handleVimAction(state, action);
557
- expect(result).toHaveOnlyValidCharacters();
558
- expect(result.lines[0]).toBe('test');
559
- expect(result.cursorCol).toBe(0);
560
- });
561
-
562
- it('should delete to end if no more words', () => {
563
- const state = createTestState(['hello world'], 0, 6);
564
- const action = {
565
- type: 'vim_delete_word_forward' as const,
566
- payload: { count: 2 },
567
- };
568
-
569
- const result = handleVimAction(state, action);
570
- expect(result).toHaveOnlyValidCharacters();
571
- expect(result.lines[0]).toBe('hello ');
572
- expect(result.cursorCol).toBe(6);
573
- });
574
- });
575
-
576
- describe('vim_delete_word_backward', () => {
577
- it('should delete from cursor to previous word start', () => {
578
- const state = createTestState(['hello world test'], 0, 12);
579
- const action = {
580
- type: 'vim_delete_word_backward' as const,
581
- payload: { count: 1 },
582
- };
583
-
584
- const result = handleVimAction(state, action);
585
- expect(result).toHaveOnlyValidCharacters();
586
- expect(result.lines[0]).toBe('hello test');
587
- expect(result.cursorCol).toBe(6);
588
- });
589
-
590
- it('should delete multiple words backward', () => {
591
- const state = createTestState(['hello world test'], 0, 12);
592
- const action = {
593
- type: 'vim_delete_word_backward' as const,
594
- payload: { count: 2 },
595
- };
596
-
597
- const result = handleVimAction(state, action);
598
- expect(result).toHaveOnlyValidCharacters();
599
- expect(result.lines[0]).toBe('test');
600
- expect(result.cursorCol).toBe(0);
601
- });
602
- });
603
-
604
- describe('vim_delete_line', () => {
605
- it('should delete current line', () => {
606
- const state = createTestState(['line1', 'line2', 'line3'], 1, 2);
607
- const action = {
608
- type: 'vim_delete_line' as const,
609
- payload: { count: 1 },
610
- };
611
-
612
- const result = handleVimAction(state, action);
613
- expect(result).toHaveOnlyValidCharacters();
614
- expect(result.lines).toEqual(['line1', 'line3']);
615
- expect(result.cursorRow).toBe(1);
616
- expect(result.cursorCol).toBe(0);
617
- });
618
-
619
- it('should delete multiple lines', () => {
620
- const state = createTestState(['line1', 'line2', 'line3'], 0, 2);
621
- const action = {
622
- type: 'vim_delete_line' as const,
623
- payload: { count: 2 },
624
- };
625
-
626
- const result = handleVimAction(state, action);
627
- expect(result).toHaveOnlyValidCharacters();
628
- expect(result.lines).toEqual(['line3']);
629
- expect(result.cursorRow).toBe(0);
630
- expect(result.cursorCol).toBe(0);
631
- });
632
-
633
- it('should leave empty line when deleting all lines', () => {
634
- const state = createTestState(['only line'], 0, 0);
635
- const action = {
636
- type: 'vim_delete_line' as const,
637
- payload: { count: 1 },
638
- };
639
-
640
- const result = handleVimAction(state, action);
641
- expect(result).toHaveOnlyValidCharacters();
642
- expect(result.lines).toEqual(['']);
643
- expect(result.cursorRow).toBe(0);
644
- expect(result.cursorCol).toBe(0);
645
- });
646
- });
647
-
648
- describe('vim_delete_to_end_of_line', () => {
649
- it('should delete from cursor to end of line', () => {
650
- const state = createTestState(['hello world'], 0, 5);
651
- const action = { type: 'vim_delete_to_end_of_line' as const };
652
-
653
- const result = handleVimAction(state, action);
654
- expect(result).toHaveOnlyValidCharacters();
655
- expect(result.lines[0]).toBe('hello');
656
- expect(result.cursorCol).toBe(5);
657
- });
658
-
659
- it('should do nothing at end of line', () => {
660
- const state = createTestState(['hello'], 0, 5);
661
- const action = { type: 'vim_delete_to_end_of_line' as const };
662
-
663
- const result = handleVimAction(state, action);
664
- expect(result).toHaveOnlyValidCharacters();
665
- expect(result.lines[0]).toBe('hello');
666
- });
667
- });
668
- });
669
-
670
- describe('Insert mode commands', () => {
671
- describe('vim_insert_at_cursor', () => {
672
- it('should not change cursor position', () => {
673
- const state = createTestState(['hello'], 0, 2);
674
- const action = { type: 'vim_insert_at_cursor' as const };
675
-
676
- const result = handleVimAction(state, action);
677
- expect(result).toHaveOnlyValidCharacters();
678
- expect(result.cursorRow).toBe(0);
679
- expect(result.cursorCol).toBe(2);
680
- });
681
- });
682
-
683
- describe('vim_append_at_cursor', () => {
684
- it('should move cursor right by one', () => {
685
- const state = createTestState(['hello'], 0, 2);
686
- const action = { type: 'vim_append_at_cursor' as const };
687
-
688
- const result = handleVimAction(state, action);
689
- expect(result).toHaveOnlyValidCharacters();
690
- expect(result.cursorCol).toBe(3);
691
- });
692
-
693
- it('should not move past end of line', () => {
694
- const state = createTestState(['hello'], 0, 5);
695
- const action = { type: 'vim_append_at_cursor' as const };
696
-
697
- const result = handleVimAction(state, action);
698
- expect(result).toHaveOnlyValidCharacters();
699
- expect(result.cursorCol).toBe(5);
700
- });
701
- });
702
-
703
- describe('vim_append_at_line_end', () => {
704
- it('should move cursor to end of line', () => {
705
- const state = createTestState(['hello world'], 0, 3);
706
- const action = { type: 'vim_append_at_line_end' as const };
707
-
708
- const result = handleVimAction(state, action);
709
- expect(result).toHaveOnlyValidCharacters();
710
- expect(result.cursorCol).toBe(11);
711
- });
712
- });
713
-
714
- describe('vim_insert_at_line_start', () => {
715
- it('should move to first non-whitespace character', () => {
716
- const state = createTestState([' hello world'], 0, 5);
717
- const action = { type: 'vim_insert_at_line_start' as const };
718
-
719
- const result = handleVimAction(state, action);
720
- expect(result).toHaveOnlyValidCharacters();
721
- expect(result.cursorCol).toBe(2);
722
- });
723
-
724
- it('should move to column 0 for line with only whitespace', () => {
725
- const state = createTestState([' '], 0, 1);
726
- const action = { type: 'vim_insert_at_line_start' as const };
727
-
728
- const result = handleVimAction(state, action);
729
- expect(result).toHaveOnlyValidCharacters();
730
- expect(result.cursorCol).toBe(3);
731
- });
732
- });
733
-
734
- describe('vim_open_line_below', () => {
735
- it('should insert a new line below the current one', () => {
736
- const state = createTestState(['hello world'], 0, 5);
737
- const action = { type: 'vim_open_line_below' as const };
738
-
739
- const result = handleVimAction(state, action);
740
- expect(result).toHaveOnlyValidCharacters();
741
- expect(result.lines).toEqual(['hello world', '']);
742
- expect(result.cursorRow).toBe(1);
743
- expect(result.cursorCol).toBe(0);
744
- });
745
- });
746
-
747
- describe('vim_open_line_above', () => {
748
- it('should insert a new line above the current one', () => {
749
- const state = createTestState(['hello', 'world'], 1, 2);
750
- const action = { type: 'vim_open_line_above' as const };
751
-
752
- const result = handleVimAction(state, action);
753
- expect(result).toHaveOnlyValidCharacters();
754
- expect(result.lines).toEqual(['hello', '', 'world']);
755
- expect(result.cursorRow).toBe(1);
756
- expect(result.cursorCol).toBe(0);
757
- });
758
- });
759
-
760
- describe('vim_escape_insert_mode', () => {
761
- it('should move cursor left', () => {
762
- const state = createTestState(['hello'], 0, 3);
763
- const action = { type: 'vim_escape_insert_mode' as const };
764
-
765
- const result = handleVimAction(state, action);
766
- expect(result).toHaveOnlyValidCharacters();
767
- expect(result.cursorCol).toBe(2);
768
- });
769
-
770
- it('should not move past beginning of line', () => {
771
- const state = createTestState(['hello'], 0, 0);
772
- const action = { type: 'vim_escape_insert_mode' as const };
773
-
774
- const result = handleVimAction(state, action);
775
- expect(result).toHaveOnlyValidCharacters();
776
- expect(result.cursorCol).toBe(0);
777
- });
778
- });
779
- });
780
-
781
- describe('Change commands', () => {
782
- describe('vim_change_word_forward', () => {
783
- it('should delete from cursor to next word start', () => {
784
- const state = createTestState(['hello world test'], 0, 0);
785
- const action = {
786
- type: 'vim_change_word_forward' as const,
787
- payload: { count: 1 },
788
- };
789
-
790
- const result = handleVimAction(state, action);
791
- expect(result).toHaveOnlyValidCharacters();
792
- expect(result.lines[0]).toBe('world test');
793
- expect(result.cursorCol).toBe(0);
794
- });
795
- });
796
-
797
- describe('vim_change_line', () => {
798
- it('should delete entire line content', () => {
799
- const state = createTestState(['hello world'], 0, 5);
800
- const action = {
801
- type: 'vim_change_line' as const,
802
- payload: { count: 1 },
803
- };
804
-
805
- const result = handleVimAction(state, action);
806
- expect(result).toHaveOnlyValidCharacters();
807
- expect(result.lines[0]).toBe('');
808
- expect(result.cursorCol).toBe(0);
809
- });
810
- });
811
-
812
- describe('vim_change_movement', () => {
813
- it('should change characters to the left', () => {
814
- const state = createTestState(['hello world'], 0, 5);
815
- const action = {
816
- type: 'vim_change_movement' as const,
817
- payload: { movement: 'h', count: 2 },
818
- };
819
-
820
- const result = handleVimAction(state, action);
821
- expect(result).toHaveOnlyValidCharacters();
822
- expect(result.lines[0]).toBe('hel world');
823
- expect(result.cursorCol).toBe(3);
824
- });
825
-
826
- it('should change characters to the right', () => {
827
- const state = createTestState(['hello world'], 0, 5);
828
- const action = {
829
- type: 'vim_change_movement' as const,
830
- payload: { movement: 'l', count: 3 },
831
- };
832
-
833
- const result = handleVimAction(state, action);
834
- expect(result).toHaveOnlyValidCharacters();
835
- expect(result.lines[0]).toBe('hellorld'); // Deletes ' wo' (3 chars to the right)
836
- expect(result.cursorCol).toBe(5);
837
- });
838
-
839
- it('should change multiple lines down', () => {
840
- const state = createTestState(['line1', 'line2', 'line3'], 0, 2);
841
- const action = {
842
- type: 'vim_change_movement' as const,
843
- payload: { movement: 'j', count: 2 },
844
- };
845
-
846
- const result = handleVimAction(state, action);
847
- expect(result).toHaveOnlyValidCharacters();
848
- // The movement 'j' with count 2 changes 2 lines starting from cursor row
849
- // Since we're at cursor position 2, it changes lines starting from current row
850
- expect(result.lines).toEqual(['line1', 'line2', 'line3']); // No change because count > available lines
851
- expect(result.cursorRow).toBe(0);
852
- expect(result.cursorCol).toBe(2);
853
- });
854
- });
855
- });
856
-
857
- describe('Edge cases', () => {
858
- it('should handle empty text', () => {
859
- const state = createTestState([''], 0, 0);
860
- const action = {
861
- type: 'vim_move_word_forward' as const,
862
- payload: { count: 1 },
863
- };
864
-
865
- const result = handleVimAction(state, action);
866
- expect(result).toHaveOnlyValidCharacters();
867
- expect(result.cursorRow).toBe(0);
868
- expect(result.cursorCol).toBe(0);
869
- });
870
-
871
- it('should handle single character line', () => {
872
- const state = createTestState(['a'], 0, 0);
873
- const action = { type: 'vim_move_to_line_end' as const };
874
-
875
- const result = handleVimAction(state, action);
876
- expect(result).toHaveOnlyValidCharacters();
877
- expect(result.cursorCol).toBe(0); // Should be last character position
878
- });
879
-
880
- it('should handle empty lines in multi-line text', () => {
881
- const state = createTestState(['line1', '', 'line3'], 1, 0);
882
- const action = {
883
- type: 'vim_move_word_forward' as const,
884
- payload: { count: 1 },
885
- };
886
-
887
- const result = handleVimAction(state, action);
888
- expect(result).toHaveOnlyValidCharacters();
889
- // Should move to next line with content
890
- expect(result.cursorRow).toBe(2);
891
- expect(result.cursorCol).toBe(0);
892
- });
893
-
894
- it('should preserve undo stack in operations', () => {
895
- const state = createTestState(['hello'], 0, 0);
896
- state.undoStack = [{ lines: ['previous'], cursorRow: 0, cursorCol: 0 }];
897
-
898
- const action = {
899
- type: 'vim_delete_char' as const,
900
- payload: { count: 1 },
901
- };
902
-
903
- const result = handleVimAction(state, action);
904
- expect(result).toHaveOnlyValidCharacters();
905
- expect(result.undoStack).toHaveLength(2); // Original plus new snapshot
906
- });
907
- });
908
-
909
- describe('UTF-32 character handling in word/line operations', () => {
910
- describe('Right-to-left text handling', () => {
911
- it('should handle Arabic text in word movements', () => {
912
- const state = createTestState(['hello مرحبا world'], 0, 0);
913
-
914
- // Move to end of 'hello'
915
- let result = handleVimAction(state, {
916
- type: 'vim_move_word_end' as const,
917
- payload: { count: 1 },
918
- });
919
- expect(result).toHaveOnlyValidCharacters();
920
- expect(result.cursorCol).toBe(4); // End of 'hello'
921
-
922
- // Move to end of Arabic word
923
- result = handleVimAction(result, {
924
- type: 'vim_move_word_end' as const,
925
- payload: { count: 1 },
926
- });
927
- expect(result).toHaveOnlyValidCharacters();
928
- expect(result.cursorCol).toBe(10); // End of Arabic word 'مرحبا'
929
- });
930
- });
931
-
932
- describe('Chinese character handling', () => {
933
- it('should handle Chinese characters in word movements', () => {
934
- const state = createTestState(['hello 你好 world'], 0, 0);
935
-
936
- // Move to end of 'hello'
937
- let result = handleVimAction(state, {
938
- type: 'vim_move_word_end' as const,
939
- payload: { count: 1 },
940
- });
941
- expect(result).toHaveOnlyValidCharacters();
942
- expect(result.cursorCol).toBe(4); // End of 'hello'
943
-
944
- // Move forward to start of 'world'
945
- result = handleVimAction(result, {
946
- type: 'vim_move_word_forward' as const,
947
- payload: { count: 1 },
948
- });
949
- expect(result).toHaveOnlyValidCharacters();
950
- expect(result.cursorCol).toBe(6); // Start of '你好'
951
- });
952
- });
953
-
954
- describe('Mixed script handling', () => {
955
- it('should handle mixed Latin and non-Latin scripts with word end commands', () => {
956
- const state = createTestState(['test中文test'], 0, 0);
957
-
958
- let result = handleVimAction(state, {
959
- type: 'vim_move_word_end' as const,
960
- payload: { count: 1 },
961
- });
962
- expect(result).toHaveOnlyValidCharacters();
963
- expect(result.cursorCol).toBe(3); // End of 'test'
964
-
965
- // Second word end command should move to end of '中文'
966
- result = handleVimAction(result, {
967
- type: 'vim_move_word_end' as const,
968
- payload: { count: 1 },
969
- });
970
- expect(result).toHaveOnlyValidCharacters();
971
- expect(result.cursorCol).toBe(5); // End of '中文'
972
- });
973
-
974
- it('should handle mixed Latin and non-Latin scripts with word forward commands', () => {
975
- const state = createTestState(['test中文test'], 0, 0);
976
-
977
- let result = handleVimAction(state, {
978
- type: 'vim_move_word_forward' as const,
979
- payload: { count: 1 },
980
- });
981
- expect(result).toHaveOnlyValidCharacters();
982
- expect(result.cursorCol).toBe(4); // Start of '中'
983
-
984
- // Second word forward command should move to start of final 'test'
985
- result = handleVimAction(result, {
986
- type: 'vim_move_word_forward' as const,
987
- payload: { count: 1 },
988
- });
989
- expect(result).toHaveOnlyValidCharacters();
990
- expect(result.cursorCol).toBe(6); // Start of final 'test'
991
- });
992
-
993
- it('should handle mixed Latin and non-Latin scripts with word backward commands', () => {
994
- const state = createTestState(['test中文test'], 0, 9); // Start at end of final 'test'
995
-
996
- let result = handleVimAction(state, {
997
- type: 'vim_move_word_backward' as const,
998
- payload: { count: 1 },
999
- });
1000
- expect(result).toHaveOnlyValidCharacters();
1001
- expect(result.cursorCol).toBe(6); // Start of final 'test'
1002
-
1003
- // Second word backward command should move to start of '中文'
1004
- result = handleVimAction(result, {
1005
- type: 'vim_move_word_backward' as const,
1006
- payload: { count: 1 },
1007
- });
1008
- expect(result).toHaveOnlyValidCharacters();
1009
- expect(result.cursorCol).toBe(4); // Start of '中'
1010
- });
1011
-
1012
- it('should handle Unicode block characters consistently with w and e commands', () => {
1013
- const state = createTestState(['██ █████ ██'], 0, 0);
1014
-
1015
- // Test w command progression
1016
- let wResult = handleVimAction(state, {
1017
- type: 'vim_move_word_forward' as const,
1018
- payload: { count: 1 },
1019
- });
1020
- expect(wResult).toHaveOnlyValidCharacters();
1021
- expect(wResult.cursorCol).toBe(3); // Start of second block sequence
1022
-
1023
- wResult = handleVimAction(wResult, {
1024
- type: 'vim_move_word_forward' as const,
1025
- payload: { count: 1 },
1026
- });
1027
- expect(wResult).toHaveOnlyValidCharacters();
1028
- expect(wResult.cursorCol).toBe(9); // Start of third block sequence
1029
-
1030
- // Test e command progression from beginning
1031
- let eResult = handleVimAction(state, {
1032
- type: 'vim_move_word_end' as const,
1033
- payload: { count: 1 },
1034
- });
1035
- expect(eResult).toHaveOnlyValidCharacters();
1036
- expect(eResult.cursorCol).toBe(1); // End of first block sequence
1037
-
1038
- eResult = handleVimAction(eResult, {
1039
- type: 'vim_move_word_end' as const,
1040
- payload: { count: 1 },
1041
- });
1042
- expect(eResult).toHaveOnlyValidCharacters();
1043
- expect(eResult.cursorCol).toBe(7); // End of second block sequence
1044
-
1045
- eResult = handleVimAction(eResult, {
1046
- type: 'vim_move_word_end' as const,
1047
- payload: { count: 1 },
1048
- });
1049
- expect(eResult).toHaveOnlyValidCharacters();
1050
- expect(eResult.cursorCol).toBe(10); // End of third block sequence
1051
- });
1052
-
1053
- it('should handle strings starting with Chinese characters', () => {
1054
- const state = createTestState(['中文test英文word'], 0, 0);
1055
-
1056
- // Test 'w' command - when at start of non-Latin word, w moves to next word
1057
- let wResult = handleVimAction(state, {
1058
- type: 'vim_move_word_forward' as const,
1059
- payload: { count: 1 },
1060
- });
1061
- expect(wResult).toHaveOnlyValidCharacters();
1062
- expect(wResult.cursorCol).toBe(2); // Start of 'test'
1063
-
1064
- wResult = handleVimAction(wResult, {
1065
- type: 'vim_move_word_forward' as const,
1066
- payload: { count: 1 },
1067
- });
1068
- expect(wResult.cursorCol).toBe(6); // Start of '英文'
1069
-
1070
- // Test 'e' command
1071
- let eResult = handleVimAction(state, {
1072
- type: 'vim_move_word_end' as const,
1073
- payload: { count: 1 },
1074
- });
1075
- expect(eResult).toHaveOnlyValidCharacters();
1076
- expect(eResult.cursorCol).toBe(1); // End of 中文
1077
-
1078
- eResult = handleVimAction(eResult, {
1079
- type: 'vim_move_word_end' as const,
1080
- payload: { count: 1 },
1081
- });
1082
- expect(eResult.cursorCol).toBe(5); // End of test
1083
- });
1084
-
1085
- it('should handle strings starting with Arabic characters', () => {
1086
- const state = createTestState(['مرحباhelloسلام'], 0, 0);
1087
-
1088
- // Test 'w' command - when at start of non-Latin word, w moves to next word
1089
- let wResult = handleVimAction(state, {
1090
- type: 'vim_move_word_forward' as const,
1091
- payload: { count: 1 },
1092
- });
1093
- expect(wResult).toHaveOnlyValidCharacters();
1094
- expect(wResult.cursorCol).toBe(5); // Start of 'hello'
1095
-
1096
- wResult = handleVimAction(wResult, {
1097
- type: 'vim_move_word_forward' as const,
1098
- payload: { count: 1 },
1099
- });
1100
- expect(wResult.cursorCol).toBe(10); // Start of 'سلام'
1101
-
1102
- // Test 'b' command from end
1103
- const bState = createTestState(['مرحباhelloسلام'], 0, 13);
1104
- let bResult = handleVimAction(bState, {
1105
- type: 'vim_move_word_backward' as const,
1106
- payload: { count: 1 },
1107
- });
1108
- expect(bResult).toHaveOnlyValidCharacters();
1109
- expect(bResult.cursorCol).toBe(10); // Start of سلام
1110
-
1111
- bResult = handleVimAction(bResult, {
1112
- type: 'vim_move_word_backward' as const,
1113
- payload: { count: 1 },
1114
- });
1115
- expect(bResult.cursorCol).toBe(5); // Start of hello
1116
- });
1117
- });
1118
- });
1119
- });