claude-code-workflow 6.1.4 → 6.2.2

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 (437) hide show
  1. package/.claude/CLAUDE.md +10 -0
  2. package/.claude/agents/action-planning-agent.md +857 -778
  3. package/.claude/agents/cli-execution-agent.md +266 -269
  4. package/.claude/agents/cli-explore-agent.md +2 -2
  5. package/.claude/agents/cli-lite-planning-agent.md +142 -92
  6. package/.claude/agents/cli-planning-agent.md +4 -4
  7. package/.claude/agents/code-developer.md +7 -6
  8. package/.claude/agents/conceptual-planning-agent.md +2 -2
  9. package/.claude/agents/context-search-agent.md +31 -32
  10. package/.claude/agents/doc-generator.md +4 -4
  11. package/.claude/agents/memory-bridge.md +93 -93
  12. package/.claude/agents/test-context-search-agent.md +8 -7
  13. package/.claude/agents/test-fix-agent.md +7 -6
  14. package/.claude/commands/clean.md +516 -0
  15. package/.claude/commands/memory/compact.md +383 -0
  16. package/.claude/commands/memory/docs-full-cli.md +471 -471
  17. package/.claude/commands/memory/docs-related-cli.md +386 -386
  18. package/.claude/commands/memory/docs.md +615 -615
  19. package/.claude/commands/memory/load.md +5 -5
  20. package/.claude/commands/memory/tech-research-rules.md +310 -0
  21. package/.claude/commands/memory/update-full.md +332 -332
  22. package/.claude/commands/memory/workflow-skill-memory.md +4 -4
  23. package/.claude/commands/task/create.md +151 -151
  24. package/.claude/commands/version.md +254 -254
  25. package/.claude/commands/workflow/brainstorm/api-designer.md +587 -585
  26. package/.claude/commands/workflow/brainstorm/artifacts.md +1 -0
  27. package/.claude/commands/workflow/brainstorm/auto-parallel.md +443 -443
  28. package/.claude/commands/workflow/brainstorm/data-architect.md +220 -220
  29. package/.claude/commands/workflow/brainstorm/product-manager.md +200 -200
  30. package/.claude/commands/workflow/brainstorm/product-owner.md +200 -200
  31. package/.claude/commands/workflow/brainstorm/scrum-master.md +200 -200
  32. package/.claude/commands/workflow/brainstorm/subject-matter-expert.md +200 -200
  33. package/.claude/commands/workflow/brainstorm/system-architect.md +389 -387
  34. package/.claude/commands/workflow/brainstorm/ui-designer.md +221 -221
  35. package/.claude/commands/workflow/brainstorm/ux-expert.md +221 -221
  36. package/.claude/commands/workflow/debug.md +321 -0
  37. package/.claude/commands/workflow/execute.md +13 -0
  38. package/.claude/commands/workflow/init.md +165 -164
  39. package/.claude/commands/workflow/lite-execute.md +119 -13
  40. package/.claude/commands/workflow/lite-fix.md +623 -621
  41. package/.claude/commands/workflow/lite-plan.md +610 -592
  42. package/.claude/commands/workflow/plan.md +5 -5
  43. package/.claude/commands/workflow/review-module-cycle.md +2 -0
  44. package/.claude/commands/workflow/review-session-cycle.md +2 -0
  45. package/.claude/commands/workflow/review.md +297 -291
  46. package/.claude/commands/workflow/session/complete.md +153 -500
  47. package/.claude/commands/workflow/session/list.md +95 -95
  48. package/.claude/commands/workflow/session/resume.md +60 -60
  49. package/.claude/commands/workflow/session/start.md +199 -199
  50. package/.claude/commands/workflow/tdd-plan.md +3 -3
  51. package/.claude/commands/workflow/tdd-verify.md +23 -9
  52. package/.claude/commands/workflow/test-cycle-execute.md +2 -0
  53. package/.claude/commands/workflow/test-fix-gen.md +699 -699
  54. package/.claude/commands/workflow/tools/conflict-resolution.md +104 -18
  55. package/.claude/commands/workflow/tools/context-gather.md +436 -434
  56. package/.claude/commands/workflow/tools/task-generate-agent.md +490 -291
  57. package/.claude/commands/workflow/tools/task-generate-tdd.md +18 -10
  58. package/.claude/commands/workflow/tools/test-concept-enhanced.md +2 -1
  59. package/.claude/commands/workflow/tools/test-context-gather.md +1 -0
  60. package/.claude/commands/workflow/tools/test-task-generate.md +1 -0
  61. package/.claude/commands/workflow/ui-design/import-from-code.md +9 -6
  62. package/.claude/skills/command-guide/SKILL.md +5 -5
  63. package/.claude/skills/command-guide/index/all-commands.json +1 -1
  64. package/.claude/skills/command-guide/index/by-category.json +1 -1
  65. package/.claude/skills/command-guide/index/by-use-case.json +1 -1
  66. package/.claude/skills/command-guide/reference/agents/action-planning-agent.md +857 -778
  67. package/.claude/skills/command-guide/reference/agents/cli-execution-agent.md +266 -269
  68. package/.claude/skills/command-guide/reference/agents/cli-explore-agent.md +2 -2
  69. package/.claude/skills/command-guide/reference/agents/cli-lite-planning-agent.md +142 -92
  70. package/.claude/skills/command-guide/reference/agents/cli-planning-agent.md +4 -4
  71. package/.claude/skills/command-guide/reference/agents/code-developer.md +7 -6
  72. package/.claude/skills/command-guide/reference/agents/conceptual-planning-agent.md +2 -2
  73. package/.claude/skills/command-guide/reference/agents/context-search-agent.md +31 -32
  74. package/.claude/skills/command-guide/reference/agents/doc-generator.md +4 -4
  75. package/.claude/skills/command-guide/reference/agents/memory-bridge.md +93 -93
  76. package/.claude/skills/command-guide/reference/agents/test-context-search-agent.md +8 -7
  77. package/.claude/skills/command-guide/reference/agents/test-fix-agent.md +7 -6
  78. package/.claude/skills/command-guide/reference/commands/memory/docs-full-cli.md +471 -471
  79. package/.claude/skills/command-guide/reference/commands/memory/docs-related-cli.md +386 -386
  80. package/.claude/skills/command-guide/reference/commands/memory/docs.md +17 -16
  81. package/.claude/skills/command-guide/reference/commands/memory/load.md +5 -5
  82. package/.claude/skills/command-guide/reference/commands/memory/tech-research.md +194 -357
  83. package/.claude/skills/command-guide/reference/commands/memory/update-full.md +332 -332
  84. package/.claude/skills/command-guide/reference/commands/memory/workflow-skill-memory.md +4 -4
  85. package/.claude/skills/command-guide/reference/commands/task/create.md +151 -151
  86. package/.claude/skills/command-guide/reference/commands/version.md +254 -254
  87. package/.claude/skills/command-guide/reference/commands/workflow/brainstorm/api-designer.md +585 -585
  88. package/.claude/skills/command-guide/reference/commands/workflow/brainstorm/auto-parallel.md +443 -443
  89. package/.claude/skills/command-guide/reference/commands/workflow/brainstorm/data-architect.md +220 -220
  90. package/.claude/skills/command-guide/reference/commands/workflow/brainstorm/product-manager.md +200 -200
  91. package/.claude/skills/command-guide/reference/commands/workflow/brainstorm/product-owner.md +200 -200
  92. package/.claude/skills/command-guide/reference/commands/workflow/brainstorm/scrum-master.md +200 -200
  93. package/.claude/skills/command-guide/reference/commands/workflow/brainstorm/subject-matter-expert.md +200 -200
  94. package/.claude/skills/command-guide/reference/commands/workflow/brainstorm/system-architect.md +387 -387
  95. package/.claude/skills/command-guide/reference/commands/workflow/brainstorm/ui-designer.md +221 -221
  96. package/.claude/skills/command-guide/reference/commands/workflow/brainstorm/ux-expert.md +221 -221
  97. package/.claude/skills/command-guide/reference/commands/workflow/execute.md +25 -20
  98. package/.claude/skills/command-guide/reference/commands/workflow/init.md +164 -164
  99. package/.claude/skills/command-guide/reference/commands/workflow/lite-execute.md +748 -686
  100. package/.claude/skills/command-guide/reference/commands/workflow/lite-fix.md +664 -621
  101. package/.claude/skills/command-guide/reference/commands/workflow/lite-plan.md +645 -592
  102. package/.claude/skills/command-guide/reference/commands/workflow/plan.md +5 -5
  103. package/.claude/skills/command-guide/reference/commands/workflow/review.md +25 -18
  104. package/.claude/skills/command-guide/reference/commands/workflow/session/complete.md +547 -500
  105. package/.claude/skills/command-guide/reference/commands/workflow/session/list.md +45 -27
  106. package/.claude/skills/command-guide/reference/commands/workflow/session/resume.md +35 -19
  107. package/.claude/skills/command-guide/reference/commands/workflow/session/start.md +90 -33
  108. package/.claude/skills/command-guide/reference/commands/workflow/tdd-plan.md +3 -3
  109. package/.claude/skills/command-guide/reference/commands/workflow/tdd-verify.md +23 -9
  110. package/.claude/skills/command-guide/reference/commands/workflow/test-fix-gen.md +699 -699
  111. package/.claude/skills/command-guide/reference/commands/workflow/tools/conflict-resolution.md +103 -17
  112. package/.claude/skills/command-guide/reference/commands/workflow/tools/context-gather.md +434 -434
  113. package/.claude/skills/command-guide/reference/commands/workflow/tools/task-generate-agent.md +487 -291
  114. package/.claude/skills/command-guide/reference/commands/workflow/tools/task-generate-tdd.md +17 -10
  115. package/.claude/skills/command-guide/reference/commands/workflow/tools/test-concept-enhanced.md +1 -1
  116. package/.claude/skills/command-guide/reference/commands/workflow/ui-design/import-from-code.md +6 -6
  117. package/.claude/workflows/chinese-response.md +38 -0
  118. package/.claude/workflows/cli-templates/prompts/rules/rule-api.txt +122 -0
  119. package/.claude/workflows/cli-templates/prompts/rules/rule-components.txt +122 -0
  120. package/.claude/workflows/cli-templates/prompts/rules/rule-config.txt +89 -0
  121. package/.claude/workflows/cli-templates/prompts/rules/rule-core.txt +60 -0
  122. package/.claude/workflows/cli-templates/prompts/rules/rule-patterns.txt +70 -0
  123. package/.claude/workflows/cli-templates/prompts/rules/rule-testing.txt +81 -0
  124. package/.claude/workflows/cli-templates/prompts/rules/tech-rules-agent-prompt.txt +89 -0
  125. package/.claude/workflows/cli-templates/prompts/workflow/gemini-solution-design.txt +131 -131
  126. package/.claude/workflows/cli-templates/prompts/workflow/skill-conflict-patterns.txt +5 -9
  127. package/.claude/workflows/cli-templates/prompts/workflow/skill-lessons-learned.txt +5 -9
  128. package/.claude/workflows/cli-templates/protocols/analysis-protocol.md +112 -0
  129. package/.claude/workflows/cli-templates/protocols/write-protocol.md +201 -0
  130. package/.claude/workflows/cli-templates/schemas/conflict-resolution-schema.json +137 -0
  131. package/.claude/workflows/cli-templates/schemas/debug-log-json-schema.json +127 -0
  132. package/.claude/workflows/cli-templates/schemas/fix-plan-json-schema.json +25 -0
  133. package/.claude/workflows/cli-templates/schemas/plan-json-schema.json +25 -0
  134. package/.claude/workflows/cli-tools-usage.md +526 -0
  135. package/{CLAUDE.md → .claude/workflows/coding-philosophy.md} +24 -45
  136. package/.claude/workflows/context-tools.md +84 -0
  137. package/.claude/workflows/file-modification.md +64 -0
  138. package/.claude/workflows/tool-strategy.md +216 -79
  139. package/.claude/workflows/windows-platform.md +16 -0
  140. package/.claude/workflows/workflow-architecture.md +942 -942
  141. package/.codex/AGENTS.md +63 -330
  142. package/.codex/prompts/debug.md +318 -0
  143. package/.codex/prompts/execute.md +273 -0
  144. package/.codex/prompts/lite-execute.md +164 -0
  145. package/.codex/prompts/lite-plan.md +469 -0
  146. package/.codex/prompts.zip +0 -0
  147. package/.gemini/GEMINI.md +25 -164
  148. package/.qwen/QWEN.md +0 -139
  149. package/README.md +29 -9
  150. package/ccw/README.md +30 -6
  151. package/ccw/bin/ccw-mcp.js +7 -0
  152. package/ccw/bin/ccw.js +9 -9
  153. package/ccw/package.json +65 -47
  154. package/ccw/src/.workflow/.cli-history/history.db +0 -0
  155. package/ccw/src/.workflow/.cli-history/history.db-shm +0 -0
  156. package/ccw/src/.workflow/.cli-history/history.db-wal +0 -0
  157. package/ccw/src/cli.ts +244 -0
  158. package/ccw/src/commands/cli.ts +740 -0
  159. package/ccw/src/commands/core-memory.ts +770 -0
  160. package/ccw/src/commands/hook.ts +315 -0
  161. package/ccw/src/commands/install.ts +519 -0
  162. package/ccw/src/commands/{list.js → list.ts} +1 -1
  163. package/ccw/src/commands/memory.ts +1090 -0
  164. package/ccw/src/commands/{serve.js → serve.ts} +14 -5
  165. package/ccw/src/commands/session-path-resolver.ts +372 -0
  166. package/ccw/src/commands/session.ts +1141 -0
  167. package/ccw/src/commands/{stop.js → stop.ts} +16 -6
  168. package/ccw/src/commands/tool.ts +201 -0
  169. package/ccw/src/commands/{uninstall.js → uninstall.ts} +89 -40
  170. package/ccw/src/commands/{upgrade.js → upgrade.ts} +68 -23
  171. package/ccw/src/commands/{view.js → view.ts} +22 -8
  172. package/ccw/src/config/storage-paths.ts +670 -0
  173. package/ccw/src/core/cache-manager.ts +294 -0
  174. package/ccw/src/core/claude-freshness.ts +319 -0
  175. package/ccw/src/core/core-memory-store.ts +1528 -0
  176. package/ccw/src/core/{dashboard-generator-patch.js → dashboard-generator-patch.ts} +18 -0
  177. package/ccw/src/core/{dashboard-generator.js → dashboard-generator.ts} +69 -12
  178. package/ccw/src/core/data-aggregator.ts +584 -0
  179. package/ccw/src/core/history-importer.ts +625 -0
  180. package/ccw/src/core/{lite-scanner.js → lite-scanner-complete.ts} +162 -66
  181. package/ccw/src/core/lite-scanner.ts +469 -0
  182. package/ccw/src/core/{manifest.js → manifest.ts} +104 -34
  183. package/ccw/src/core/memory-embedder-bridge.ts +262 -0
  184. package/ccw/src/core/memory-store.ts +978 -0
  185. package/ccw/src/core/routes/ccw-routes.ts +96 -0
  186. package/ccw/src/core/routes/claude-routes.ts +1183 -0
  187. package/ccw/src/core/routes/cli-routes.ts +561 -0
  188. package/ccw/src/core/routes/codexlens-routes.ts +806 -0
  189. package/ccw/src/core/routes/core-memory-routes.ts +605 -0
  190. package/ccw/src/core/routes/files-routes.ts +428 -0
  191. package/ccw/src/core/routes/graph-routes.md +164 -0
  192. package/ccw/src/core/routes/graph-routes.ts +626 -0
  193. package/ccw/src/core/routes/help-routes.ts +308 -0
  194. package/ccw/src/core/routes/hooks-routes.ts +405 -0
  195. package/ccw/src/core/routes/mcp-routes.ts +1271 -0
  196. package/ccw/src/core/routes/mcp-routes.ts.backup +550 -0
  197. package/ccw/src/core/routes/mcp-templates-db.ts +268 -0
  198. package/ccw/src/core/routes/memory-routes.ts +1206 -0
  199. package/ccw/src/core/routes/rules-routes.ts +526 -0
  200. package/ccw/src/core/routes/session-routes.ts +467 -0
  201. package/ccw/src/core/routes/skills-routes.ts +599 -0
  202. package/ccw/src/core/routes/status-routes.ts +57 -0
  203. package/ccw/src/core/routes/system-routes.ts +427 -0
  204. package/ccw/src/core/server.ts +431 -0
  205. package/ccw/src/core/session-clustering-service.ts +1258 -0
  206. package/ccw/src/core/session-scanner.ts +283 -0
  207. package/ccw/src/core/websocket.ts +190 -0
  208. package/ccw/src/{index.js → index.ts} +1 -0
  209. package/ccw/src/mcp-server/index.ts +186 -0
  210. package/ccw/src/templates/assets/css/github-dark.min.css +10 -0
  211. package/ccw/src/templates/assets/css/github.min.css +10 -0
  212. package/ccw/src/templates/assets/js/cytoscape.min.js +32 -0
  213. package/ccw/src/templates/assets/js/d3.min.js +2 -0
  214. package/ccw/src/templates/assets/js/highlight.min.js +1244 -0
  215. package/ccw/src/templates/assets/js/lucide.min.js +12 -0
  216. package/ccw/src/templates/assets/js/marked.min.js +69 -0
  217. package/ccw/src/templates/assets/js/tailwind.js +83 -0
  218. package/ccw/src/templates/dashboard-css/01-base.css +11 -0
  219. package/ccw/src/templates/dashboard-css/02-session.css +22 -0
  220. package/ccw/src/templates/dashboard-css/04-lite-tasks.css +10 -0
  221. package/ccw/src/templates/dashboard-css/06-cards.css +10 -4
  222. package/ccw/src/templates/dashboard-css/07-managers.css +1178 -7
  223. package/ccw/src/templates/dashboard-css/09-explorer.css +23 -12
  224. package/ccw/src/templates/dashboard-css/10-cli-status.css +337 -0
  225. package/ccw/src/templates/dashboard-css/11-cli-history.css +271 -0
  226. package/ccw/src/templates/dashboard-css/12-cli-legacy.css +796 -0
  227. package/ccw/src/templates/dashboard-css/13-cli-ccw.css +199 -0
  228. package/ccw/src/templates/dashboard-css/14-cli-modals.css +258 -0
  229. package/ccw/src/templates/dashboard-css/15-cli-endpoints.css +305 -0
  230. package/ccw/src/templates/dashboard-css/16-cli-session.css +241 -0
  231. package/ccw/src/templates/dashboard-css/17-cli-conversation.css +283 -0
  232. package/ccw/src/templates/dashboard-css/18-cli-settings.css +160 -0
  233. package/ccw/src/templates/dashboard-css/19-cli-native-session.css +496 -0
  234. package/ccw/src/templates/dashboard-css/20-cli-taskqueue.css +188 -0
  235. package/ccw/src/templates/dashboard-css/21-cli-toolmgmt.css +310 -0
  236. package/ccw/src/templates/dashboard-css/22-cli-semantic.css +240 -0
  237. package/ccw/src/templates/dashboard-css/23-memory.css +2390 -0
  238. package/ccw/src/templates/dashboard-css/24-prompt-history.css +1089 -0
  239. package/ccw/src/templates/dashboard-css/25-skills-rules.css +326 -0
  240. package/ccw/src/templates/dashboard-css/26-claude-manager.css +908 -0
  241. package/ccw/src/templates/dashboard-css/27-graph-explorer.css +1678 -0
  242. package/ccw/src/templates/dashboard-css/28-mcp-manager.css +748 -0
  243. package/ccw/src/templates/dashboard-css/29-help.css +264 -0
  244. package/ccw/src/templates/dashboard-css/30-core-memory.css +1700 -0
  245. package/ccw/src/templates/dashboard-js/api.js +162 -142
  246. package/ccw/src/templates/dashboard-js/components/carousel.js +4 -4
  247. package/ccw/src/templates/dashboard-js/components/cli-history.js +876 -0
  248. package/ccw/src/templates/dashboard-js/components/cli-status.js +978 -0
  249. package/ccw/src/templates/dashboard-js/components/global-notifications.js +508 -219
  250. package/ccw/src/templates/dashboard-js/components/hook-manager.js +1277 -282
  251. package/ccw/src/templates/dashboard-js/components/index-manager.js +302 -0
  252. package/ccw/src/templates/dashboard-js/components/mcp-manager.js +718 -27
  253. package/ccw/src/templates/dashboard-js/components/modals.js +66 -0
  254. package/ccw/src/templates/dashboard-js/components/navigation.js +80 -12
  255. package/ccw/src/templates/dashboard-js/components/notifications.js +758 -194
  256. package/ccw/src/templates/dashboard-js/components/storage-manager.js +478 -0
  257. package/ccw/src/templates/dashboard-js/components/tabs-other.js +157 -6
  258. package/ccw/src/templates/dashboard-js/components/task-queue-sidebar.js +716 -0
  259. package/ccw/src/templates/dashboard-js/help-i18n.js +272 -0
  260. package/ccw/src/templates/dashboard-js/i18n.js +2807 -0
  261. package/ccw/src/templates/dashboard-js/main.js +15 -0
  262. package/ccw/src/templates/dashboard-js/state.js +243 -42
  263. package/ccw/src/templates/dashboard-js/utils.js +47 -1
  264. package/ccw/src/templates/dashboard-js/views/claude-manager.js +912 -0
  265. package/ccw/src/templates/dashboard-js/views/cli-manager.js +2272 -0
  266. package/ccw/src/templates/dashboard-js/views/codexlens-manager.js +964 -0
  267. package/ccw/src/templates/dashboard-js/views/core-memory-clusters.js +503 -0
  268. package/ccw/src/templates/dashboard-js/views/core-memory.js +782 -0
  269. package/ccw/src/templates/dashboard-js/views/explorer.js +888 -852
  270. package/ccw/src/templates/dashboard-js/views/graph-explorer.js +1157 -0
  271. package/ccw/src/templates/dashboard-js/views/help.js +856 -0
  272. package/ccw/src/templates/dashboard-js/views/history.js +337 -0
  273. package/ccw/src/templates/dashboard-js/views/home.js +61 -15
  274. package/ccw/src/templates/dashboard-js/views/hook-manager.js +311 -43
  275. package/ccw/src/templates/dashboard-js/views/lite-tasks.js +204 -28
  276. package/ccw/src/templates/dashboard-js/views/mcp-manager.js +2187 -411
  277. package/ccw/src/templates/dashboard-js/views/mcp-manager.js.backup +1729 -0
  278. package/ccw/src/templates/dashboard-js/views/mcp-manager.js.new +928 -0
  279. package/ccw/src/templates/dashboard-js/views/memory.js +1221 -0
  280. package/ccw/src/templates/dashboard-js/views/prompt-history.js +713 -0
  281. package/ccw/src/templates/dashboard-js/views/rules-manager.js +828 -0
  282. package/ccw/src/templates/dashboard-js/views/session-detail.js +54 -53
  283. package/ccw/src/templates/dashboard-js/views/skills-manager.js +819 -0
  284. package/ccw/src/templates/dashboard.html +185 -85
  285. package/ccw/src/templates/hooks-config-example.json +60 -0
  286. package/ccw/src/tools/classify-folders.ts +245 -0
  287. package/ccw/src/tools/cli-config-manager.ts +268 -0
  288. package/ccw/src/tools/cli-executor.ts +2014 -0
  289. package/ccw/src/tools/cli-history-store.ts +1195 -0
  290. package/ccw/src/tools/codex-lens.ts +1141 -0
  291. package/ccw/src/tools/{convert-tokens-to-css.js → convert-tokens-to-css.ts} +73 -23
  292. package/ccw/src/tools/core-memory.ts +444 -0
  293. package/ccw/src/tools/detect-changed-modules.ts +325 -0
  294. package/ccw/src/tools/{discover-design-files.js → discover-design-files.ts} +74 -24
  295. package/ccw/src/tools/edit-file.ts +568 -0
  296. package/ccw/src/tools/{generate-module-docs.js → generate-module-docs.ts} +207 -185
  297. package/ccw/src/tools/{get-modules-by-depth.js → get-modules-by-depth.ts} +120 -79
  298. package/ccw/src/tools/index.ts +370 -0
  299. package/ccw/src/tools/native-session-discovery.ts +795 -0
  300. package/ccw/src/tools/notifier.ts +129 -0
  301. package/ccw/src/tools/read-file.ts +410 -0
  302. package/ccw/src/tools/resume-strategy.ts +345 -0
  303. package/ccw/src/tools/session-content-parser.ts +619 -0
  304. package/ccw/src/tools/session-manager.ts +1026 -0
  305. package/ccw/src/tools/smart-context.ts +228 -0
  306. package/ccw/src/tools/smart-search.ts +2065 -0
  307. package/ccw/src/tools/smart-search.ts.backup +1233 -0
  308. package/ccw/src/tools/storage-manager.ts +455 -0
  309. package/ccw/src/tools/write-file.ts +222 -0
  310. package/ccw/src/types/config.ts +11 -0
  311. package/ccw/src/types/index.ts +3 -0
  312. package/ccw/src/types/session.ts +25 -0
  313. package/ccw/src/types/tool.ts +41 -0
  314. package/ccw/src/utils/{browser-launcher.js → browser-launcher.ts} +10 -8
  315. package/ccw/src/utils/file-utils.ts +48 -0
  316. package/ccw/src/utils/{path-resolver.js → path-resolver.ts} +114 -78
  317. package/ccw/src/utils/path-validator.ts +153 -0
  318. package/ccw/src/utils/{ui.js → ui.ts} +32 -25
  319. package/codex-lens/pyproject.toml +48 -0
  320. package/codex-lens/src/codexlens/.workflow/.cli-history/history.db +0 -0
  321. package/codex-lens/src/codexlens/__init__.py +28 -0
  322. package/codex-lens/src/codexlens/__main__.py +14 -0
  323. package/codex-lens/src/codexlens/__pycache__/__init__.cpython-313.pyc +0 -0
  324. package/codex-lens/src/codexlens/__pycache__/__main__.cpython-313.pyc +0 -0
  325. package/codex-lens/src/codexlens/__pycache__/config.cpython-313.pyc +0 -0
  326. package/codex-lens/src/codexlens/__pycache__/entities.cpython-313.pyc +0 -0
  327. package/codex-lens/src/codexlens/__pycache__/errors.cpython-313.pyc +0 -0
  328. package/codex-lens/src/codexlens/cli/__init__.py +27 -0
  329. package/codex-lens/src/codexlens/cli/__pycache__/__init__.cpython-313.pyc +0 -0
  330. package/codex-lens/src/codexlens/cli/__pycache__/commands.cpython-313.pyc +0 -0
  331. package/codex-lens/src/codexlens/cli/__pycache__/embedding_manager.cpython-313.pyc +0 -0
  332. package/codex-lens/src/codexlens/cli/__pycache__/model_manager.cpython-313.pyc +0 -0
  333. package/codex-lens/src/codexlens/cli/__pycache__/output.cpython-313.pyc +0 -0
  334. package/codex-lens/src/codexlens/cli/commands.py +1931 -0
  335. package/codex-lens/src/codexlens/cli/embedding_manager.py +620 -0
  336. package/codex-lens/src/codexlens/cli/model_manager.py +289 -0
  337. package/codex-lens/src/codexlens/cli/output.py +124 -0
  338. package/codex-lens/src/codexlens/config.py +201 -0
  339. package/codex-lens/src/codexlens/entities.py +121 -0
  340. package/codex-lens/src/codexlens/errors.py +55 -0
  341. package/codex-lens/src/codexlens/indexing/README.md +77 -0
  342. package/codex-lens/src/codexlens/indexing/__init__.py +4 -0
  343. package/codex-lens/src/codexlens/indexing/__pycache__/__init__.cpython-313.pyc +0 -0
  344. package/codex-lens/src/codexlens/indexing/__pycache__/symbol_extractor.cpython-313.pyc +0 -0
  345. package/codex-lens/src/codexlens/indexing/symbol_extractor.py +243 -0
  346. package/codex-lens/src/codexlens/parsers/__init__.py +8 -0
  347. package/codex-lens/src/codexlens/parsers/__pycache__/__init__.cpython-313.pyc +0 -0
  348. package/codex-lens/src/codexlens/parsers/__pycache__/encoding.cpython-313.pyc +0 -0
  349. package/codex-lens/src/codexlens/parsers/__pycache__/factory.cpython-313.pyc +0 -0
  350. package/codex-lens/src/codexlens/parsers/__pycache__/tokenizer.cpython-313.pyc +0 -0
  351. package/codex-lens/src/codexlens/parsers/__pycache__/treesitter_parser.cpython-313.pyc +0 -0
  352. package/codex-lens/src/codexlens/parsers/encoding.py +202 -0
  353. package/codex-lens/src/codexlens/parsers/factory.py +256 -0
  354. package/codex-lens/src/codexlens/parsers/tokenizer.py +98 -0
  355. package/codex-lens/src/codexlens/parsers/treesitter_parser.py +335 -0
  356. package/codex-lens/src/codexlens/search/__init__.py +15 -0
  357. package/codex-lens/src/codexlens/search/__pycache__/__init__.cpython-313.pyc +0 -0
  358. package/codex-lens/src/codexlens/search/__pycache__/chain_search.cpython-313.pyc +0 -0
  359. package/codex-lens/src/codexlens/search/__pycache__/enrichment.cpython-313.pyc +0 -0
  360. package/codex-lens/src/codexlens/search/__pycache__/hybrid_search.cpython-313.pyc +0 -0
  361. package/codex-lens/src/codexlens/search/__pycache__/query_parser.cpython-313.pyc +0 -0
  362. package/codex-lens/src/codexlens/search/__pycache__/ranking.cpython-313.pyc +0 -0
  363. package/codex-lens/src/codexlens/search/chain_search.py +647 -0
  364. package/codex-lens/src/codexlens/search/enrichment.py +150 -0
  365. package/codex-lens/src/codexlens/search/hybrid_search.py +313 -0
  366. package/codex-lens/src/codexlens/search/query_parser.py +242 -0
  367. package/codex-lens/src/codexlens/search/ranking.py +274 -0
  368. package/codex-lens/src/codexlens/semantic/__init__.py +39 -0
  369. package/codex-lens/src/codexlens/semantic/__pycache__/__init__.cpython-313.pyc +0 -0
  370. package/codex-lens/src/codexlens/semantic/__pycache__/ann_index.cpython-313.pyc +0 -0
  371. package/codex-lens/src/codexlens/semantic/__pycache__/chunker.cpython-313.pyc +0 -0
  372. package/codex-lens/src/codexlens/semantic/__pycache__/code_extractor.cpython-313.pyc +0 -0
  373. package/codex-lens/src/codexlens/semantic/__pycache__/embedder.cpython-313.pyc +0 -0
  374. package/codex-lens/src/codexlens/semantic/__pycache__/graph_analyzer.cpython-313.pyc +0 -0
  375. package/codex-lens/src/codexlens/semantic/__pycache__/llm_enhancer.cpython-313.pyc +0 -0
  376. package/codex-lens/src/codexlens/semantic/__pycache__/vector_store.cpython-313.pyc +0 -0
  377. package/codex-lens/src/codexlens/semantic/ann_index.py +414 -0
  378. package/codex-lens/src/codexlens/semantic/chunker.py +448 -0
  379. package/codex-lens/src/codexlens/semantic/code_extractor.py +274 -0
  380. package/codex-lens/src/codexlens/semantic/embedder.py +185 -0
  381. package/codex-lens/src/codexlens/semantic/vector_store.py +955 -0
  382. package/codex-lens/src/codexlens/storage/__init__.py +29 -0
  383. package/codex-lens/src/codexlens/storage/__pycache__/__init__.cpython-313.pyc +0 -0
  384. package/codex-lens/src/codexlens/storage/__pycache__/dir_index.cpython-313.pyc +0 -0
  385. package/codex-lens/src/codexlens/storage/__pycache__/file_cache.cpython-313.pyc +0 -0
  386. package/codex-lens/src/codexlens/storage/__pycache__/index_tree.cpython-313.pyc +0 -0
  387. package/codex-lens/src/codexlens/storage/__pycache__/migration_manager.cpython-313.pyc +0 -0
  388. package/codex-lens/src/codexlens/storage/__pycache__/path_mapper.cpython-313.pyc +0 -0
  389. package/codex-lens/src/codexlens/storage/__pycache__/registry.cpython-313.pyc +0 -0
  390. package/codex-lens/src/codexlens/storage/__pycache__/sqlite_store.cpython-313.pyc +0 -0
  391. package/codex-lens/src/codexlens/storage/__pycache__/sqlite_utils.cpython-313.pyc +0 -0
  392. package/codex-lens/src/codexlens/storage/dir_index.py +1850 -0
  393. package/codex-lens/src/codexlens/storage/file_cache.py +32 -0
  394. package/codex-lens/src/codexlens/storage/index_tree.py +776 -0
  395. package/codex-lens/src/codexlens/storage/migration_manager.py +154 -0
  396. package/codex-lens/src/codexlens/storage/migrations/__init__.py +1 -0
  397. package/codex-lens/src/codexlens/storage/migrations/__pycache__/__init__.cpython-313.pyc +0 -0
  398. package/codex-lens/src/codexlens/storage/migrations/__pycache__/migration_001_normalize_keywords.cpython-313.pyc +0 -0
  399. package/codex-lens/src/codexlens/storage/migrations/__pycache__/migration_002_add_token_metadata.cpython-313.pyc +0 -0
  400. package/codex-lens/src/codexlens/storage/migrations/__pycache__/migration_003_code_relationships.cpython-313.pyc +0 -0
  401. package/codex-lens/src/codexlens/storage/migrations/__pycache__/migration_004_dual_fts.cpython-313.pyc +0 -0
  402. package/codex-lens/src/codexlens/storage/migrations/__pycache__/migration_005_cleanup_unused_fields.cpython-313.pyc +0 -0
  403. package/codex-lens/src/codexlens/storage/migrations/migration_001_normalize_keywords.py +123 -0
  404. package/codex-lens/src/codexlens/storage/migrations/migration_002_add_token_metadata.py +48 -0
  405. package/codex-lens/src/codexlens/storage/migrations/migration_004_dual_fts.py +232 -0
  406. package/codex-lens/src/codexlens/storage/migrations/migration_005_cleanup_unused_fields.py +196 -0
  407. package/codex-lens/src/codexlens/storage/path_mapper.py +274 -0
  408. package/codex-lens/src/codexlens/storage/registry.py +670 -0
  409. package/codex-lens/src/codexlens/storage/sqlite_store.py +576 -0
  410. package/codex-lens/src/codexlens/storage/sqlite_utils.py +64 -0
  411. package/package.json +4 -1
  412. package/.claude/commands/memory/tech-research.md +0 -477
  413. package/.claude/scripts/classify-folders.sh +0 -39
  414. package/.claude/scripts/convert_tokens_to_css.sh +0 -229
  415. package/.claude/scripts/detect_changed_modules.sh +0 -161
  416. package/.claude/scripts/discover-design-files.sh +0 -87
  417. package/.claude/scripts/extract-animations.js +0 -243
  418. package/.claude/scripts/extract-computed-styles.js +0 -118
  419. package/.claude/scripts/extract-layout-structure.js +0 -411
  420. package/.claude/scripts/generate_module_docs.sh +0 -717
  421. package/.claude/scripts/get_modules_by_depth.sh +0 -170
  422. package/.claude/scripts/ui-generate-preview.sh +0 -395
  423. package/.claude/scripts/ui-instantiate-prototypes.sh +0 -815
  424. package/.claude/scripts/update_module_claude.sh +0 -337
  425. package/.claude/workflows/context-search-strategy.md +0 -77
  426. package/.claude/workflows/intelligent-tools-strategy.md +0 -662
  427. package/ccw/src/cli.js +0 -119
  428. package/ccw/src/commands/install.js +0 -324
  429. package/ccw/src/commands/tool.js +0 -138
  430. package/ccw/src/core/data-aggregator.js +0 -409
  431. package/ccw/src/core/server.js +0 -2063
  432. package/ccw/src/core/session-scanner.js +0 -235
  433. package/ccw/src/tools/classify-folders.js +0 -204
  434. package/ccw/src/tools/detect-changed-modules.js +0 -288
  435. package/ccw/src/tools/edit-file.js +0 -266
  436. package/ccw/src/tools/index.js +0 -176
  437. package/ccw/src/utils/file-utils.js +0 -48
@@ -0,0 +1,964 @@
1
+ // CodexLens Manager - Configuration, Model Management, and Semantic Dependencies
2
+ // Extracted from cli-manager.js for better maintainability
3
+
4
+ // ============================================================
5
+ // CODEXLENS CONFIGURATION MODAL
6
+ // ============================================================
7
+
8
+ /**
9
+ * Show CodexLens configuration modal
10
+ */
11
+ async function showCodexLensConfigModal() {
12
+ try {
13
+ showRefreshToast(t('codexlens.loadingConfig'), 'info');
14
+
15
+ // Fetch current config
16
+ const response = await fetch('/api/codexlens/config');
17
+ const config = await response.json();
18
+
19
+ const modalHtml = buildCodexLensConfigContent(config);
20
+
21
+ // Create and show modal
22
+ const tempContainer = document.createElement('div');
23
+ tempContainer.innerHTML = modalHtml;
24
+ const modal = tempContainer.firstElementChild;
25
+ document.body.appendChild(modal);
26
+
27
+ // Initialize icons
28
+ if (window.lucide) lucide.createIcons();
29
+
30
+ // Initialize event handlers
31
+ initCodexLensConfigEvents(config);
32
+ } catch (err) {
33
+ showRefreshToast(t('common.error') + ': ' + err.message, 'error');
34
+ }
35
+ }
36
+
37
+ /**
38
+ * Build CodexLens configuration modal content
39
+ */
40
+ function buildCodexLensConfigContent(config) {
41
+ const indexDir = config.index_dir || '~/.codexlens/indexes';
42
+ const indexCount = config.index_count || 0;
43
+ const isInstalled = window.cliToolsStatus?.codexlens?.installed || false;
44
+
45
+ return '<div class="modal-backdrop" id="codexlensConfigModal">' +
46
+ '<div class="modal-container">' +
47
+ '<div class="modal-header">' +
48
+ '<div class="flex items-center gap-3">' +
49
+ '<div class="modal-icon">' +
50
+ '<i data-lucide="database" class="w-5 h-5"></i>' +
51
+ '</div>' +
52
+ '<div>' +
53
+ '<h2 class="text-lg font-bold">' + t('codexlens.config') + '</h2>' +
54
+ '<p class="text-xs text-muted-foreground">' + t('codexlens.whereIndexesStored') + '</p>' +
55
+ '</div>' +
56
+ '</div>' +
57
+ '<button onclick="closeModal()" class="text-muted-foreground hover:text-foreground">' +
58
+ '<i data-lucide="x" class="w-5 h-5"></i>' +
59
+ '</button>' +
60
+ '</div>' +
61
+
62
+ '<div class="modal-body">' +
63
+ // Status Section
64
+ '<div class="tool-config-section">' +
65
+ '<h4>' + t('codexlens.status') + '</h4>' +
66
+ '<div class="flex items-center gap-4 text-sm">' +
67
+ '<div class="flex items-center gap-2">' +
68
+ '<span class="text-muted-foreground">' + t('codexlens.currentWorkspace') + ':</span>' +
69
+ (isInstalled
70
+ ? '<span class="inline-flex items-center gap-1.5 px-2.5 py-0.5 rounded-full text-xs font-medium bg-success/10 text-success border border-success/20">' +
71
+ '<i data-lucide="check-circle" class="w-3.5 h-3.5"></i>' +
72
+ t('codexlens.installed') +
73
+ '</span>'
74
+ : '<span class="inline-flex items-center gap-1.5 px-2.5 py-0.5 rounded-full text-xs font-medium bg-muted text-muted-foreground border border-border">' +
75
+ '<i data-lucide="circle" class="w-3.5 h-3.5"></i>' +
76
+ t('codexlens.notInstalled') +
77
+ '</span>') +
78
+ '</div>' +
79
+ '<div class="flex items-center gap-2">' +
80
+ '<span class="text-muted-foreground">' + t('codexlens.indexes') + ':</span>' +
81
+ '<span class="inline-flex items-center px-2 py-0.5 rounded-md text-xs font-medium bg-primary/10 text-primary border border-primary/20">' +
82
+ indexCount +
83
+ '</span>' +
84
+ '</div>' +
85
+ '</div>' +
86
+ '</div>' +
87
+
88
+ // Index Storage Path Section
89
+ '<div class="tool-config-section">' +
90
+ '<h4>' + t('codexlens.indexStoragePath') + '</h4>' +
91
+ '<div class="space-y-3">' +
92
+ '<div>' +
93
+ '<label class="block text-sm font-medium mb-1.5">' + t('codexlens.currentPath') + '</label>' +
94
+ '<div class="text-sm text-muted-foreground bg-muted/30 rounded-lg px-3 py-2 font-mono">' +
95
+ indexDir +
96
+ '</div>' +
97
+ '</div>' +
98
+ '<div>' +
99
+ '<label class="block text-sm font-medium mb-1.5">' + t('codexlens.newStoragePath') + '</label>' +
100
+ '<input type="text" id="indexDirInput" value="' + indexDir + '" ' +
101
+ 'placeholder="' + t('codexlens.pathPlaceholder') + '" ' +
102
+ 'class="tool-config-input w-full" />' +
103
+ '<p class="text-xs text-muted-foreground mt-1">' + t('codexlens.pathInfo') + '</p>' +
104
+ '</div>' +
105
+ '<div class="flex items-start gap-2 bg-warning/10 border border-warning/30 rounded-lg p-3">' +
106
+ '<i data-lucide="alert-triangle" class="w-4 h-4 text-warning mt-0.5"></i>' +
107
+ '<div class="text-sm">' +
108
+ '<p class="font-medium text-warning">' + t('codexlens.migrationRequired') + '</p>' +
109
+ '<p class="text-muted-foreground mt-1">' + t('codexlens.migrationWarning') + '</p>' +
110
+ '</div>' +
111
+ '</div>' +
112
+ '</div>' +
113
+ '</div>' +
114
+
115
+ // Actions Section
116
+ '<div class="tool-config-section">' +
117
+ '<h4>' + t('codexlens.actions') + '</h4>' +
118
+ '<div class="tool-config-actions">' +
119
+ (isInstalled
120
+ ? '<button class="inline-flex items-center gap-1.5 px-3 py-1.5 text-xs font-medium rounded-md border border-primary/30 bg-primary/5 text-primary hover:bg-primary/10 transition-colors" onclick="initCodexLensIndex()">' +
121
+ '<i data-lucide="database" class="w-3.5 h-3.5"></i> ' + t('codexlens.initializeIndex') +
122
+ '</button>' +
123
+ '<button class="inline-flex items-center gap-1.5 px-3 py-1.5 text-xs font-medium rounded-md border border-border bg-background hover:bg-muted/50 transition-colors" onclick="cleanCurrentWorkspaceIndex()">' +
124
+ '<i data-lucide="folder-x" class="w-3.5 h-3.5"></i> ' + t('codexlens.cleanCurrentWorkspace') +
125
+ '</button>' +
126
+ '<button class="inline-flex items-center gap-1.5 px-3 py-1.5 text-xs font-medium rounded-md border border-border bg-background hover:bg-muted/50 transition-colors" onclick="cleanCodexLensIndexes()">' +
127
+ '<i data-lucide="trash" class="w-3.5 h-3.5"></i> ' + t('codexlens.cleanAllIndexes') +
128
+ '</button>' +
129
+ '<button class="inline-flex items-center gap-1.5 px-3 py-1.5 text-xs font-medium rounded-md border border-destructive/30 bg-destructive/5 text-destructive hover:bg-destructive/10 transition-colors" onclick="uninstallCodexLens()">' +
130
+ '<i data-lucide="trash-2" class="w-3.5 h-3.5"></i> ' + t('cli.uninstall') +
131
+ '</button>'
132
+ : '<button class="inline-flex items-center gap-1.5 px-3 py-1.5 text-xs font-medium rounded-md bg-primary text-primary-foreground hover:bg-primary/90 transition-colors" onclick="installCodexLens()">' +
133
+ '<i data-lucide="download" class="w-3.5 h-3.5"></i> ' + t('codexlens.installCodexLens') +
134
+ '</button>') +
135
+ '</div>' +
136
+ '</div>' +
137
+
138
+ // Semantic Dependencies Section
139
+ (isInstalled
140
+ ? '<div class="tool-config-section">' +
141
+ '<h4>' + t('codexlens.semanticDeps') + '</h4>' +
142
+ '<div id="semanticDepsStatus" class="space-y-2">' +
143
+ '<div class="text-sm text-muted-foreground">' + t('codexlens.checkingDeps') + '</div>' +
144
+ '</div>' +
145
+ '</div>'
146
+ : '') +
147
+
148
+ // Model Management Section
149
+ (isInstalled
150
+ ? '<div class="tool-config-section">' +
151
+ '<h4>' + t('codexlens.modelManagement') + '</h4>' +
152
+ '<div id="modelListContainer" class="space-y-2">' +
153
+ '<div class="text-sm text-muted-foreground">' + t('codexlens.loadingModels') + '</div>' +
154
+ '</div>' +
155
+ '</div>'
156
+ : '') +
157
+
158
+ // Test Search Section
159
+ (isInstalled
160
+ ? '<div class="tool-config-section">' +
161
+ '<h4>' + t('codexlens.testSearch') + ' <span class="text-muted">(' + t('codexlens.testFunctionality') + ')</span></h4>' +
162
+ '<div class="space-y-3">' +
163
+ '<div class="flex gap-2">' +
164
+ '<select id="searchTypeSelect" class="tool-config-select flex-1">' +
165
+ '<option value="search">' + t('codexlens.textSearch') + '</option>' +
166
+ '<option value="search_files">' + t('codexlens.fileSearch') + '</option>' +
167
+ '<option value="symbol">' + t('codexlens.symbolSearch') + '</option>' +
168
+ '</select>' +
169
+ '<select id="searchModeSelect" class="tool-config-select flex-1">' +
170
+ '<option value="exact">' + t('codexlens.exactMode') + '</option>' +
171
+ '<option value="fuzzy">' + t('codexlens.fuzzyMode') + '</option>' +
172
+ '<option value="hybrid">' + t('codexlens.hybridMode') + '</option>' +
173
+ '<option value="vector">' + t('codexlens.vectorMode') + '</option>' +
174
+ '</select>' +
175
+ '</div>' +
176
+ '<div>' +
177
+ '<input type="text" id="searchQueryInput" class="tool-config-input w-full" ' +
178
+ 'placeholder="' + t('codexlens.searchPlaceholder') + '" />' +
179
+ '</div>' +
180
+ '<div>' +
181
+ '<button class="btn-sm btn-primary w-full" id="runSearchBtn">' +
182
+ '<i data-lucide="search" class="w-3 h-3"></i> ' + t('codexlens.runSearch') +
183
+ '</button>' +
184
+ '</div>' +
185
+ '<div id="searchResults" class="hidden">' +
186
+ '<div>' +
187
+ '<div class="flex items-center justify-between">' +
188
+ '<p class="text-sm font-medium">' + t('codexlens.results') + ':</p>' +
189
+ '<span id="searchResultCount" class="text-xs text-muted-foreground"></span>' +
190
+ '</div>' +
191
+ '<pre id="searchResultContent"></pre>' +
192
+ '</div>' +
193
+ '</div>' +
194
+ '</div>' +
195
+ '</div>'
196
+ : '') +
197
+ '</div>' +
198
+
199
+ // Footer
200
+ '<div class="tool-config-footer">' +
201
+ '<button class="btn btn-outline" onclick="closeModal()">' + t('common.cancel') + '</button>' +
202
+ '<button class="btn btn-primary" id="saveCodexLensConfigBtn">' +
203
+ '<i data-lucide="save" class="w-3.5 h-3.5"></i> ' + t('codexlens.saveConfig') +
204
+ '</button>' +
205
+ '</div>' +
206
+ '</div>';
207
+ }
208
+
209
+ /**
210
+ * Initialize CodexLens config modal event handlers
211
+ */
212
+ function initCodexLensConfigEvents(currentConfig) {
213
+ // Save button
214
+ var saveBtn = document.getElementById('saveCodexLensConfigBtn');
215
+ if (saveBtn) {
216
+ saveBtn.onclick = async function() {
217
+ var indexDirInput = document.getElementById('indexDirInput');
218
+ var newIndexDir = indexDirInput ? indexDirInput.value.trim() : '';
219
+
220
+ if (!newIndexDir) {
221
+ showRefreshToast(t('codexlens.pathEmpty'), 'error');
222
+ return;
223
+ }
224
+
225
+ if (newIndexDir === currentConfig.index_dir) {
226
+ closeModal();
227
+ return;
228
+ }
229
+
230
+ saveBtn.disabled = true;
231
+ saveBtn.innerHTML = '<span class="animate-pulse">' + t('common.saving') + '</span>';
232
+
233
+ try {
234
+ var response = await fetch('/api/codexlens/config', {
235
+ method: 'POST',
236
+ headers: { 'Content-Type': 'application/json' },
237
+ body: JSON.stringify({ index_dir: newIndexDir })
238
+ });
239
+
240
+ var result = await response.json();
241
+
242
+ if (result.success) {
243
+ showRefreshToast(t('codexlens.configSaved'), 'success');
244
+ closeModal();
245
+
246
+ // Refresh CodexLens status
247
+ if (typeof loadCodexLensStatus === 'function') {
248
+ await loadCodexLensStatus();
249
+ renderToolsSection();
250
+ if (window.lucide) lucide.createIcons();
251
+ }
252
+ } else {
253
+ showRefreshToast(t('common.saveFailed') + ': ' + result.error, 'error');
254
+ saveBtn.disabled = false;
255
+ saveBtn.innerHTML = '<i data-lucide="save" class="w-3.5 h-3.5"></i> ' + t('codexlens.saveConfig');
256
+ if (window.lucide) lucide.createIcons();
257
+ }
258
+ } catch (err) {
259
+ showRefreshToast(t('common.error') + ': ' + err.message, 'error');
260
+ saveBtn.disabled = false;
261
+ saveBtn.innerHTML = '<i data-lucide="save" class="w-3.5 h-3.5"></i> ' + t('codexlens.saveConfig');
262
+ if (window.lucide) lucide.createIcons();
263
+ }
264
+ };
265
+ }
266
+
267
+ // Test Search Button
268
+ var runSearchBtn = document.getElementById('runSearchBtn');
269
+ if (runSearchBtn) {
270
+ runSearchBtn.onclick = async function() {
271
+ var searchType = document.getElementById('searchTypeSelect').value;
272
+ var searchMode = document.getElementById('searchModeSelect').value;
273
+ var query = document.getElementById('searchQueryInput').value.trim();
274
+ var resultsDiv = document.getElementById('searchResults');
275
+ var resultCount = document.getElementById('searchResultCount');
276
+ var resultContent = document.getElementById('searchResultContent');
277
+
278
+ if (!query) {
279
+ showRefreshToast(t('codexlens.enterQuery'), 'warning');
280
+ return;
281
+ }
282
+
283
+ runSearchBtn.disabled = true;
284
+ runSearchBtn.innerHTML = '<span class="animate-pulse">' + t('codexlens.searching') + '</span>';
285
+ resultsDiv.classList.add('hidden');
286
+
287
+ try {
288
+ var endpoint = '/api/codexlens/' + searchType;
289
+ var params = new URLSearchParams({ query: query, limit: '20' });
290
+ // Add mode parameter for search and search_files (not for symbol search)
291
+ if (searchType === 'search' || searchType === 'search_files') {
292
+ params.append('mode', searchMode);
293
+ }
294
+
295
+ var response = await fetch(endpoint + '?' + params.toString());
296
+ var result = await response.json();
297
+
298
+ console.log('[CodexLens Test] Search result:', result);
299
+
300
+ if (result.success) {
301
+ var results = result.results || result.files || [];
302
+ resultCount.textContent = results.length + ' ' + t('codexlens.resultsCount');
303
+ resultContent.textContent = JSON.stringify(results, null, 2);
304
+ resultsDiv.classList.remove('hidden');
305
+ showRefreshToast(t('codexlens.searchCompleted') + ': ' + results.length + ' ' + t('codexlens.resultsCount'), 'success');
306
+ } else {
307
+ resultContent.textContent = t('common.error') + ': ' + (result.error || t('common.unknownError'));
308
+ resultsDiv.classList.remove('hidden');
309
+ showRefreshToast(t('codexlens.searchFailed') + ': ' + result.error, 'error');
310
+ }
311
+
312
+ runSearchBtn.disabled = false;
313
+ runSearchBtn.innerHTML = '<i data-lucide="search" class="w-3 h-3"></i> ' + t('codexlens.runSearch');
314
+ if (window.lucide) lucide.createIcons();
315
+ } catch (err) {
316
+ console.error('[CodexLens Test] Error:', err);
317
+ resultContent.textContent = t('common.exception') + ': ' + err.message;
318
+ resultsDiv.classList.remove('hidden');
319
+ showRefreshToast(t('common.error') + ': ' + err.message, 'error');
320
+ runSearchBtn.disabled = false;
321
+ runSearchBtn.innerHTML = '<i data-lucide="search" class="w-3 h-3"></i> ' + t('codexlens.runSearch');
322
+ if (window.lucide) lucide.createIcons();
323
+ }
324
+ };
325
+ }
326
+
327
+ // Load semantic dependencies status
328
+ loadSemanticDepsStatus();
329
+
330
+ // Load model list
331
+ loadModelList();
332
+ }
333
+
334
+ // ============================================================
335
+ // SEMANTIC DEPENDENCIES MANAGEMENT
336
+ // ============================================================
337
+
338
+ /**
339
+ * Load semantic dependencies status
340
+ */
341
+ async function loadSemanticDepsStatus() {
342
+ var container = document.getElementById('semanticDepsStatus');
343
+ if (!container) return;
344
+
345
+ try {
346
+ var response = await fetch('/api/codexlens/semantic/status');
347
+ var result = await response.json();
348
+
349
+ if (result.available) {
350
+ container.innerHTML =
351
+ '<div class="flex items-center gap-2 text-sm">' +
352
+ '<i data-lucide="check-circle" class="w-4 h-4 text-success"></i>' +
353
+ '<span>' + t('codexlens.semanticInstalled') + '</span>' +
354
+ '<span class="text-muted-foreground">(' + (result.backend || 'fastembed') + ')</span>' +
355
+ '</div>';
356
+ } else {
357
+ container.innerHTML =
358
+ '<div class="space-y-2">' +
359
+ '<div class="flex items-center gap-2 text-sm text-muted-foreground">' +
360
+ '<i data-lucide="alert-circle" class="w-4 h-4"></i>' +
361
+ '<span>' + t('codexlens.semanticNotInstalled') + '</span>' +
362
+ '</div>' +
363
+ '<button class="btn-sm btn-outline" onclick="installSemanticDeps()">' +
364
+ '<i data-lucide="download" class="w-3 h-3"></i> ' + t('codexlens.installDeps') +
365
+ '</button>' +
366
+ '</div>';
367
+ }
368
+ if (window.lucide) lucide.createIcons();
369
+ } catch (err) {
370
+ container.innerHTML =
371
+ '<div class="text-sm text-error">' + t('common.error') + ': ' + err.message + '</div>';
372
+ }
373
+ }
374
+
375
+ /**
376
+ * Install semantic dependencies
377
+ */
378
+ async function installSemanticDeps() {
379
+ var container = document.getElementById('semanticDepsStatus');
380
+ if (!container) return;
381
+
382
+ container.innerHTML =
383
+ '<div class="text-sm text-muted-foreground animate-pulse">' + t('codexlens.installingDeps') + '</div>';
384
+
385
+ try {
386
+ var response = await fetch('/api/codexlens/semantic/install', { method: 'POST' });
387
+ var result = await response.json();
388
+
389
+ if (result.success) {
390
+ showRefreshToast(t('codexlens.depsInstalled'), 'success');
391
+ await loadSemanticDepsStatus();
392
+ await loadModelList();
393
+ } else {
394
+ showRefreshToast(t('codexlens.depsInstallFailed') + ': ' + result.error, 'error');
395
+ await loadSemanticDepsStatus();
396
+ }
397
+ } catch (err) {
398
+ showRefreshToast(t('common.error') + ': ' + err.message, 'error');
399
+ await loadSemanticDepsStatus();
400
+ }
401
+ }
402
+
403
+ // ============================================================
404
+ // MODEL MANAGEMENT
405
+ // ============================================================
406
+
407
+ /**
408
+ * Load model list
409
+ */
410
+ async function loadModelList() {
411
+ var container = document.getElementById('modelListContainer');
412
+ if (!container) return;
413
+
414
+ try {
415
+ var response = await fetch('/api/codexlens/models');
416
+ var result = await response.json();
417
+
418
+ if (!result.success) {
419
+ // Check if the error is specifically about fastembed not being installed
420
+ var errorMsg = result.error || '';
421
+ if (errorMsg.includes('fastembed not installed') || errorMsg.includes('Semantic')) {
422
+ container.innerHTML =
423
+ '<div class="text-sm text-muted-foreground">' + t('codexlens.semanticNotInstalled') + '</div>';
424
+ } else {
425
+ // Show actual error message for other failures
426
+ container.innerHTML =
427
+ '<div class="text-sm text-error">' + t('codexlens.modelListError') + ': ' + (errorMsg || t('common.unknownError')) + '</div>';
428
+ }
429
+ return;
430
+ }
431
+
432
+ if (!result.result || !result.result.models) {
433
+ container.innerHTML =
434
+ '<div class="text-sm text-muted-foreground">' + t('codexlens.noModelsAvailable') + '</div>';
435
+ return;
436
+ }
437
+
438
+ var models = result.result.models;
439
+ var html = '<div class="space-y-2">';
440
+
441
+ models.forEach(function(model) {
442
+ var statusIcon = model.installed
443
+ ? '<i data-lucide="check-circle" class="w-4 h-4 text-success"></i>'
444
+ : '<i data-lucide="circle" class="w-4 h-4 text-muted"></i>';
445
+
446
+ var sizeText = model.installed
447
+ ? model.actual_size_mb.toFixed(1) + ' MB'
448
+ : '~' + model.estimated_size_mb + ' MB';
449
+
450
+ var actionBtn = model.installed
451
+ ? '<button class="btn-sm btn-outline btn-danger" onclick="deleteModel(\'' + model.profile + '\')">' +
452
+ '<i data-lucide="trash-2" class="w-3 h-3"></i> ' + t('codexlens.deleteModel') +
453
+ '</button>'
454
+ : '<button class="btn-sm btn-outline" onclick="downloadModel(\'' + model.profile + '\')">' +
455
+ '<i data-lucide="download" class="w-3 h-3"></i> ' + t('codexlens.downloadModel') +
456
+ '</button>';
457
+
458
+ html +=
459
+ '<div class="border rounded-lg p-3 space-y-2" id="model-' + model.profile + '">' +
460
+ '<div class="flex items-start justify-between">' +
461
+ '<div class="flex-1">' +
462
+ '<div class="flex items-center gap-2 mb-1">' +
463
+ statusIcon +
464
+ '<span class="font-medium">' + model.profile + '</span>' +
465
+ '<span class="text-xs text-muted-foreground">(' + model.dimensions + ' dims)</span>' +
466
+ '</div>' +
467
+ '<div class="text-xs text-muted-foreground mb-1">' + model.model_name + '</div>' +
468
+ '<div class="text-xs text-muted-foreground">' + model.use_case + '</div>' +
469
+ '</div>' +
470
+ '<div class="text-right">' +
471
+ '<div class="text-xs text-muted-foreground mb-2">' + sizeText + '</div>' +
472
+ actionBtn +
473
+ '</div>' +
474
+ '</div>' +
475
+ '</div>';
476
+ });
477
+
478
+ html += '</div>';
479
+ container.innerHTML = html;
480
+ if (window.lucide) lucide.createIcons();
481
+ } catch (err) {
482
+ container.innerHTML =
483
+ '<div class="text-sm text-error">' + t('common.error') + ': ' + err.message + '</div>';
484
+ }
485
+ }
486
+
487
+ /**
488
+ * Download model
489
+ */
490
+ async function downloadModel(profile) {
491
+ var modelCard = document.getElementById('model-' + profile);
492
+ if (!modelCard) return;
493
+
494
+ var originalHTML = modelCard.innerHTML;
495
+ modelCard.innerHTML =
496
+ '<div class="flex items-center justify-center p-3">' +
497
+ '<span class="text-sm text-muted-foreground animate-pulse">' + t('codexlens.downloading') + '</span>' +
498
+ '</div>';
499
+
500
+ try {
501
+ var response = await fetch('/api/codexlens/models/download', {
502
+ method: 'POST',
503
+ headers: { 'Content-Type': 'application/json' },
504
+ body: JSON.stringify({ profile: profile })
505
+ });
506
+
507
+ var result = await response.json();
508
+
509
+ if (result.success) {
510
+ showRefreshToast(t('codexlens.modelDownloaded') + ': ' + profile, 'success');
511
+ await loadModelList();
512
+ } else {
513
+ showRefreshToast(t('codexlens.modelDownloadFailed') + ': ' + result.error, 'error');
514
+ modelCard.innerHTML = originalHTML;
515
+ if (window.lucide) lucide.createIcons();
516
+ }
517
+ } catch (err) {
518
+ showRefreshToast(t('common.error') + ': ' + err.message, 'error');
519
+ modelCard.innerHTML = originalHTML;
520
+ if (window.lucide) lucide.createIcons();
521
+ }
522
+ }
523
+
524
+ /**
525
+ * Delete model
526
+ */
527
+ async function deleteModel(profile) {
528
+ if (!confirm(t('codexlens.deleteModelConfirm') + ' ' + profile + '?')) {
529
+ return;
530
+ }
531
+
532
+ var modelCard = document.getElementById('model-' + profile);
533
+ if (!modelCard) return;
534
+
535
+ var originalHTML = modelCard.innerHTML;
536
+ modelCard.innerHTML =
537
+ '<div class="flex items-center justify-center p-3">' +
538
+ '<span class="text-sm text-muted-foreground animate-pulse">' + t('codexlens.deleting') + '</span>' +
539
+ '</div>';
540
+
541
+ try {
542
+ var response = await fetch('/api/codexlens/models/delete', {
543
+ method: 'POST',
544
+ headers: { 'Content-Type': 'application/json' },
545
+ body: JSON.stringify({ profile: profile })
546
+ });
547
+
548
+ var result = await response.json();
549
+
550
+ if (result.success) {
551
+ showRefreshToast(t('codexlens.modelDeleted') + ': ' + profile, 'success');
552
+ await loadModelList();
553
+ } else {
554
+ showRefreshToast(t('codexlens.modelDeleteFailed') + ': ' + result.error, 'error');
555
+ modelCard.innerHTML = originalHTML;
556
+ if (window.lucide) lucide.createIcons();
557
+ }
558
+ } catch (err) {
559
+ showRefreshToast(t('common.error') + ': ' + err.message, 'error');
560
+ modelCard.innerHTML = originalHTML;
561
+ if (window.lucide) lucide.createIcons();
562
+ }
563
+ }
564
+
565
+ // ============================================================
566
+ // CODEXLENS ACTIONS
567
+ // ============================================================
568
+
569
+ /**
570
+ * Initialize CodexLens index with bottom floating progress bar
571
+ * @param {string} indexType - 'vector' (with embeddings), 'normal' (FTS only), or 'full' (FTS + Vector)
572
+ * @param {string} embeddingModel - Model profile: 'code', 'fast', 'multilingual', 'balanced'
573
+ */
574
+ async function initCodexLensIndex(indexType, embeddingModel) {
575
+ indexType = indexType || 'vector';
576
+ embeddingModel = embeddingModel || 'code';
577
+
578
+ // For vector or full index, check if semantic dependencies are available
579
+ if (indexType === 'vector' || indexType === 'full') {
580
+ try {
581
+ var semanticResponse = await fetch('/api/codexlens/semantic/status');
582
+ var semanticStatus = await semanticResponse.json();
583
+
584
+ if (!semanticStatus.available) {
585
+ // Semantic deps not installed - show confirmation dialog
586
+ var installDeps = confirm(
587
+ (t('codexlens.semanticNotInstalled') || 'Semantic search dependencies are not installed.') + '\n\n' +
588
+ (t('codexlens.installDepsPrompt') || 'Would you like to install them now? (This may take a few minutes)\n\nClick "Cancel" to create FTS index only.')
589
+ );
590
+
591
+ if (installDeps) {
592
+ // Install semantic dependencies first
593
+ showRefreshToast(t('codexlens.installingDeps') || 'Installing semantic dependencies...', 'info');
594
+ try {
595
+ var installResponse = await fetch('/api/codexlens/semantic/install', { method: 'POST' });
596
+ var installResult = await installResponse.json();
597
+
598
+ if (!installResult.success) {
599
+ showRefreshToast((t('codexlens.depsInstallFailed') || 'Failed to install dependencies') + ': ' + installResult.error, 'error');
600
+ // Fall back to FTS only
601
+ indexType = 'normal';
602
+ } else {
603
+ showRefreshToast(t('codexlens.depsInstalled') || 'Dependencies installed successfully', 'success');
604
+ }
605
+ } catch (err) {
606
+ showRefreshToast((t('common.error') || 'Error') + ': ' + err.message, 'error');
607
+ indexType = 'normal';
608
+ }
609
+ } else {
610
+ // User chose to skip - create FTS only
611
+ indexType = 'normal';
612
+ }
613
+ }
614
+ } catch (err) {
615
+ console.warn('[CodexLens] Could not check semantic status:', err);
616
+ // Continue with requested type, backend will handle fallback
617
+ }
618
+ }
619
+
620
+ // Remove existing progress bar if any
621
+ closeCodexLensIndexModal();
622
+
623
+ // Create bottom floating progress bar
624
+ var progressBar = document.createElement('div');
625
+ progressBar.id = 'codexlensIndexFloating';
626
+ progressBar.className = 'fixed bottom-0 left-0 right-0 z-50 bg-card border-t border-border shadow-lg transform transition-transform duration-300';
627
+
628
+ // Determine display label
629
+ var indexTypeLabel;
630
+ if (indexType === 'full') {
631
+ indexTypeLabel = 'FTS + Vector';
632
+ } else if (indexType === 'vector') {
633
+ indexTypeLabel = 'Vector';
634
+ } else {
635
+ indexTypeLabel = 'FTS';
636
+ }
637
+
638
+ // Add model info for vector indexes
639
+ var modelLabel = '';
640
+ if (indexType !== 'normal') {
641
+ var modelNames = { code: 'Code', fast: 'Fast', multilingual: 'Multi', balanced: 'Balanced' };
642
+ modelLabel = ' [' + (modelNames[embeddingModel] || embeddingModel) + ']';
643
+ }
644
+
645
+ progressBar.innerHTML =
646
+ '<div class="max-w-4xl mx-auto px-4 py-3">' +
647
+ '<div class="flex items-center justify-between gap-4">' +
648
+ '<div class="flex items-center gap-3 flex-1 min-w-0">' +
649
+ '<div class="animate-spin w-5 h-5 border-2 border-primary border-t-transparent rounded-full flex-shrink-0" id="codexlensIndexSpinner"></div>' +
650
+ '<div class="flex-1 min-w-0">' +
651
+ '<div class="flex items-center gap-2">' +
652
+ '<span class="font-medium text-sm">' + t('codexlens.indexing') + ' (' + indexTypeLabel + modelLabel + ')</span>' +
653
+ '<span class="text-xs text-muted-foreground" id="codexlensIndexPercent">0%</span>' +
654
+ '</div>' +
655
+ '<div class="text-xs text-muted-foreground truncate" id="codexlensIndexStatus">' + t('codexlens.preparingIndex') + '</div>' +
656
+ '</div>' +
657
+ '</div>' +
658
+ '<div class="flex-1 max-w-xs hidden sm:block">' +
659
+ '<div class="h-2 bg-muted rounded-full overflow-hidden">' +
660
+ '<div id="codexlensIndexProgressBar" class="h-full bg-primary transition-all duration-300 ease-out" style="width: 0%"></div>' +
661
+ '</div>' +
662
+ '</div>' +
663
+ '<button id="codexlensIndexCancelBtn" class="px-2 py-1 text-xs bg-destructive/10 hover:bg-destructive/20 text-destructive rounded-md transition-colors flex-shrink-0" onclick="cancelCodexLensIndexing()" title="' + t('common.cancel') + '">' +
664
+ t('common.cancel') +
665
+ '</button>' +
666
+ '<button class="p-1.5 hover:bg-muted rounded-md transition-colors flex-shrink-0" onclick="closeCodexLensIndexModal()" title="' + t('common.close') + '">' +
667
+ '<i data-lucide="x" class="w-4 h-4"></i>' +
668
+ '</button>' +
669
+ '</div>' +
670
+ '</div>';
671
+
672
+ document.body.appendChild(progressBar);
673
+ if (window.lucide) lucide.createIcons();
674
+
675
+ // For 'full' type, use 'vector' in the API (it creates FTS + embeddings)
676
+ var apiIndexType = (indexType === 'full') ? 'vector' : indexType;
677
+
678
+ // Start indexing with specified type and model
679
+ startCodexLensIndexing(apiIndexType, embeddingModel);
680
+ }
681
+
682
+ /**
683
+ * Start the indexing process
684
+ * @param {string} indexType - 'vector' or 'normal'
685
+ * @param {string} embeddingModel - Model profile: 'code', 'fast', 'multilingual', 'balanced'
686
+ */
687
+ async function startCodexLensIndexing(indexType, embeddingModel) {
688
+ indexType = indexType || 'vector';
689
+ embeddingModel = embeddingModel || 'code';
690
+ var statusText = document.getElementById('codexlensIndexStatus');
691
+ var progressBar = document.getElementById('codexlensIndexProgressBar');
692
+ var percentText = document.getElementById('codexlensIndexPercent');
693
+ var spinner = document.getElementById('codexlensIndexSpinner');
694
+
695
+ // Setup WebSocket listener for progress events
696
+ window.codexlensIndexProgressHandler = function(data) {
697
+ var payload = data.payload || data;
698
+ console.log('[CodexLens] Progress event received:', payload);
699
+
700
+ if (statusText) statusText.textContent = payload.message || t('codexlens.indexing');
701
+ if (progressBar) progressBar.style.width = (payload.percent || 0) + '%';
702
+ if (percentText) percentText.textContent = (payload.percent || 0) + '%';
703
+
704
+ // Handle completion
705
+ if (payload.stage === 'complete') {
706
+ handleIndexComplete(true, payload.message);
707
+ } else if (payload.stage === 'error') {
708
+ handleIndexComplete(false, payload.message);
709
+ }
710
+ };
711
+
712
+ // Register with notification system
713
+ if (typeof registerWsEventHandler === 'function') {
714
+ registerWsEventHandler('CODEXLENS_INDEX_PROGRESS', window.codexlensIndexProgressHandler);
715
+ console.log('[CodexLens] Registered WebSocket progress handler');
716
+ } else {
717
+ console.warn('[CodexLens] registerWsEventHandler not available');
718
+ }
719
+
720
+ try {
721
+ console.log('[CodexLens] Starting index for:', projectPath, 'type:', indexType, 'model:', embeddingModel);
722
+ var response = await fetch('/api/codexlens/init', {
723
+ method: 'POST',
724
+ headers: { 'Content-Type': 'application/json' },
725
+ body: JSON.stringify({ path: projectPath, indexType: indexType, embeddingModel: embeddingModel })
726
+ });
727
+
728
+ var result = await response.json();
729
+ console.log('[CodexLens] Init result:', result);
730
+
731
+ // Check if completed successfully (WebSocket might have already reported)
732
+ if (result.success) {
733
+ handleIndexComplete(true, t('codexlens.indexComplete'));
734
+ } else if (!result.success) {
735
+ handleIndexComplete(false, result.error || t('common.unknownError'));
736
+ }
737
+ } catch (err) {
738
+ console.error('[CodexLens] Init error:', err);
739
+ handleIndexComplete(false, err.message);
740
+ }
741
+ }
742
+
743
+ /**
744
+ * Handle index completion
745
+ */
746
+ function handleIndexComplete(success, message) {
747
+ var statusText = document.getElementById('codexlensIndexStatus');
748
+ var progressBar = document.getElementById('codexlensIndexProgressBar');
749
+ var percentText = document.getElementById('codexlensIndexPercent');
750
+ var spinner = document.getElementById('codexlensIndexSpinner');
751
+ var floatingBar = document.getElementById('codexlensIndexFloating');
752
+
753
+ // Unregister WebSocket handler
754
+ if (typeof unregisterWsEventHandler === 'function' && window.codexlensIndexProgressHandler) {
755
+ unregisterWsEventHandler('CODEXLENS_INDEX_PROGRESS', window.codexlensIndexProgressHandler);
756
+ }
757
+
758
+ if (success) {
759
+ if (progressBar) progressBar.style.width = '100%';
760
+ if (percentText) percentText.textContent = '100%';
761
+ if (statusText) statusText.textContent = t('codexlens.indexComplete');
762
+ if (spinner) {
763
+ spinner.classList.remove('animate-spin', 'border-primary');
764
+ spinner.classList.add('border-green-500');
765
+ spinner.innerHTML = '<i data-lucide="check" class="w-5 h-5 text-green-500"></i>';
766
+ if (window.lucide) lucide.createIcons();
767
+ }
768
+ if (floatingBar) {
769
+ floatingBar.classList.add('bg-green-500/10');
770
+ }
771
+
772
+ showRefreshToast(t('codexlens.indexSuccess'), 'success');
773
+
774
+ // Auto-close after 3 seconds
775
+ setTimeout(function() {
776
+ closeCodexLensIndexModal();
777
+ // Refresh status
778
+ if (typeof loadCodexLensStatus === 'function') {
779
+ loadCodexLensStatus().then(function() {
780
+ renderToolsSection();
781
+ if (window.lucide) lucide.createIcons();
782
+ });
783
+ }
784
+ }, 3000);
785
+ } else {
786
+ if (progressBar) {
787
+ progressBar.classList.remove('bg-primary');
788
+ progressBar.classList.add('bg-destructive');
789
+ }
790
+ if (statusText) statusText.textContent = message || t('codexlens.indexFailed');
791
+ if (spinner) {
792
+ spinner.classList.remove('animate-spin', 'border-primary');
793
+ spinner.innerHTML = '<i data-lucide="alert-circle" class="w-5 h-5 text-destructive"></i>';
794
+ if (window.lucide) lucide.createIcons();
795
+ }
796
+ if (floatingBar) {
797
+ floatingBar.classList.add('bg-destructive/10');
798
+ }
799
+
800
+ showRefreshToast(t('codexlens.indexFailed') + ': ' + message, 'error');
801
+ }
802
+ }
803
+
804
+ /**
805
+ * Close floating progress bar
806
+ */
807
+ function closeCodexLensIndexModal() {
808
+ var floatingBar = document.getElementById('codexlensIndexFloating');
809
+ if (floatingBar) {
810
+ floatingBar.classList.add('translate-y-full');
811
+ setTimeout(function() {
812
+ floatingBar.remove();
813
+ }, 300);
814
+ }
815
+
816
+ // Unregister WebSocket handler
817
+ if (typeof unregisterWsEventHandler === 'function' && window.codexlensIndexProgressHandler) {
818
+ unregisterWsEventHandler('CODEXLENS_INDEX_PROGRESS', window.codexlensIndexProgressHandler);
819
+ }
820
+ }
821
+
822
+ /**
823
+ * Cancel the running indexing process
824
+ */
825
+ async function cancelCodexLensIndexing() {
826
+ var cancelBtn = document.getElementById('codexlensIndexCancelBtn');
827
+ var statusText = document.getElementById('codexlensIndexStatus');
828
+
829
+ // Disable button to prevent double-click
830
+ if (cancelBtn) {
831
+ cancelBtn.disabled = true;
832
+ cancelBtn.textContent = t('common.canceling') || 'Canceling...';
833
+ }
834
+
835
+ try {
836
+ var response = await fetch('/api/codexlens/cancel', {
837
+ method: 'POST',
838
+ headers: { 'Content-Type': 'application/json' }
839
+ });
840
+
841
+ var result = await response.json();
842
+
843
+ if (result.success) {
844
+ if (statusText) statusText.textContent = t('codexlens.indexCanceled') || 'Indexing canceled';
845
+ showRefreshToast(t('codexlens.indexCanceled') || 'Indexing canceled', 'info');
846
+
847
+ // Close the modal after a short delay
848
+ setTimeout(function() {
849
+ closeCodexLensIndexModal();
850
+ // Refresh status
851
+ if (typeof loadCodexLensStatus === 'function') {
852
+ loadCodexLensStatus().then(function() {
853
+ renderToolsSection();
854
+ if (window.lucide) lucide.createIcons();
855
+ });
856
+ }
857
+ }, 1000);
858
+ } else {
859
+ showRefreshToast(t('codexlens.cancelFailed') + ': ' + result.error, 'error');
860
+ // Re-enable button on failure
861
+ if (cancelBtn) {
862
+ cancelBtn.disabled = false;
863
+ cancelBtn.textContent = t('common.cancel');
864
+ }
865
+ }
866
+ } catch (err) {
867
+ console.error('[CodexLens] Cancel error:', err);
868
+ showRefreshToast(t('common.error') + ': ' + err.message, 'error');
869
+ // Re-enable button on error
870
+ if (cancelBtn) {
871
+ cancelBtn.disabled = false;
872
+ cancelBtn.textContent = t('common.cancel');
873
+ }
874
+ }
875
+ }
876
+
877
+ /**
878
+ * Install CodexLens
879
+ */
880
+ function installCodexLens() {
881
+ openCliInstallWizard('codexlens');
882
+ }
883
+
884
+ /**
885
+ * Uninstall CodexLens
886
+ */
887
+ function uninstallCodexLens() {
888
+ openCliUninstallWizard('codexlens');
889
+ }
890
+
891
+ /**
892
+ * Clean current workspace index
893
+ */
894
+ async function cleanCurrentWorkspaceIndex() {
895
+ if (!confirm(t('codexlens.cleanCurrentWorkspaceConfirm'))) {
896
+ return;
897
+ }
898
+
899
+ try {
900
+ showRefreshToast(t('codexlens.cleaning'), 'info');
901
+
902
+ // Get current workspace path (projectPath is a global variable from state.js)
903
+ var workspacePath = projectPath;
904
+
905
+ var response = await fetch('/api/codexlens/clean', {
906
+ method: 'POST',
907
+ headers: { 'Content-Type': 'application/json' },
908
+ body: JSON.stringify({ path: workspacePath })
909
+ });
910
+
911
+ var result = await response.json();
912
+
913
+ if (result.success) {
914
+ showRefreshToast(t('codexlens.cleanCurrentWorkspaceSuccess'), 'success');
915
+
916
+ // Refresh status
917
+ if (typeof loadCodexLensStatus === 'function') {
918
+ await loadCodexLensStatus();
919
+ renderToolsSection();
920
+ if (window.lucide) lucide.createIcons();
921
+ }
922
+ } else {
923
+ showRefreshToast(t('codexlens.cleanFailed') + ': ' + result.error, 'error');
924
+ }
925
+ } catch (err) {
926
+ showRefreshToast(t('common.error') + ': ' + err.message, 'error');
927
+ }
928
+ }
929
+
930
+ /**
931
+ * Clean all CodexLens indexes
932
+ */
933
+ async function cleanCodexLensIndexes() {
934
+ if (!confirm(t('codexlens.cleanConfirm'))) {
935
+ return;
936
+ }
937
+
938
+ try {
939
+ showRefreshToast(t('codexlens.cleaning'), 'info');
940
+
941
+ var response = await fetch('/api/codexlens/clean', {
942
+ method: 'POST',
943
+ headers: { 'Content-Type': 'application/json' },
944
+ body: JSON.stringify({ all: true })
945
+ });
946
+
947
+ var result = await response.json();
948
+
949
+ if (result.success) {
950
+ showRefreshToast(t('codexlens.cleanSuccess'), 'success');
951
+
952
+ // Refresh status
953
+ if (typeof loadCodexLensStatus === 'function') {
954
+ await loadCodexLensStatus();
955
+ renderToolsSection();
956
+ if (window.lucide) lucide.createIcons();
957
+ }
958
+ } else {
959
+ showRefreshToast(t('codexlens.cleanFailed') + ': ' + result.error, 'error');
960
+ }
961
+ } catch (err) {
962
+ showRefreshToast(t('common.error') + ': ' + err.message, 'error');
963
+ }
964
+ }