claude-code-workflow 6.1.3 → 6.2.1

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 (879) hide show
  1. package/README.md +145 -274
  2. package/bin/ccw-mcp.js +7 -0
  3. package/bin/ccw.js +10 -0
  4. package/dist/cli.d.ts +2 -0
  5. package/dist/cli.d.ts.map +1 -0
  6. package/dist/cli.js +219 -0
  7. package/dist/cli.js.map +1 -0
  8. package/dist/commands/cli.d.ts +32 -0
  9. package/dist/commands/cli.d.ts.map +1 -0
  10. package/dist/commands/cli.js +619 -0
  11. package/dist/commands/cli.js.map +1 -0
  12. package/dist/commands/core-memory.d.ts +32 -0
  13. package/dist/commands/core-memory.d.ts.map +1 -0
  14. package/dist/commands/core-memory.js +640 -0
  15. package/dist/commands/core-memory.js.map +1 -0
  16. package/dist/commands/hook.d.ts +16 -0
  17. package/dist/commands/hook.d.ts.map +1 -0
  18. package/dist/commands/hook.js +276 -0
  19. package/dist/commands/hook.js.map +1 -0
  20. package/dist/commands/install.d.ts +12 -0
  21. package/dist/commands/install.d.ts.map +1 -0
  22. package/dist/commands/install.js +443 -0
  23. package/dist/commands/install.js.map +1 -0
  24. package/dist/commands/list.d.ts +5 -0
  25. package/dist/commands/list.d.ts.map +1 -0
  26. package/dist/commands/list.js +32 -0
  27. package/dist/commands/list.js.map +1 -0
  28. package/dist/commands/memory.d.ts +57 -0
  29. package/dist/commands/memory.d.ts.map +1 -0
  30. package/dist/commands/memory.js +890 -0
  31. package/dist/commands/memory.js.map +1 -0
  32. package/dist/commands/serve.d.ts +12 -0
  33. package/dist/commands/serve.d.ts.map +1 -0
  34. package/dist/commands/serve.js +63 -0
  35. package/dist/commands/serve.js.map +1 -0
  36. package/dist/commands/session-path-resolver.d.ts +45 -0
  37. package/dist/commands/session-path-resolver.d.ts.map +1 -0
  38. package/dist/commands/session-path-resolver.js +302 -0
  39. package/dist/commands/session-path-resolver.js.map +1 -0
  40. package/dist/commands/session.d.ts +12 -0
  41. package/dist/commands/session.d.ts.map +1 -0
  42. package/dist/commands/session.js +954 -0
  43. package/dist/commands/session.js.map +1 -0
  44. package/dist/commands/stop.d.ts +11 -0
  45. package/dist/commands/stop.d.ts.map +1 -0
  46. package/dist/commands/stop.js +96 -0
  47. package/dist/commands/stop.js.map +1 -0
  48. package/dist/commands/tool.d.ts +29 -0
  49. package/dist/commands/tool.d.ts.map +1 -0
  50. package/dist/commands/tool.js +173 -0
  51. package/dist/commands/tool.js.map +1 -0
  52. package/dist/commands/uninstall.d.ts +9 -0
  53. package/dist/commands/uninstall.d.ts.map +1 -0
  54. package/dist/commands/uninstall.js +239 -0
  55. package/dist/commands/uninstall.js.map +1 -0
  56. package/dist/commands/upgrade.d.ts +10 -0
  57. package/dist/commands/upgrade.d.ts.map +1 -0
  58. package/dist/commands/upgrade.js +288 -0
  59. package/dist/commands/upgrade.js.map +1 -0
  60. package/dist/commands/view.d.ts +14 -0
  61. package/dist/commands/view.d.ts.map +1 -0
  62. package/dist/commands/view.js +100 -0
  63. package/dist/commands/view.js.map +1 -0
  64. package/dist/config/storage-paths.d.ts +184 -0
  65. package/dist/config/storage-paths.d.ts.map +1 -0
  66. package/dist/config/storage-paths.js +536 -0
  67. package/dist/config/storage-paths.js.map +1 -0
  68. package/dist/core/cache-manager.d.ts +80 -0
  69. package/dist/core/cache-manager.d.ts.map +1 -0
  70. package/dist/core/cache-manager.js +260 -0
  71. package/dist/core/cache-manager.js.map +1 -0
  72. package/dist/core/claude-freshness.d.ts +53 -0
  73. package/dist/core/claude-freshness.d.ts.map +1 -0
  74. package/dist/core/claude-freshness.js +232 -0
  75. package/dist/core/claude-freshness.js.map +1 -0
  76. package/dist/core/core-memory-store.d.ts +320 -0
  77. package/dist/core/core-memory-store.d.ts.map +1 -0
  78. package/dist/core/core-memory-store.js +1177 -0
  79. package/dist/core/core-memory-store.js.map +1 -0
  80. package/dist/core/dashboard-generator-patch.d.ts +2 -0
  81. package/dist/core/dashboard-generator-patch.d.ts.map +1 -0
  82. package/dist/core/dashboard-generator-patch.js +48 -0
  83. package/dist/core/dashboard-generator-patch.js.map +1 -0
  84. package/dist/core/dashboard-generator.d.ts +8 -0
  85. package/dist/core/dashboard-generator.d.ts.map +1 -0
  86. package/dist/core/dashboard-generator.js +695 -0
  87. package/dist/core/dashboard-generator.js.map +1 -0
  88. package/dist/core/data-aggregator.d.ts +145 -0
  89. package/dist/core/data-aggregator.d.ts.map +1 -0
  90. package/dist/core/data-aggregator.js +416 -0
  91. package/dist/core/data-aggregator.js.map +1 -0
  92. package/dist/core/history-importer.d.ts +102 -0
  93. package/dist/core/history-importer.d.ts.map +1 -0
  94. package/dist/core/history-importer.js +493 -0
  95. package/dist/core/history-importer.js.map +1 -0
  96. package/dist/core/lite-scanner-complete.d.ts +81 -0
  97. package/dist/core/lite-scanner-complete.d.ts.map +1 -0
  98. package/dist/core/lite-scanner-complete.js +368 -0
  99. package/dist/core/lite-scanner-complete.js.map +1 -0
  100. package/dist/core/lite-scanner.d.ts +81 -0
  101. package/dist/core/lite-scanner.d.ts.map +1 -0
  102. package/dist/core/lite-scanner.js +368 -0
  103. package/dist/core/lite-scanner.js.map +1 -0
  104. package/dist/core/manifest.d.ts +88 -0
  105. package/dist/core/manifest.d.ts.map +1 -0
  106. package/dist/core/manifest.js +214 -0
  107. package/dist/core/manifest.js.map +1 -0
  108. package/dist/core/memory-embedder-bridge.d.ts +83 -0
  109. package/dist/core/memory-embedder-bridge.d.ts.map +1 -0
  110. package/dist/core/memory-embedder-bridge.js +181 -0
  111. package/dist/core/memory-embedder-bridge.js.map +1 -0
  112. package/dist/core/memory-store.d.ts +249 -0
  113. package/dist/core/memory-store.d.ts.map +1 -0
  114. package/dist/core/memory-store.js +781 -0
  115. package/dist/core/memory-store.js.map +1 -0
  116. package/dist/core/routes/ccw-routes.d.ts +20 -0
  117. package/dist/core/routes/ccw-routes.d.ts.map +1 -0
  118. package/dist/core/routes/ccw-routes.js +70 -0
  119. package/dist/core/routes/ccw-routes.js.map +1 -0
  120. package/dist/core/routes/claude-routes.d.ts +19 -0
  121. package/dist/core/routes/claude-routes.d.ts.map +1 -0
  122. package/dist/core/routes/claude-routes.js +1017 -0
  123. package/dist/core/routes/claude-routes.js.map +1 -0
  124. package/dist/core/routes/cli-routes.d.ts +20 -0
  125. package/dist/core/routes/cli-routes.d.ts.map +1 -0
  126. package/dist/core/routes/cli-routes.js +468 -0
  127. package/dist/core/routes/cli-routes.js.map +1 -0
  128. package/dist/core/routes/codexlens-routes.d.ts +20 -0
  129. package/dist/core/routes/codexlens-routes.d.ts.map +1 -0
  130. package/dist/core/routes/codexlens-routes.js +754 -0
  131. package/dist/core/routes/codexlens-routes.js.map +1 -0
  132. package/dist/core/routes/core-memory-routes.d.ts +21 -0
  133. package/dist/core/routes/core-memory-routes.d.ts.map +1 -0
  134. package/dist/core/routes/core-memory-routes.js +520 -0
  135. package/dist/core/routes/core-memory-routes.js.map +1 -0
  136. package/dist/core/routes/files-routes.d.ts +20 -0
  137. package/dist/core/routes/files-routes.d.ts.map +1 -0
  138. package/dist/core/routes/files-routes.js +374 -0
  139. package/dist/core/routes/files-routes.js.map +1 -0
  140. package/dist/core/routes/graph-routes.d.ts +20 -0
  141. package/dist/core/routes/graph-routes.d.ts.map +1 -0
  142. package/dist/core/routes/graph-routes.js +517 -0
  143. package/dist/core/routes/graph-routes.js.map +1 -0
  144. package/dist/core/routes/help-routes.d.ts +20 -0
  145. package/dist/core/routes/help-routes.d.ts.map +1 -0
  146. package/dist/core/routes/help-routes.js +250 -0
  147. package/dist/core/routes/help-routes.js.map +1 -0
  148. package/dist/core/routes/hooks-routes.d.ts +21 -0
  149. package/dist/core/routes/hooks-routes.d.ts.map +1 -0
  150. package/dist/core/routes/hooks-routes.js +346 -0
  151. package/dist/core/routes/hooks-routes.js.map +1 -0
  152. package/dist/core/routes/mcp-routes.d.ts +20 -0
  153. package/dist/core/routes/mcp-routes.d.ts.map +1 -0
  154. package/dist/core/routes/mcp-routes.js +1129 -0
  155. package/dist/core/routes/mcp-routes.js.map +1 -0
  156. package/dist/core/routes/mcp-templates-db.d.ts +54 -0
  157. package/dist/core/routes/mcp-templates-db.d.ts.map +1 -0
  158. package/dist/core/routes/mcp-templates-db.js +226 -0
  159. package/dist/core/routes/mcp-templates-db.js.map +1 -0
  160. package/dist/core/routes/memory-routes.d.ts +21 -0
  161. package/dist/core/routes/memory-routes.d.ts.map +1 -0
  162. package/dist/core/routes/memory-routes.js +1095 -0
  163. package/dist/core/routes/memory-routes.js.map +1 -0
  164. package/dist/core/routes/rules-routes.d.ts +20 -0
  165. package/dist/core/routes/rules-routes.d.ts.map +1 -0
  166. package/dist/core/routes/rules-routes.js +442 -0
  167. package/dist/core/routes/rules-routes.js.map +1 -0
  168. package/dist/core/routes/session-routes.d.ts +20 -0
  169. package/dist/core/routes/session-routes.d.ts.map +1 -0
  170. package/dist/core/routes/session-routes.js +423 -0
  171. package/dist/core/routes/session-routes.js.map +1 -0
  172. package/dist/core/routes/skills-routes.d.ts +20 -0
  173. package/dist/core/routes/skills-routes.d.ts.map +1 -0
  174. package/dist/core/routes/skills-routes.js +533 -0
  175. package/dist/core/routes/skills-routes.js.map +1 -0
  176. package/dist/core/routes/status-routes.d.ts +20 -0
  177. package/dist/core/routes/status-routes.d.ts.map +1 -0
  178. package/dist/core/routes/status-routes.js +38 -0
  179. package/dist/core/routes/status-routes.js.map +1 -0
  180. package/dist/core/routes/system-routes.d.ts +22 -0
  181. package/dist/core/routes/system-routes.d.ts.map +1 -0
  182. package/dist/core/routes/system-routes.js +354 -0
  183. package/dist/core/routes/system-routes.js.map +1 -0
  184. package/dist/core/server.d.ts +17 -0
  185. package/dist/core/server.d.ts.map +1 -0
  186. package/dist/core/server.js +386 -0
  187. package/dist/core/server.js.map +1 -0
  188. package/dist/core/session-clustering-service.d.ts +153 -0
  189. package/dist/core/session-clustering-service.d.ts.map +1 -0
  190. package/dist/core/session-clustering-service.js +1065 -0
  191. package/dist/core/session-clustering-service.js.map +1 -0
  192. package/dist/core/session-scanner.d.ts +32 -0
  193. package/dist/core/session-scanner.d.ts.map +1 -0
  194. package/dist/core/session-scanner.js +253 -0
  195. package/dist/core/session-scanner.js.map +1 -0
  196. package/dist/core/websocket.d.ts +23 -0
  197. package/dist/core/websocket.d.ts.map +1 -0
  198. package/dist/core/websocket.js +168 -0
  199. package/dist/core/websocket.js.map +1 -0
  200. package/dist/index.d.ts +10 -0
  201. package/dist/index.d.ts.map +1 -0
  202. package/dist/index.js +10 -0
  203. package/dist/index.js.map +1 -0
  204. package/dist/mcp-server/index.d.ts +7 -0
  205. package/dist/mcp-server/index.d.ts.map +1 -0
  206. package/dist/mcp-server/index.js +157 -0
  207. package/dist/mcp-server/index.js.map +1 -0
  208. package/dist/tools/classify-folders.d.ts +26 -0
  209. package/dist/tools/classify-folders.d.ts.map +1 -0
  210. package/dist/tools/classify-folders.js +201 -0
  211. package/dist/tools/classify-folders.js.map +1 -0
  212. package/dist/tools/cli-config-manager.d.ts +62 -0
  213. package/dist/tools/cli-config-manager.d.ts.map +1 -0
  214. package/dist/tools/cli-config-manager.js +221 -0
  215. package/dist/tools/cli-config-manager.js.map +1 -0
  216. package/dist/tools/cli-executor.d.ts +373 -0
  217. package/dist/tools/cli-executor.d.ts.map +1 -0
  218. package/dist/tools/cli-executor.js +1625 -0
  219. package/dist/tools/cli-executor.js.map +1 -0
  220. package/dist/tools/cli-history-store.d.ts +330 -0
  221. package/dist/tools/cli-history-store.d.ts.map +1 -0
  222. package/dist/tools/cli-history-store.js +916 -0
  223. package/dist/tools/cli-history-store.js.map +1 -0
  224. package/dist/tools/codex-lens.d.ts +118 -0
  225. package/dist/tools/codex-lens.d.ts.map +1 -0
  226. package/dist/tools/codex-lens.js +962 -0
  227. package/dist/tools/codex-lens.js.map +1 -0
  228. package/dist/tools/convert-tokens-to-css.d.ts +14 -0
  229. package/dist/tools/convert-tokens-to-css.d.ts.map +1 -0
  230. package/dist/tools/convert-tokens-to-css.js +244 -0
  231. package/dist/tools/convert-tokens-to-css.js.map +1 -0
  232. package/dist/tools/core-memory.d.ts +66 -0
  233. package/dist/tools/core-memory.d.ts.map +1 -0
  234. package/dist/tools/core-memory.js +324 -0
  235. package/dist/tools/core-memory.js.map +1 -0
  236. package/dist/tools/detect-changed-modules.d.ts +24 -0
  237. package/dist/tools/detect-changed-modules.d.ts.map +1 -0
  238. package/dist/tools/detect-changed-modules.js +277 -0
  239. package/dist/tools/detect-changed-modules.js.map +1 -0
  240. package/dist/tools/discover-design-files.d.ts +36 -0
  241. package/dist/tools/discover-design-files.d.ts.map +1 -0
  242. package/dist/tools/discover-design-files.js +147 -0
  243. package/dist/tools/discover-design-files.js.map +1 -0
  244. package/dist/tools/edit-file.d.ts +28 -0
  245. package/dist/tools/edit-file.d.ts.map +1 -0
  246. package/dist/tools/edit-file.js +479 -0
  247. package/dist/tools/edit-file.js.map +1 -0
  248. package/dist/tools/generate-module-docs.d.ts +22 -0
  249. package/dist/tools/generate-module-docs.d.ts.map +1 -0
  250. package/dist/tools/generate-module-docs.js +379 -0
  251. package/dist/tools/generate-module-docs.js.map +1 -0
  252. package/dist/tools/get-modules-by-depth.d.ts +15 -0
  253. package/dist/tools/get-modules-by-depth.d.ts.map +1 -0
  254. package/dist/tools/get-modules-by-depth.js +296 -0
  255. package/dist/tools/get-modules-by-depth.js.map +1 -0
  256. package/dist/tools/index.d.ts +55 -0
  257. package/dist/tools/index.d.ts.map +1 -0
  258. package/dist/tools/index.js +304 -0
  259. package/dist/tools/index.js.map +1 -0
  260. package/dist/tools/native-session-discovery.d.ts +97 -0
  261. package/dist/tools/native-session-discovery.d.ts.map +1 -0
  262. package/dist/tools/native-session-discovery.js +700 -0
  263. package/dist/tools/native-session-discovery.js.map +1 -0
  264. package/dist/tools/notifier.d.ts +50 -0
  265. package/dist/tools/notifier.d.ts.map +1 -0
  266. package/dist/tools/notifier.js +90 -0
  267. package/dist/tools/notifier.js.map +1 -0
  268. package/dist/tools/read-file.d.ts +32 -0
  269. package/dist/tools/read-file.d.ts.map +1 -0
  270. package/dist/tools/read-file.js +329 -0
  271. package/dist/tools/read-file.js.map +1 -0
  272. package/dist/tools/resume-strategy.d.ts +48 -0
  273. package/dist/tools/resume-strategy.d.ts.map +1 -0
  274. package/dist/tools/resume-strategy.js +248 -0
  275. package/dist/tools/resume-strategy.js.map +1 -0
  276. package/dist/tools/session-content-parser.d.ts +58 -0
  277. package/dist/tools/session-content-parser.d.ts.map +1 -0
  278. package/dist/tools/session-content-parser.js +420 -0
  279. package/dist/tools/session-content-parser.js.map +1 -0
  280. package/dist/tools/session-manager.d.ts +9 -0
  281. package/dist/tools/session-manager.d.ts.map +1 -0
  282. package/dist/tools/session-manager.js +834 -0
  283. package/dist/tools/session-manager.js.map +1 -0
  284. package/dist/tools/smart-context.d.ts +35 -0
  285. package/dist/tools/smart-context.d.ts.map +1 -0
  286. package/dist/tools/smart-context.js +182 -0
  287. package/dist/tools/smart-context.js.map +1 -0
  288. package/dist/tools/smart-search.d.ts +105 -0
  289. package/dist/tools/smart-search.d.ts.map +1 -0
  290. package/dist/tools/smart-search.js +1753 -0
  291. package/dist/tools/smart-search.js.map +1 -0
  292. package/dist/tools/storage-manager.d.ts +114 -0
  293. package/dist/tools/storage-manager.d.ts.map +1 -0
  294. package/dist/tools/storage-manager.js +392 -0
  295. package/dist/tools/storage-manager.js.map +1 -0
  296. package/dist/tools/ui-generate-preview.d.ts +39 -0
  297. package/dist/tools/ui-generate-preview.d.ts.map +1 -0
  298. package/dist/tools/ui-generate-preview.js +300 -0
  299. package/dist/tools/ui-generate-preview.js.map +1 -0
  300. package/dist/tools/ui-instantiate-prototypes.d.ts +75 -0
  301. package/dist/tools/ui-instantiate-prototypes.d.ts.map +1 -0
  302. package/dist/tools/ui-instantiate-prototypes.js +256 -0
  303. package/dist/tools/ui-instantiate-prototypes.js.map +1 -0
  304. package/dist/tools/update-module-claude.d.ts +80 -0
  305. package/dist/tools/update-module-claude.d.ts.map +1 -0
  306. package/dist/tools/update-module-claude.js +351 -0
  307. package/dist/tools/update-module-claude.js.map +1 -0
  308. package/dist/tools/write-file.d.ts +19 -0
  309. package/dist/tools/write-file.d.ts.map +1 -0
  310. package/dist/tools/write-file.js +193 -0
  311. package/dist/tools/write-file.js.map +1 -0
  312. package/dist/types/config.d.ts +11 -0
  313. package/dist/types/config.d.ts.map +1 -0
  314. package/dist/types/config.js +2 -0
  315. package/dist/types/config.js.map +1 -0
  316. package/dist/types/index.d.ts +4 -0
  317. package/dist/types/index.d.ts.map +1 -0
  318. package/dist/types/index.js +4 -0
  319. package/dist/types/index.js.map +1 -0
  320. package/dist/types/session.d.ts +20 -0
  321. package/dist/types/session.d.ts.map +1 -0
  322. package/dist/types/session.js +2 -0
  323. package/dist/types/session.js.map +1 -0
  324. package/dist/types/tool.d.ts +36 -0
  325. package/dist/types/tool.d.ts.map +1 -0
  326. package/dist/types/tool.js +11 -0
  327. package/dist/types/tool.js.map +1 -0
  328. package/dist/utils/browser-launcher.d.ts +13 -0
  329. package/dist/utils/browser-launcher.d.ts.map +1 -0
  330. package/dist/utils/browser-launcher.js +60 -0
  331. package/dist/utils/browser-launcher.js.map +1 -0
  332. package/dist/utils/file-utils.d.ts +25 -0
  333. package/dist/utils/file-utils.d.ts.map +1 -0
  334. package/dist/utils/file-utils.js +48 -0
  335. package/dist/utils/file-utils.js.map +1 -0
  336. package/dist/utils/path-resolver.d.ts +80 -0
  337. package/dist/utils/path-resolver.d.ts.map +1 -0
  338. package/dist/utils/path-resolver.js +260 -0
  339. package/dist/utils/path-resolver.js.map +1 -0
  340. package/dist/utils/path-validator.d.ts +49 -0
  341. package/dist/utils/path-validator.d.ts.map +1 -0
  342. package/dist/utils/path-validator.js +123 -0
  343. package/dist/utils/path-validator.js.map +1 -0
  344. package/dist/utils/ui.d.ts +62 -0
  345. package/dist/utils/ui.d.ts.map +1 -0
  346. package/dist/utils/ui.js +129 -0
  347. package/dist/utils/ui.js.map +1 -0
  348. package/package.json +65 -67
  349. package/src/.workflow/.cli-history/history.db +0 -0
  350. package/src/.workflow/.cli-history/history.db-shm +0 -0
  351. package/src/.workflow/.cli-history/history.db-wal +0 -0
  352. package/src/cli.ts +244 -0
  353. package/src/commands/cli.ts +740 -0
  354. package/src/commands/core-memory.ts +770 -0
  355. package/src/commands/hook.ts +315 -0
  356. package/src/commands/install.ts +519 -0
  357. package/src/commands/list.ts +37 -0
  358. package/src/commands/memory.ts +1090 -0
  359. package/src/commands/serve.ts +76 -0
  360. package/src/commands/session-path-resolver.ts +372 -0
  361. package/src/commands/session.ts +1141 -0
  362. package/src/commands/stop.ts +111 -0
  363. package/src/commands/tool.ts +201 -0
  364. package/src/commands/uninstall.ts +287 -0
  365. package/src/commands/upgrade.ts +352 -0
  366. package/src/commands/view.ts +119 -0
  367. package/src/config/storage-paths.ts +670 -0
  368. package/src/core/cache-manager.ts +294 -0
  369. package/src/core/claude-freshness.ts +319 -0
  370. package/src/core/core-memory-store.ts +1528 -0
  371. package/src/core/dashboard-generator-patch.ts +47 -0
  372. package/src/core/dashboard-generator.ts +739 -0
  373. package/src/core/data-aggregator.ts +584 -0
  374. package/src/core/history-importer.ts +625 -0
  375. package/src/core/lite-scanner-complete.ts +469 -0
  376. package/src/core/lite-scanner.ts +469 -0
  377. package/src/core/manifest.ts +271 -0
  378. package/src/core/memory-embedder-bridge.ts +262 -0
  379. package/src/core/memory-store.ts +978 -0
  380. package/src/core/routes/ccw-routes.ts +96 -0
  381. package/src/core/routes/claude-routes.ts +1183 -0
  382. package/src/core/routes/cli-routes.ts +561 -0
  383. package/src/core/routes/codexlens-routes.ts +806 -0
  384. package/src/core/routes/core-memory-routes.ts +605 -0
  385. package/src/core/routes/files-routes.ts +428 -0
  386. package/src/core/routes/graph-routes.md +164 -0
  387. package/src/core/routes/graph-routes.ts +626 -0
  388. package/src/core/routes/help-routes.ts +308 -0
  389. package/src/core/routes/hooks-routes.ts +405 -0
  390. package/src/core/routes/mcp-routes.ts +1271 -0
  391. package/src/core/routes/mcp-routes.ts.backup +550 -0
  392. package/src/core/routes/mcp-templates-db.ts +268 -0
  393. package/src/core/routes/memory-routes.ts +1206 -0
  394. package/src/core/routes/rules-routes.ts +526 -0
  395. package/src/core/routes/session-routes.ts +467 -0
  396. package/src/core/routes/skills-routes.ts +599 -0
  397. package/src/core/routes/status-routes.ts +57 -0
  398. package/src/core/routes/system-routes.ts +427 -0
  399. package/src/core/server.ts +431 -0
  400. package/src/core/session-clustering-service.ts +1258 -0
  401. package/src/core/session-scanner.ts +283 -0
  402. package/src/core/websocket.ts +190 -0
  403. package/src/index.ts +10 -0
  404. package/src/mcp-server/index.ts +186 -0
  405. package/src/templates/assets/css/github-dark.min.css +10 -0
  406. package/src/templates/assets/css/github.min.css +10 -0
  407. package/src/templates/assets/js/cytoscape.min.js +32 -0
  408. package/src/templates/assets/js/d3.min.js +2 -0
  409. package/src/templates/assets/js/highlight.min.js +1244 -0
  410. package/src/templates/assets/js/lucide.min.js +12 -0
  411. package/src/templates/assets/js/marked.min.js +69 -0
  412. package/src/templates/assets/js/tailwind.js +83 -0
  413. package/src/templates/dashboard-css/01-base.css +302 -0
  414. package/src/templates/dashboard-css/02-session.css +748 -0
  415. package/src/templates/dashboard-css/04-lite-tasks.css +1181 -0
  416. package/src/templates/dashboard-css/06-cards.css +1576 -0
  417. package/src/templates/dashboard-css/07-managers.css +2107 -0
  418. package/src/templates/dashboard-css/09-explorer.css +1408 -0
  419. package/src/templates/dashboard-css/10-cli-status.css +337 -0
  420. package/src/templates/dashboard-css/11-cli-history.css +271 -0
  421. package/src/templates/dashboard-css/12-cli-legacy.css +796 -0
  422. package/src/templates/dashboard-css/13-cli-ccw.css +199 -0
  423. package/src/templates/dashboard-css/14-cli-modals.css +258 -0
  424. package/src/templates/dashboard-css/15-cli-endpoints.css +305 -0
  425. package/src/templates/dashboard-css/16-cli-session.css +241 -0
  426. package/src/templates/dashboard-css/17-cli-conversation.css +283 -0
  427. package/src/templates/dashboard-css/18-cli-settings.css +160 -0
  428. package/src/templates/dashboard-css/19-cli-native-session.css +496 -0
  429. package/src/templates/dashboard-css/20-cli-taskqueue.css +188 -0
  430. package/src/templates/dashboard-css/21-cli-toolmgmt.css +310 -0
  431. package/src/templates/dashboard-css/22-cli-semantic.css +240 -0
  432. package/src/templates/dashboard-css/23-memory.css +2390 -0
  433. package/src/templates/dashboard-css/24-prompt-history.css +1089 -0
  434. package/src/templates/dashboard-css/25-skills-rules.css +326 -0
  435. package/src/templates/dashboard-css/26-claude-manager.css +908 -0
  436. package/src/templates/dashboard-css/27-graph-explorer.css +1678 -0
  437. package/src/templates/dashboard-css/28-mcp-manager.css +748 -0
  438. package/src/templates/dashboard-css/29-help.css +264 -0
  439. package/src/templates/dashboard-css/30-core-memory.css +1700 -0
  440. package/src/templates/dashboard-js/api.js +220 -0
  441. package/src/templates/dashboard-js/components/carousel.js +398 -0
  442. package/src/templates/dashboard-js/components/cli-history.js +876 -0
  443. package/src/templates/dashboard-js/components/cli-status.js +978 -0
  444. package/src/templates/dashboard-js/components/global-notifications.js +508 -0
  445. package/src/templates/dashboard-js/components/hook-manager.js +1278 -0
  446. package/src/templates/dashboard-js/components/index-manager.js +302 -0
  447. package/src/templates/dashboard-js/components/mcp-manager.js +1219 -0
  448. package/src/templates/dashboard-js/components/modals.js +326 -0
  449. package/src/templates/dashboard-js/components/navigation.js +313 -0
  450. package/src/templates/dashboard-js/components/notifications.js +758 -0
  451. package/src/templates/dashboard-js/components/storage-manager.js +478 -0
  452. package/src/templates/dashboard-js/components/tabs-other.js +424 -0
  453. package/src/templates/dashboard-js/components/task-queue-sidebar.js +716 -0
  454. package/src/templates/dashboard-js/help-i18n.js +272 -0
  455. package/src/templates/dashboard-js/i18n.js +2807 -0
  456. package/src/templates/dashboard-js/main.js +87 -0
  457. package/src/templates/dashboard-js/state.js +243 -0
  458. package/src/templates/dashboard-js/utils.js +199 -0
  459. package/src/templates/dashboard-js/views/claude-manager.js +912 -0
  460. package/src/templates/dashboard-js/views/cli-manager.js +2272 -0
  461. package/src/templates/dashboard-js/views/codexlens-manager.js +964 -0
  462. package/src/templates/dashboard-js/views/core-memory-clusters.js +503 -0
  463. package/src/templates/dashboard-js/views/core-memory.js +782 -0
  464. package/src/templates/dashboard-js/views/explorer.js +888 -0
  465. package/src/templates/dashboard-js/views/graph-explorer.js +1157 -0
  466. package/src/templates/dashboard-js/views/help.js +856 -0
  467. package/src/templates/dashboard-js/views/history.js +337 -0
  468. package/src/templates/dashboard-js/views/home.js +243 -0
  469. package/src/templates/dashboard-js/views/hook-manager.js +660 -0
  470. package/src/templates/dashboard-js/views/lite-tasks.js +861 -0
  471. package/src/templates/dashboard-js/views/mcp-manager.js +2187 -0
  472. package/src/templates/dashboard-js/views/mcp-manager.js.backup +1729 -0
  473. package/src/templates/dashboard-js/views/mcp-manager.js.new +928 -0
  474. package/src/templates/dashboard-js/views/memory.js +1221 -0
  475. package/src/templates/dashboard-js/views/prompt-history.js +713 -0
  476. package/src/templates/dashboard-js/views/rules-manager.js +828 -0
  477. package/src/templates/dashboard-js/views/session-detail.js +781 -0
  478. package/src/templates/dashboard-js/views/skills-manager.js +819 -0
  479. package/src/templates/dashboard.html +831 -0
  480. package/src/templates/hooks-config-example.json +60 -0
  481. package/src/tools/classify-folders.ts +245 -0
  482. package/src/tools/cli-config-manager.ts +268 -0
  483. package/src/tools/cli-executor.ts +2014 -0
  484. package/src/tools/cli-history-store.ts +1195 -0
  485. package/src/tools/codex-lens.ts +1141 -0
  486. package/src/tools/convert-tokens-to-css.ts +300 -0
  487. package/src/tools/core-memory.ts +444 -0
  488. package/src/tools/detect-changed-modules.ts +325 -0
  489. package/src/tools/discover-design-files.ts +184 -0
  490. package/src/tools/edit-file.ts +568 -0
  491. package/src/tools/generate-module-docs.ts +438 -0
  492. package/src/tools/get-modules-by-depth.ts +349 -0
  493. package/src/tools/index.ts +370 -0
  494. package/src/tools/native-session-discovery.ts +795 -0
  495. package/src/tools/notifier.ts +129 -0
  496. package/src/tools/read-file.ts +410 -0
  497. package/src/tools/resume-strategy.ts +345 -0
  498. package/src/tools/session-content-parser.ts +619 -0
  499. package/src/tools/session-manager.ts +1026 -0
  500. package/src/tools/smart-context.ts +228 -0
  501. package/src/tools/smart-search.ts +2065 -0
  502. package/src/tools/smart-search.ts.backup +1233 -0
  503. package/src/tools/storage-manager.ts +455 -0
  504. package/src/tools/write-file.ts +222 -0
  505. package/src/types/config.ts +11 -0
  506. package/src/types/index.ts +3 -0
  507. package/src/types/session.ts +25 -0
  508. package/src/types/tool.ts +41 -0
  509. package/src/utils/browser-launcher.ts +62 -0
  510. package/src/utils/file-utils.ts +48 -0
  511. package/src/utils/path-resolver.ts +315 -0
  512. package/src/utils/path-validator.ts +153 -0
  513. package/src/utils/ui.ts +155 -0
  514. package/.claude/agents/action-planning-agent.md +0 -778
  515. package/.claude/agents/cli-execution-agent.md +0 -270
  516. package/.claude/agents/cli-explore-agent.md +0 -182
  517. package/.claude/agents/cli-lite-planning-agent.md +0 -396
  518. package/.claude/agents/cli-planning-agent.md +0 -558
  519. package/.claude/agents/code-developer.md +0 -310
  520. package/.claude/agents/conceptual-planning-agent.md +0 -308
  521. package/.claude/agents/context-search-agent.md +0 -582
  522. package/.claude/agents/doc-generator.md +0 -330
  523. package/.claude/agents/memory-bridge.md +0 -94
  524. package/.claude/agents/test-context-search-agent.md +0 -399
  525. package/.claude/agents/test-fix-agent.md +0 -343
  526. package/.claude/agents/ui-design-agent.md +0 -593
  527. package/.claude/agents/universal-executor.md +0 -131
  528. package/.claude/commands/cli/cli-init.md +0 -440
  529. package/.claude/commands/enhance-prompt.md +0 -93
  530. package/.claude/commands/memory/code-map-memory.md +0 -687
  531. package/.claude/commands/memory/docs-full-cli.md +0 -471
  532. package/.claude/commands/memory/docs-related-cli.md +0 -386
  533. package/.claude/commands/memory/docs.md +0 -615
  534. package/.claude/commands/memory/load-skill-memory.md +0 -182
  535. package/.claude/commands/memory/load.md +0 -240
  536. package/.claude/commands/memory/skill-memory.md +0 -525
  537. package/.claude/commands/memory/style-skill-memory.md +0 -396
  538. package/.claude/commands/memory/tech-research.md +0 -477
  539. package/.claude/commands/memory/update-full.md +0 -332
  540. package/.claude/commands/memory/update-related.md +0 -332
  541. package/.claude/commands/memory/workflow-skill-memory.md +0 -517
  542. package/.claude/commands/task/breakdown.md +0 -204
  543. package/.claude/commands/task/create.md +0 -152
  544. package/.claude/commands/task/execute.md +0 -270
  545. package/.claude/commands/task/replan.md +0 -437
  546. package/.claude/commands/version.md +0 -254
  547. package/.claude/commands/workflow/action-plan-verify.md +0 -447
  548. package/.claude/commands/workflow/brainstorm/api-designer.md +0 -585
  549. package/.claude/commands/workflow/brainstorm/artifacts.md +0 -452
  550. package/.claude/commands/workflow/brainstorm/auto-parallel.md +0 -443
  551. package/.claude/commands/workflow/brainstorm/data-architect.md +0 -220
  552. package/.claude/commands/workflow/brainstorm/product-manager.md +0 -200
  553. package/.claude/commands/workflow/brainstorm/product-owner.md +0 -200
  554. package/.claude/commands/workflow/brainstorm/scrum-master.md +0 -200
  555. package/.claude/commands/workflow/brainstorm/subject-matter-expert.md +0 -200
  556. package/.claude/commands/workflow/brainstorm/synthesis.md +0 -398
  557. package/.claude/commands/workflow/brainstorm/system-architect.md +0 -387
  558. package/.claude/commands/workflow/brainstorm/ui-designer.md +0 -221
  559. package/.claude/commands/workflow/brainstorm/ux-expert.md +0 -221
  560. package/.claude/commands/workflow/execute.md +0 -460
  561. package/.claude/commands/workflow/init.md +0 -164
  562. package/.claude/commands/workflow/lite-execute.md +0 -686
  563. package/.claude/commands/workflow/lite-fix.md +0 -621
  564. package/.claude/commands/workflow/lite-plan.md +0 -592
  565. package/.claude/commands/workflow/plan.md +0 -551
  566. package/.claude/commands/workflow/replan.md +0 -515
  567. package/.claude/commands/workflow/review-fix.md +0 -606
  568. package/.claude/commands/workflow/review-module-cycle.md +0 -765
  569. package/.claude/commands/workflow/review-session-cycle.md +0 -776
  570. package/.claude/commands/workflow/review.md +0 -291
  571. package/.claude/commands/workflow/session/complete.md +0 -500
  572. package/.claude/commands/workflow/session/list.md +0 -96
  573. package/.claude/commands/workflow/session/resume.md +0 -61
  574. package/.claude/commands/workflow/session/start.md +0 -200
  575. package/.claude/commands/workflow/tdd-plan.md +0 -460
  576. package/.claude/commands/workflow/tdd-verify.md +0 -386
  577. package/.claude/commands/workflow/test-cycle-execute.md +0 -498
  578. package/.claude/commands/workflow/test-fix-gen.md +0 -699
  579. package/.claude/commands/workflow/test-gen.md +0 -529
  580. package/.claude/commands/workflow/tools/conflict-resolution.md +0 -680
  581. package/.claude/commands/workflow/tools/context-gather.md +0 -434
  582. package/.claude/commands/workflow/tools/task-generate-agent.md +0 -291
  583. package/.claude/commands/workflow/tools/task-generate-tdd.md +0 -518
  584. package/.claude/commands/workflow/tools/tdd-coverage-analysis.md +0 -309
  585. package/.claude/commands/workflow/tools/test-concept-enhanced.md +0 -163
  586. package/.claude/commands/workflow/tools/test-context-gather.md +0 -235
  587. package/.claude/commands/workflow/tools/test-task-generate.md +0 -256
  588. package/.claude/commands/workflow/ui-design/animation-extract.md +0 -1150
  589. package/.claude/commands/workflow/ui-design/codify-style.md +0 -652
  590. package/.claude/commands/workflow/ui-design/design-sync.md +0 -454
  591. package/.claude/commands/workflow/ui-design/explore-auto.md +0 -678
  592. package/.claude/commands/workflow/ui-design/generate.md +0 -504
  593. package/.claude/commands/workflow/ui-design/imitate-auto.md +0 -745
  594. package/.claude/commands/workflow/ui-design/import-from-code.md +0 -537
  595. package/.claude/commands/workflow/ui-design/layout-extract.md +0 -788
  596. package/.claude/commands/workflow/ui-design/reference-page-generator.md +0 -356
  597. package/.claude/commands/workflow/ui-design/style-extract.md +0 -773
  598. package/.claude/scripts/classify-folders.sh +0 -39
  599. package/.claude/scripts/convert_tokens_to_css.sh +0 -229
  600. package/.claude/scripts/detect_changed_modules.sh +0 -161
  601. package/.claude/scripts/discover-design-files.sh +0 -87
  602. package/.claude/scripts/extract-animations.js +0 -243
  603. package/.claude/scripts/extract-computed-styles.js +0 -118
  604. package/.claude/scripts/extract-layout-structure.js +0 -411
  605. package/.claude/scripts/generate_module_docs.sh +0 -717
  606. package/.claude/scripts/get_modules_by_depth.sh +0 -170
  607. package/.claude/scripts/ui-generate-preview.sh +0 -395
  608. package/.claude/scripts/ui-instantiate-prototypes.sh +0 -815
  609. package/.claude/scripts/update_module_claude.sh +0 -337
  610. package/.claude/skills/command-guide/SKILL.md +0 -388
  611. package/.claude/skills/command-guide/UPDATE-GUIDELINE.md +0 -592
  612. package/.claude/skills/command-guide/guides/cli-tools-guide.md +0 -410
  613. package/.claude/skills/command-guide/guides/examples.md +0 -537
  614. package/.claude/skills/command-guide/guides/getting-started.md +0 -242
  615. package/.claude/skills/command-guide/guides/implementation-details.md +0 -1010
  616. package/.claude/skills/command-guide/guides/index-structure.md +0 -326
  617. package/.claude/skills/command-guide/guides/troubleshooting.md +0 -92
  618. package/.claude/skills/command-guide/guides/ui-design-workflow-guide.md +0 -316
  619. package/.claude/skills/command-guide/guides/workflow-patterns.md +0 -662
  620. package/.claude/skills/command-guide/index/all-commands.json +0 -772
  621. package/.claude/skills/command-guide/index/by-category.json +0 -800
  622. package/.claude/skills/command-guide/index/by-use-case.json +0 -786
  623. package/.claude/skills/command-guide/index/command-relationships.json +0 -307
  624. package/.claude/skills/command-guide/index/essential-commands.json +0 -112
  625. package/.claude/skills/command-guide/reference/agents/action-planning-agent.md +0 -778
  626. package/.claude/skills/command-guide/reference/agents/cli-execution-agent.md +0 -270
  627. package/.claude/skills/command-guide/reference/agents/cli-explore-agent.md +0 -182
  628. package/.claude/skills/command-guide/reference/agents/cli-lite-planning-agent.md +0 -396
  629. package/.claude/skills/command-guide/reference/agents/cli-planning-agent.md +0 -558
  630. package/.claude/skills/command-guide/reference/agents/code-developer.md +0 -310
  631. package/.claude/skills/command-guide/reference/agents/conceptual-planning-agent.md +0 -308
  632. package/.claude/skills/command-guide/reference/agents/context-search-agent.md +0 -582
  633. package/.claude/skills/command-guide/reference/agents/doc-generator.md +0 -330
  634. package/.claude/skills/command-guide/reference/agents/memory-bridge.md +0 -94
  635. package/.claude/skills/command-guide/reference/agents/test-context-search-agent.md +0 -399
  636. package/.claude/skills/command-guide/reference/agents/test-fix-agent.md +0 -343
  637. package/.claude/skills/command-guide/reference/agents/ui-design-agent.md +0 -593
  638. package/.claude/skills/command-guide/reference/agents/universal-executor.md +0 -131
  639. package/.claude/skills/command-guide/reference/commands/cli/cli-init.md +0 -440
  640. package/.claude/skills/command-guide/reference/commands/enhance-prompt.md +0 -93
  641. package/.claude/skills/command-guide/reference/commands/memory/code-map-memory.md +0 -687
  642. package/.claude/skills/command-guide/reference/commands/memory/docs-full-cli.md +0 -471
  643. package/.claude/skills/command-guide/reference/commands/memory/docs-related-cli.md +0 -386
  644. package/.claude/skills/command-guide/reference/commands/memory/docs.md +0 -615
  645. package/.claude/skills/command-guide/reference/commands/memory/load-skill-memory.md +0 -182
  646. package/.claude/skills/command-guide/reference/commands/memory/load.md +0 -240
  647. package/.claude/skills/command-guide/reference/commands/memory/skill-memory.md +0 -525
  648. package/.claude/skills/command-guide/reference/commands/memory/style-skill-memory.md +0 -396
  649. package/.claude/skills/command-guide/reference/commands/memory/tech-research.md +0 -477
  650. package/.claude/skills/command-guide/reference/commands/memory/update-full.md +0 -332
  651. package/.claude/skills/command-guide/reference/commands/memory/update-related.md +0 -332
  652. package/.claude/skills/command-guide/reference/commands/memory/workflow-skill-memory.md +0 -517
  653. package/.claude/skills/command-guide/reference/commands/task/breakdown.md +0 -204
  654. package/.claude/skills/command-guide/reference/commands/task/create.md +0 -152
  655. package/.claude/skills/command-guide/reference/commands/task/execute.md +0 -270
  656. package/.claude/skills/command-guide/reference/commands/task/replan.md +0 -437
  657. package/.claude/skills/command-guide/reference/commands/version.md +0 -254
  658. package/.claude/skills/command-guide/reference/commands/workflow/action-plan-verify.md +0 -447
  659. package/.claude/skills/command-guide/reference/commands/workflow/brainstorm/api-designer.md +0 -585
  660. package/.claude/skills/command-guide/reference/commands/workflow/brainstorm/artifacts.md +0 -452
  661. package/.claude/skills/command-guide/reference/commands/workflow/brainstorm/auto-parallel.md +0 -443
  662. package/.claude/skills/command-guide/reference/commands/workflow/brainstorm/data-architect.md +0 -220
  663. package/.claude/skills/command-guide/reference/commands/workflow/brainstorm/product-manager.md +0 -200
  664. package/.claude/skills/command-guide/reference/commands/workflow/brainstorm/product-owner.md +0 -200
  665. package/.claude/skills/command-guide/reference/commands/workflow/brainstorm/scrum-master.md +0 -200
  666. package/.claude/skills/command-guide/reference/commands/workflow/brainstorm/subject-matter-expert.md +0 -200
  667. package/.claude/skills/command-guide/reference/commands/workflow/brainstorm/synthesis.md +0 -398
  668. package/.claude/skills/command-guide/reference/commands/workflow/brainstorm/system-architect.md +0 -387
  669. package/.claude/skills/command-guide/reference/commands/workflow/brainstorm/ui-designer.md +0 -221
  670. package/.claude/skills/command-guide/reference/commands/workflow/brainstorm/ux-expert.md +0 -221
  671. package/.claude/skills/command-guide/reference/commands/workflow/execute.md +0 -460
  672. package/.claude/skills/command-guide/reference/commands/workflow/init.md +0 -164
  673. package/.claude/skills/command-guide/reference/commands/workflow/lite-execute.md +0 -686
  674. package/.claude/skills/command-guide/reference/commands/workflow/lite-fix.md +0 -621
  675. package/.claude/skills/command-guide/reference/commands/workflow/lite-plan.md +0 -592
  676. package/.claude/skills/command-guide/reference/commands/workflow/plan.md +0 -551
  677. package/.claude/skills/command-guide/reference/commands/workflow/replan.md +0 -515
  678. package/.claude/skills/command-guide/reference/commands/workflow/review-fix.md +0 -606
  679. package/.claude/skills/command-guide/reference/commands/workflow/review-module-cycle.md +0 -765
  680. package/.claude/skills/command-guide/reference/commands/workflow/review-session-cycle.md +0 -776
  681. package/.claude/skills/command-guide/reference/commands/workflow/review.md +0 -291
  682. package/.claude/skills/command-guide/reference/commands/workflow/session/complete.md +0 -500
  683. package/.claude/skills/command-guide/reference/commands/workflow/session/list.md +0 -96
  684. package/.claude/skills/command-guide/reference/commands/workflow/session/resume.md +0 -61
  685. package/.claude/skills/command-guide/reference/commands/workflow/session/start.md +0 -200
  686. package/.claude/skills/command-guide/reference/commands/workflow/tdd-plan.md +0 -460
  687. package/.claude/skills/command-guide/reference/commands/workflow/tdd-verify.md +0 -386
  688. package/.claude/skills/command-guide/reference/commands/workflow/test-cycle-execute.md +0 -498
  689. package/.claude/skills/command-guide/reference/commands/workflow/test-fix-gen.md +0 -699
  690. package/.claude/skills/command-guide/reference/commands/workflow/test-gen.md +0 -529
  691. package/.claude/skills/command-guide/reference/commands/workflow/tools/conflict-resolution.md +0 -680
  692. package/.claude/skills/command-guide/reference/commands/workflow/tools/context-gather.md +0 -434
  693. package/.claude/skills/command-guide/reference/commands/workflow/tools/task-generate-agent.md +0 -291
  694. package/.claude/skills/command-guide/reference/commands/workflow/tools/task-generate-tdd.md +0 -518
  695. package/.claude/skills/command-guide/reference/commands/workflow/tools/tdd-coverage-analysis.md +0 -309
  696. package/.claude/skills/command-guide/reference/commands/workflow/tools/test-concept-enhanced.md +0 -163
  697. package/.claude/skills/command-guide/reference/commands/workflow/tools/test-context-gather.md +0 -235
  698. package/.claude/skills/command-guide/reference/commands/workflow/tools/test-task-generate.md +0 -256
  699. package/.claude/skills/command-guide/reference/commands/workflow/ui-design/animation-extract.md +0 -1150
  700. package/.claude/skills/command-guide/reference/commands/workflow/ui-design/codify-style.md +0 -652
  701. package/.claude/skills/command-guide/reference/commands/workflow/ui-design/design-sync.md +0 -454
  702. package/.claude/skills/command-guide/reference/commands/workflow/ui-design/explore-auto.md +0 -678
  703. package/.claude/skills/command-guide/reference/commands/workflow/ui-design/generate.md +0 -504
  704. package/.claude/skills/command-guide/reference/commands/workflow/ui-design/imitate-auto.md +0 -745
  705. package/.claude/skills/command-guide/reference/commands/workflow/ui-design/import-from-code.md +0 -537
  706. package/.claude/skills/command-guide/reference/commands/workflow/ui-design/layout-extract.md +0 -788
  707. package/.claude/skills/command-guide/reference/commands/workflow/ui-design/reference-page-generator.md +0 -356
  708. package/.claude/skills/command-guide/reference/commands/workflow/ui-design/style-extract.md +0 -773
  709. package/.claude/skills/command-guide/scripts/analyze_commands.py +0 -502
  710. package/.claude/skills/command-guide/scripts/update-index.sh +0 -130
  711. package/.claude/skills/command-guide/templates/issue-bug.md +0 -104
  712. package/.claude/skills/command-guide/templates/issue-diagnosis.md +0 -275
  713. package/.claude/skills/command-guide/templates/issue-feature.md +0 -97
  714. package/.claude/skills/command-guide/templates/issue-question.md +0 -141
  715. package/.claude/skills/prompt-enhancer/SKILL.md +0 -124
  716. package/.claude/workflows/_template-compare-matrix.html +0 -692
  717. package/.claude/workflows/cli-templates/fix-plan-template.json +0 -75
  718. package/.claude/workflows/cli-templates/fix-progress-template.json +0 -48
  719. package/.claude/workflows/cli-templates/memory/style-skill-memory/skill-md-template.md +0 -299
  720. package/.claude/workflows/cli-templates/planning-roles/data-architect.md +0 -120
  721. package/.claude/workflows/cli-templates/planning-roles/product-manager.md +0 -119
  722. package/.claude/workflows/cli-templates/planning-roles/product-owner.md +0 -261
  723. package/.claude/workflows/cli-templates/planning-roles/scrum-master.md +0 -186
  724. package/.claude/workflows/cli-templates/planning-roles/subject-matter-expert.md +0 -281
  725. package/.claude/workflows/cli-templates/planning-roles/synthesis-role.md +0 -414
  726. package/.claude/workflows/cli-templates/planning-roles/system-architect.md +0 -106
  727. package/.claude/workflows/cli-templates/planning-roles/test-strategist.md +0 -124
  728. package/.claude/workflows/cli-templates/planning-roles/ui-designer.md +0 -379
  729. package/.claude/workflows/cli-templates/planning-roles/ux-expert.md +0 -240
  730. package/.claude/workflows/cli-templates/prompts/analysis/01-diagnose-bug-root-cause.txt +0 -127
  731. package/.claude/workflows/cli-templates/prompts/analysis/01-trace-code-execution.txt +0 -115
  732. package/.claude/workflows/cli-templates/prompts/analysis/02-analyze-code-patterns.txt +0 -37
  733. package/.claude/workflows/cli-templates/prompts/analysis/02-analyze-technical-document.txt +0 -33
  734. package/.claude/workflows/cli-templates/prompts/analysis/02-review-architecture.txt +0 -29
  735. package/.claude/workflows/cli-templates/prompts/analysis/02-review-code-quality.txt +0 -28
  736. package/.claude/workflows/cli-templates/prompts/analysis/03-analyze-performance.txt +0 -29
  737. package/.claude/workflows/cli-templates/prompts/analysis/03-assess-security-risks.txt +0 -29
  738. package/.claude/workflows/cli-templates/prompts/analysis/03-review-quality-standards.txt +0 -29
  739. package/.claude/workflows/cli-templates/prompts/development/02-generate-tests.txt +0 -70
  740. package/.claude/workflows/cli-templates/prompts/development/02-implement-component-ui.txt +0 -55
  741. package/.claude/workflows/cli-templates/prompts/development/02-implement-feature.txt +0 -58
  742. package/.claude/workflows/cli-templates/prompts/development/02-refactor-codebase.txt +0 -55
  743. package/.claude/workflows/cli-templates/prompts/development/03-debug-runtime-issues.txt +0 -55
  744. package/.claude/workflows/cli-templates/prompts/documentation/api.txt +0 -15
  745. package/.claude/workflows/cli-templates/prompts/documentation/folder-navigation.txt +0 -27
  746. package/.claude/workflows/cli-templates/prompts/documentation/module-readme.txt +0 -49
  747. package/.claude/workflows/cli-templates/prompts/documentation/project-architecture.txt +0 -41
  748. package/.claude/workflows/cli-templates/prompts/documentation/project-examples.txt +0 -35
  749. package/.claude/workflows/cli-templates/prompts/documentation/project-readme.txt +0 -35
  750. package/.claude/workflows/cli-templates/prompts/memory/02-document-module-structure.txt +0 -165
  751. package/.claude/workflows/cli-templates/prompts/planning/01-plan-architecture-design.txt +0 -109
  752. package/.claude/workflows/cli-templates/prompts/planning/02-breakdown-task-steps.txt +0 -30
  753. package/.claude/workflows/cli-templates/prompts/planning/02-design-component-spec.txt +0 -28
  754. package/.claude/workflows/cli-templates/prompts/planning/03-evaluate-concept-feasibility.txt +0 -127
  755. package/.claude/workflows/cli-templates/prompts/planning/03-plan-migration-strategy.txt +0 -30
  756. package/.claude/workflows/cli-templates/prompts/tech/tech-module-format.txt +0 -359
  757. package/.claude/workflows/cli-templates/prompts/tech/tech-skill-index.txt +0 -185
  758. package/.claude/workflows/cli-templates/prompts/test/test-concept-analysis.txt +0 -179
  759. package/.claude/workflows/cli-templates/prompts/universal/00-universal-creative-style.txt +0 -95
  760. package/.claude/workflows/cli-templates/prompts/universal/00-universal-rigorous-style.txt +0 -92
  761. package/.claude/workflows/cli-templates/prompts/verification/codex-technical.txt +0 -28
  762. package/.claude/workflows/cli-templates/prompts/verification/cross-validation.txt +0 -28
  763. package/.claude/workflows/cli-templates/prompts/verification/gemini-strategic.txt +0 -27
  764. package/.claude/workflows/cli-templates/prompts/workflow/analysis-results-structure.txt +0 -224
  765. package/.claude/workflows/cli-templates/prompts/workflow/codex-feasibility-validation.txt +0 -176
  766. package/.claude/workflows/cli-templates/prompts/workflow/gemini-solution-design.txt +0 -131
  767. package/.claude/workflows/cli-templates/prompts/workflow/impl-plan-template.txt +0 -286
  768. package/.claude/workflows/cli-templates/prompts/workflow/skill-aggregation.txt +0 -172
  769. package/.claude/workflows/cli-templates/prompts/workflow/skill-conflict-patterns.txt +0 -98
  770. package/.claude/workflows/cli-templates/prompts/workflow/skill-index.txt +0 -224
  771. package/.claude/workflows/cli-templates/prompts/workflow/skill-lessons-learned.txt +0 -98
  772. package/.claude/workflows/cli-templates/prompts/workflow/skill-sessions-timeline.txt +0 -53
  773. package/.claude/workflows/cli-templates/prompts/workflow/task-json-agent-mode.txt +0 -123
  774. package/.claude/workflows/cli-templates/prompts/workflow/task-json-cli-mode.txt +0 -182
  775. package/.claude/workflows/cli-templates/schemas/diagnosis-json-schema.json +0 -234
  776. package/.claude/workflows/cli-templates/schemas/explore-json-schema.json +0 -124
  777. package/.claude/workflows/cli-templates/schemas/fix-plan-json-schema.json +0 -273
  778. package/.claude/workflows/cli-templates/schemas/plan-json-schema.json +0 -219
  779. package/.claude/workflows/cli-templates/schemas/project-json-schema.json +0 -221
  780. package/.claude/workflows/cli-templates/schemas/review-deep-dive-results-schema.json +0 -82
  781. package/.claude/workflows/cli-templates/schemas/review-dimension-results-schema.json +0 -51
  782. package/.claude/workflows/cli-templates/tech-stacks/go-dev.md +0 -91
  783. package/.claude/workflows/cli-templates/tech-stacks/java-dev.md +0 -107
  784. package/.claude/workflows/cli-templates/tech-stacks/javascript-dev.md +0 -58
  785. package/.claude/workflows/cli-templates/tech-stacks/python-dev.md +0 -79
  786. package/.claude/workflows/cli-templates/tech-stacks/react-dev.md +0 -103
  787. package/.claude/workflows/cli-templates/tech-stacks/typescript-dev.md +0 -83
  788. package/.claude/workflows/cli-templates/ui-design/systems/animation-tokens.json +0 -247
  789. package/.claude/workflows/cli-templates/ui-design/systems/design-tokens.json +0 -342
  790. package/.claude/workflows/cli-templates/ui-design/systems/layout-templates.json +0 -145
  791. package/.claude/workflows/context-search-strategy.md +0 -77
  792. package/.claude/workflows/intelligent-tools-strategy.md +0 -662
  793. package/.claude/workflows/review-directory-specification.md +0 -336
  794. package/.claude/workflows/task-core.md +0 -214
  795. package/.claude/workflows/tool-strategy.md +0 -79
  796. package/.claude/workflows/workflow-architecture.md +0 -942
  797. package/.codex/AGENTS.md +0 -330
  798. package/.gemini/GEMINI.md +0 -164
  799. package/.qwen/QWEN.md +0 -164
  800. package/CLAUDE.md +0 -91
  801. package/LICENSE +0 -21
  802. package/ccw/README.md +0 -121
  803. package/ccw/bin/ccw.js +0 -10
  804. package/ccw/package.json +0 -47
  805. package/ccw/src/cli.js +0 -119
  806. package/ccw/src/commands/install.js +0 -324
  807. package/ccw/src/commands/list.js +0 -37
  808. package/ccw/src/commands/serve.js +0 -67
  809. package/ccw/src/commands/stop.js +0 -101
  810. package/ccw/src/commands/tool.js +0 -138
  811. package/ccw/src/commands/uninstall.js +0 -238
  812. package/ccw/src/commands/upgrade.js +0 -307
  813. package/ccw/src/commands/view.js +0 -105
  814. package/ccw/src/core/dashboard-generator-patch.js +0 -29
  815. package/ccw/src/core/dashboard-generator.js +0 -682
  816. package/ccw/src/core/data-aggregator.js +0 -409
  817. package/ccw/src/core/lite-scanner.js +0 -314
  818. package/ccw/src/core/manifest.js +0 -201
  819. package/ccw/src/core/server.js +0 -2063
  820. package/ccw/src/core/session-scanner.js +0 -235
  821. package/ccw/src/index.js +0 -9
  822. package/ccw/src/templates/dashboard-css/01-base.css +0 -291
  823. package/ccw/src/templates/dashboard-css/02-session.css +0 -726
  824. package/ccw/src/templates/dashboard-css/04-lite-tasks.css +0 -843
  825. package/ccw/src/templates/dashboard-css/06-cards.css +0 -1570
  826. package/ccw/src/templates/dashboard-css/07-managers.css +0 -936
  827. package/ccw/src/templates/dashboard-css/09-explorer.css +0 -1397
  828. package/ccw/src/templates/dashboard-js/api.js +0 -200
  829. package/ccw/src/templates/dashboard-js/components/carousel.js +0 -398
  830. package/ccw/src/templates/dashboard-js/components/global-notifications.js +0 -219
  831. package/ccw/src/templates/dashboard-js/components/hook-manager.js +0 -283
  832. package/ccw/src/templates/dashboard-js/components/mcp-manager.js +0 -528
  833. package/ccw/src/templates/dashboard-js/components/modals.js +0 -260
  834. package/ccw/src/templates/dashboard-js/components/navigation.js +0 -245
  835. package/ccw/src/templates/dashboard-js/components/notifications.js +0 -194
  836. package/ccw/src/templates/dashboard-js/components/tabs-other.js +0 -273
  837. package/ccw/src/templates/dashboard-js/main.js +0 -72
  838. package/ccw/src/templates/dashboard-js/state.js +0 -42
  839. package/ccw/src/templates/dashboard-js/utils.js +0 -153
  840. package/ccw/src/templates/dashboard-js/views/explorer.js +0 -852
  841. package/ccw/src/templates/dashboard-js/views/home.js +0 -197
  842. package/ccw/src/templates/dashboard-js/views/hook-manager.js +0 -392
  843. package/ccw/src/templates/dashboard-js/views/lite-tasks.js +0 -395
  844. package/ccw/src/templates/dashboard-js/views/mcp-manager.js +0 -411
  845. package/ccw/src/templates/dashboard-js/views/session-detail.js +0 -780
  846. package/ccw/src/templates/dashboard.html +0 -731
  847. package/ccw/src/tools/classify-folders.js +0 -204
  848. package/ccw/src/tools/convert-tokens-to-css.js +0 -250
  849. package/ccw/src/tools/detect-changed-modules.js +0 -288
  850. package/ccw/src/tools/discover-design-files.js +0 -134
  851. package/ccw/src/tools/edit-file.js +0 -266
  852. package/ccw/src/tools/generate-module-docs.js +0 -416
  853. package/ccw/src/tools/get-modules-by-depth.js +0 -308
  854. package/ccw/src/tools/index.js +0 -176
  855. package/ccw/src/utils/browser-launcher.js +0 -60
  856. package/ccw/src/utils/file-utils.js +0 -48
  857. package/ccw/src/utils/path-resolver.js +0 -279
  858. package/ccw/src/utils/ui.js +0 -148
  859. /package/{ccw/src → src}/templates/dashboard-css/03-tasks.css +0 -0
  860. /package/{ccw/src → src}/templates/dashboard-css/05-context.css +0 -0
  861. /package/{ccw/src → src}/templates/dashboard-css/08-review.css +0 -0
  862. /package/{ccw/src → src}/templates/dashboard-js/components/_conflict_tab.js +0 -0
  863. /package/{ccw/src → src}/templates/dashboard-js/components/_exp_helpers.js +0 -0
  864. /package/{ccw/src → src}/templates/dashboard-js/components/_review_tab.js +0 -0
  865. /package/{ccw/src → src}/templates/dashboard-js/components/flowchart.js +0 -0
  866. /package/{ccw/src → src}/templates/dashboard-js/components/sidebar.js +0 -0
  867. /package/{ccw/src → src}/templates/dashboard-js/components/tabs-context.js +0 -0
  868. /package/{ccw/src → src}/templates/dashboard-js/components/task-drawer-core.js +0 -0
  869. /package/{ccw/src → src}/templates/dashboard-js/components/task-drawer-renderers.js +0 -0
  870. /package/{ccw/src → src}/templates/dashboard-js/components/theme.js +0 -0
  871. /package/{ccw/src → src}/templates/dashboard-js/components/version-check.js +0 -0
  872. /package/{ccw/src → src}/templates/dashboard-js/views/fix-session.js +0 -0
  873. /package/{ccw/src → src}/templates/dashboard-js/views/project-overview.js +0 -0
  874. /package/{ccw/src → src}/templates/dashboard-js/views/review-session.js +0 -0
  875. /package/{ccw/src → src}/templates/review-cycle-dashboard.html +0 -0
  876. /package/{ccw/src → src}/templates/workflow-dashboard.html +0 -0
  877. /package/{ccw/src → src}/tools/ui-generate-preview.js +0 -0
  878. /package/{ccw/src → src}/tools/ui-instantiate-prototypes.js +0 -0
  879. /package/{ccw/src → src}/tools/update-module-claude.js +0 -0
@@ -0,0 +1,1729 @@
1
+ // MCP Manager View
2
+ // Renders the MCP server management interface
3
+
4
+ // CCW Tools available for MCP
5
+ const CCW_MCP_TOOLS = [
6
+ // Core tools (always recommended)
7
+ { name: 'write_file', desc: 'Write/create files', core: true },
8
+ { name: 'edit_file', desc: 'Edit/replace content', core: true },
9
+ { name: 'codex_lens', desc: 'Code index & search', core: true },
10
+ { name: 'smart_search', desc: 'Quick regex/NL search', core: true },
11
+ // Optional tools
12
+ { name: 'session_manager', desc: 'Workflow sessions', core: false },
13
+ { name: 'generate_module_docs', desc: 'Generate docs', core: false },
14
+ { name: 'update_module_claude', desc: 'Update CLAUDE.md', core: false },
15
+ { name: 'cli_executor', desc: 'Gemini/Qwen/Codex CLI', core: false },
16
+ ];
17
+
18
+ // Get currently enabled tools from installed config (Claude)
19
+ function getCcwEnabledTools() {
20
+ const currentPath = projectPath; // Keep original format (forward slash)
21
+ const projectData = mcpAllProjects[currentPath] || {};
22
+ const ccwConfig = projectData.mcpServers?.['ccw-tools'];
23
+ if (ccwConfig?.env?.CCW_ENABLED_TOOLS) {
24
+ const val = ccwConfig.env.CCW_ENABLED_TOOLS;
25
+ if (val.toLowerCase() === 'all') return CCW_MCP_TOOLS.map(t => t.name);
26
+ return val.split(',').map(t => t.trim());
27
+ }
28
+ return CCW_MCP_TOOLS.filter(t => t.core).map(t => t.name);
29
+ }
30
+
31
+ // Get currently enabled tools from Codex config
32
+ function getCcwEnabledToolsCodex() {
33
+ const ccwConfig = codexMcpServers?.['ccw-tools'];
34
+ if (ccwConfig?.env?.CCW_ENABLED_TOOLS) {
35
+ const val = ccwConfig.env.CCW_ENABLED_TOOLS;
36
+ if (val.toLowerCase() === 'all') return CCW_MCP_TOOLS.map(t => t.name);
37
+ return val.split(',').map(t => t.trim());
38
+ }
39
+ // Default to core tools if not installed
40
+ return CCW_MCP_TOOLS.filter(t => t.core).map(t => t.name);
41
+ }
42
+
43
+ async function renderMcpManager() {
44
+ const container = document.getElementById('mainContent');
45
+ if (!container) return;
46
+
47
+ // Hide stats grid and search for MCP view
48
+ const statsGrid = document.getElementById('statsGrid');
49
+ const searchInput = document.getElementById('searchInput');
50
+ if (statsGrid) statsGrid.style.display = 'none';
51
+ if (searchInput) searchInput.parentElement.style.display = 'none';
52
+
53
+ // Load MCP config if not already loaded
54
+ if (!mcpConfig) {
55
+ await loadMcpConfig();
56
+ }
57
+
58
+ // Load MCP templates
59
+ await loadMcpTemplates();
60
+
61
+ const currentPath = projectPath; // Keep original format (forward slash)
62
+ const projectData = mcpAllProjects[currentPath] || {};
63
+ const projectServers = projectData.mcpServers || {};
64
+ const disabledServers = projectData.disabledMcpServers || [];
65
+ const hasMcpJson = projectData.hasMcpJson || false;
66
+ const mcpJsonPath = projectData.mcpJsonPath || null;
67
+
68
+ // Get all available servers from all projects
69
+ const allAvailableServers = getAllAvailableMcpServers();
70
+
71
+ // Separate servers by category:
72
+ // 1. Project Available = Global + Project-specific (servers available to current project)
73
+ // 2. Global Management = Global servers that can be managed
74
+ // 3. Other Projects = Servers from other projects (can install to project or global)
75
+
76
+ const currentProjectServerNames = Object.keys(projectServers);
77
+ const globalServerNames = Object.keys(mcpUserServers || {});
78
+ const enterpriseServerNames = Object.keys(mcpEnterpriseServers || {});
79
+
80
+ // Project Available MCP: servers available to current project
81
+ // This includes: Enterprise (highest priority) + Global + Project-specific
82
+ const projectAvailableEntries = [];
83
+
84
+ // Add enterprise servers first (highest priority)
85
+ for (const [name, config] of Object.entries(mcpEnterpriseServers || {})) {
86
+ projectAvailableEntries.push({
87
+ name,
88
+ config,
89
+ source: 'enterprise',
90
+ canRemove: false,
91
+ canToggle: false
92
+ });
93
+ }
94
+
95
+ // Add global servers
96
+ for (const [name, config] of Object.entries(mcpUserServers || {})) {
97
+ if (!enterpriseServerNames.includes(name)) {
98
+ projectAvailableEntries.push({
99
+ name,
100
+ config,
101
+ source: 'global',
102
+ canRemove: false, // Can't remove from project view, must go to global management
103
+ canToggle: true,
104
+ isEnabled: !disabledServers.includes(name)
105
+ });
106
+ }
107
+ }
108
+
109
+ // Add project-specific servers
110
+ for (const [name, config] of Object.entries(projectServers)) {
111
+ if (!enterpriseServerNames.includes(name) && !globalServerNames.includes(name)) {
112
+ projectAvailableEntries.push({
113
+ name,
114
+ config,
115
+ source: 'project',
116
+ canRemove: true,
117
+ canToggle: true,
118
+ isEnabled: !disabledServers.includes(name)
119
+ });
120
+ }
121
+ }
122
+
123
+ // Global Management: user global servers (for management)
124
+ const globalManagementEntries = Object.entries(mcpUserServers || {});
125
+
126
+ // Enterprise servers (for display only, read-only)
127
+ const enterpriseServerEntries = Object.entries(mcpEnterpriseServers || {});
128
+
129
+ // Other Projects: servers from other projects (not in current project, not global)
130
+ const otherProjectServers = Object.entries(allAvailableServers)
131
+ .filter(([name, info]) => !currentProjectServerNames.includes(name) && !info.isGlobal);
132
+ // Check if CCW Tools is already installed
133
+ const isCcwToolsInstalled = currentProjectServerNames.includes("ccw-tools");
134
+ const enabledTools = getCcwEnabledTools();
135
+ const enabledToolsCodex = getCcwEnabledToolsCodex();
136
+
137
+ // Prepare Codex servers data
138
+ const codexServerEntries = Object.entries(codexMcpServers || {});
139
+ const codexConfigExists = codexMcpConfig?.exists || false;
140
+ const codexConfigPath = codexMcpConfig?.configPath || '~/.codex/config.toml';
141
+
142
+ container.innerHTML = `
143
+ <div class="mcp-manager">
144
+ <!-- CLI Mode Toggle -->
145
+ <div class="mcp-cli-toggle mb-6">
146
+ <div class="flex items-center justify-between bg-card border border-border rounded-lg p-4">
147
+ <div class="flex items-center gap-3">
148
+ <span class="text-sm font-medium text-foreground">${t('mcp.cliMode')}</span>
149
+ <div class="flex items-center bg-muted rounded-lg p-1">
150
+ <button class="cli-mode-btn px-4 py-2 text-sm font-medium rounded-md transition-all ${currentCliMode === 'claude' ? 'bg-primary text-primary-foreground shadow-sm' : 'text-muted-foreground hover:text-foreground'}"
151
+ onclick="setCliMode('claude')">
152
+ <i data-lucide="bot" class="w-4 h-4 inline mr-1.5"></i>
153
+ Claude
154
+ </button>
155
+ <button class="cli-mode-btn px-4 py-2 text-sm font-medium rounded-md transition-all ${currentCliMode === 'codex' ? 'shadow-sm' : 'text-muted-foreground hover:text-foreground'}"
156
+ onclick="setCliMode('codex')"
157
+ style="${currentCliMode === 'codex' ? 'background-color: #f97316; color: white;' : ''}">
158
+ <i data-lucide="code-2" class="w-4 h-4 inline mr-1.5"></i>
159
+ Codex
160
+ </button>
161
+ </div>
162
+ </div>
163
+ <div class="text-xs text-muted-foreground">
164
+ ${currentCliMode === 'claude'
165
+ ? `<span class="flex items-center gap-1"><i data-lucide="file-json" class="w-3 h-3"></i> ~/.claude.json</span>`
166
+ : `<span class="flex items-center gap-1"><i data-lucide="file-code" class="w-3 h-3"></i> ${codexConfigPath}</span>`
167
+ }
168
+ </div>
169
+ </div>
170
+ </div>
171
+
172
+ ${currentCliMode === 'codex' ? `
173
+ <!-- CCW Tools MCP Server Card (Codex mode) -->
174
+ <div class="mcp-section mb-6">
175
+ <div class="ccw-tools-card bg-gradient-to-br from-orange-500/10 to-orange-500/5 border-2 ${codexMcpServers && codexMcpServers['ccw-tools'] ? 'border-success' : 'border-orange-500/30'} rounded-lg p-6 hover:shadow-lg transition-all">
176
+ <div class="flex items-start justify-between gap-4">
177
+ <div class="flex items-start gap-4 flex-1">
178
+ <div class="shrink-0 w-12 h-12 bg-orange-500 rounded-lg flex items-center justify-center">
179
+ <i data-lucide="wrench" class="w-6 h-6 text-white"></i>
180
+ </div>
181
+ <div class="flex-1 min-w-0">
182
+ <div class="flex items-center gap-2 mb-2">
183
+ <h3 class="text-lg font-bold text-foreground">CCW Tools MCP</h3>
184
+ <span class="text-xs px-2 py-0.5 bg-orange-100 text-orange-700 dark:bg-orange-900/30 dark:text-orange-300 rounded-full">Codex</span>
185
+ ${codexMcpServers && codexMcpServers['ccw-tools'] ? `
186
+ <span class="inline-flex items-center gap-1 px-2 py-0.5 text-xs font-semibold rounded-full bg-success-light text-success">
187
+ <i data-lucide="check" class="w-3 h-3"></i>
188
+ ${enabledToolsCodex.length} tools
189
+ </span>
190
+ ` : `
191
+ <span class="inline-flex items-center gap-1 px-2 py-0.5 text-xs font-semibold rounded-full bg-orange-500/20 text-orange-600 dark:text-orange-400">
192
+ <i data-lucide="package" class="w-3 h-3"></i>
193
+ ${t('mcp.available')}
194
+ </span>
195
+ `}
196
+ </div>
197
+ <p class="text-sm text-muted-foreground mb-3">${t('mcp.ccwToolsDesc')}</p>
198
+ <!-- Tool Selection Grid for Codex -->
199
+ <div class="grid grid-cols-3 sm:grid-cols-5 gap-2 mb-3">
200
+ ${CCW_MCP_TOOLS.map(tool => `
201
+ <label class="flex items-center gap-1.5 text-xs cursor-pointer hover:bg-muted/50 rounded px-1.5 py-1 transition-colors">
202
+ <input type="checkbox" class="ccw-tool-checkbox-codex w-3 h-3"
203
+ data-tool="${tool.name}"
204
+ ${enabledToolsCodex.includes(tool.name) ? 'checked' : ''}>
205
+ <span class="${tool.core ? 'font-medium' : 'text-muted-foreground'}">${tool.desc}</span>
206
+ </label>
207
+ `).join('')}
208
+ </div>
209
+ <div class="flex items-center gap-3 text-xs">
210
+ <button class="text-orange-500 hover:underline" onclick="selectCcwToolsCodex('core')">Core only</button>
211
+ <button class="text-orange-500 hover:underline" onclick="selectCcwToolsCodex('all')">All</button>
212
+ <button class="text-muted-foreground hover:underline" onclick="selectCcwToolsCodex('none')">None</button>
213
+ </div>
214
+ </div>
215
+ </div>
216
+ <div class="shrink-0">
217
+ <button class="px-4 py-2 text-sm bg-orange-500 text-white rounded-lg hover:opacity-90 transition-opacity flex items-center gap-1"
218
+ onclick="installCcwToolsMcpToCodex()">
219
+ <i data-lucide="download" class="w-4 h-4"></i>
220
+ ${codexMcpServers && codexMcpServers['ccw-tools'] ? t('mcp.update') : t('mcp.install')}
221
+ </button>
222
+ </div>
223
+ </div>
224
+ </div>
225
+ </div>
226
+
227
+ <!-- Codex MCP Servers Section -->
228
+ <div class="mcp-section mb-6">
229
+ <div class="flex items-center justify-between mb-4">
230
+ <div class="flex items-center gap-3">
231
+ <div class="flex items-center gap-2">
232
+ <i data-lucide="code-2" class="w-5 h-5 text-orange-500"></i>
233
+ <h3 class="text-lg font-semibold text-foreground">${t('mcp.codex.globalServers')}</h3>
234
+ </div>
235
+ <button class="px-3 py-1.5 text-sm bg-orange-500 text-white rounded-lg hover:opacity-90 transition-opacity flex items-center gap-1"
236
+ onclick="openCodexMcpCreateModal()">
237
+ <span>+</span> ${t('mcp.codex.newServer')}
238
+ </button>
239
+ ${codexConfigExists ? `
240
+ <span class="inline-flex items-center gap-1.5 px-2 py-1 text-xs bg-success/10 text-success rounded-md border border-success/20">
241
+ <i data-lucide="file-check" class="w-3.5 h-3.5"></i>
242
+ config.toml
243
+ </span>
244
+ ` : `
245
+ <span class="inline-flex items-center gap-1.5 px-2 py-1 text-xs bg-muted text-muted-foreground rounded-md border border-border" title="Will create ~/.codex/config.toml">
246
+ <i data-lucide="file-plus" class="w-3.5 h-3.5"></i>
247
+ Will create config.toml
248
+ </span>
249
+ `}
250
+ </div>
251
+ <span class="text-sm text-muted-foreground">${codexServerEntries.length} ${t('mcp.serversAvailable')}</span>
252
+ </div>
253
+
254
+ <!-- Info about Codex MCP -->
255
+ <div class="bg-orange-50 dark:bg-orange-950/30 border border-orange-200 dark:border-orange-800 rounded-lg p-4 mb-4">
256
+ <div class="flex items-start gap-3">
257
+ <i data-lucide="info" class="w-5 h-5 text-orange-500 shrink-0 mt-0.5"></i>
258
+ <div class="text-sm">
259
+ <p class="text-orange-800 dark:text-orange-200 font-medium mb-1">${t('mcp.codex.infoTitle')}</p>
260
+ <p class="text-orange-700 dark:text-orange-300 text-xs">${t('mcp.codex.infoDesc')}</p>
261
+ </div>
262
+ </div>
263
+ </div>
264
+
265
+ ${codexServerEntries.length === 0 ? `
266
+ <div class="mcp-empty-state bg-card border border-border rounded-lg p-6 text-center">
267
+ <div class="text-muted-foreground mb-3"><i data-lucide="plug" class="w-10 h-10 mx-auto"></i></div>
268
+ <p class="text-muted-foreground">${t('mcp.codex.noServers')}</p>
269
+ <p class="text-sm text-muted-foreground mt-1">${t('mcp.codex.noServersHint')}</p>
270
+ </div>
271
+ ` : `
272
+ <div class="mcp-server-grid grid gap-3">
273
+ ${codexServerEntries.map(([serverName, serverConfig]) => {
274
+ return renderCodexServerCard(serverName, serverConfig);
275
+ }).join('')}
276
+ </div>
277
+ `}
278
+ </div>
279
+
280
+ <!-- Copy Claude Servers to Codex -->
281
+ ${Object.keys(mcpUserServers || {}).length > 0 ? `
282
+ <div class="mcp-section mb-6">
283
+ <div class="flex items-center justify-between mb-4">
284
+ <h3 class="text-lg font-semibold text-foreground flex items-center gap-2">
285
+ <i data-lucide="copy" class="w-5 h-5"></i>
286
+ ${t('mcp.codex.copyFromClaude')}
287
+ </h3>
288
+ <span class="text-sm text-muted-foreground">${Object.keys(mcpUserServers || {}).length} ${t('mcp.serversAvailable')}</span>
289
+ </div>
290
+ <div class="mcp-server-grid grid gap-3">
291
+ ${Object.entries(mcpUserServers || {}).map(([serverName, serverConfig]) => {
292
+ const alreadyInCodex = codexMcpServers && codexMcpServers[serverName];
293
+ return `
294
+ <div class="mcp-server-card bg-card border ${alreadyInCodex ? 'border-success/50' : 'border-border'} border-dashed rounded-lg p-4 hover:shadow-md transition-all">
295
+ <div class="flex items-start justify-between mb-3">
296
+ <div class="flex items-center gap-2">
297
+ <i data-lucide="bot" class="w-5 h-5 text-primary"></i>
298
+ <h4 class="font-semibold text-foreground">${escapeHtml(serverName)}</h4>
299
+ <span class="text-xs px-2 py-0.5 bg-primary/10 text-primary rounded-full">Claude</span>
300
+ ${alreadyInCodex ? `<span class="text-xs px-2 py-0.5 bg-success/10 text-success rounded-full">${t('mcp.codex.alreadyAdded')}</span>` : ''}
301
+ </div>
302
+ ${!alreadyInCodex ? `
303
+ <button class="px-3 py-1 text-xs bg-orange-500 text-white rounded hover:opacity-90 transition-opacity"
304
+ onclick="copyClaudeServerToCodex('${escapeHtml(serverName)}', ${JSON.stringify(serverConfig).replace(/'/g, "&#39;")})"
305
+ title="${t('mcp.codex.copyToCodex')}">
306
+ <i data-lucide="arrow-right" class="w-3.5 h-3.5 inline"></i> Codex
307
+ </button>
308
+ ` : ''}
309
+ </div>
310
+ <div class="mcp-server-details text-sm space-y-1">
311
+ <div class="flex items-center gap-2 text-muted-foreground">
312
+ <span class="font-mono text-xs bg-muted px-1.5 py-0.5 rounded">${t('mcp.cmd')}</span>
313
+ <span class="truncate" title="${escapeHtml(serverConfig.command || 'N/A')}">${escapeHtml(serverConfig.command || 'N/A')}</span>
314
+ </div>
315
+ </div>
316
+ </div>
317
+ `;
318
+ }).join('')}
319
+ </div>
320
+ </div>
321
+ ` : ''}
322
+
323
+ <!-- Available MCP Servers from Other Projects (Codex mode) -->
324
+ <div class="mcp-section">
325
+ <div class="flex items-center justify-between mb-4">
326
+ <h3 class="text-lg font-semibold text-foreground">${t('mcp.availableOther')}</h3>
327
+ <span class="text-sm text-muted-foreground">${otherProjectServers.length} ${t('mcp.serversAvailable')}</span>
328
+ </div>
329
+
330
+ ${otherProjectServers.length === 0 ? `
331
+ <div class="mcp-empty-state bg-card border border-border rounded-lg p-6 text-center">
332
+ <p class="text-muted-foreground">${t('empty.noAdditionalMcp')}</p>
333
+ </div>
334
+ ` : `
335
+ <div class="mcp-server-grid grid gap-3">
336
+ ${otherProjectServers.map(([serverName, serverInfo]) => {
337
+ return renderAvailableServerCardForCodex(serverName, serverInfo);
338
+ }).join('')}
339
+ </div>
340
+ `}
341
+ </div>
342
+ ` : `
343
+ <!-- CCW Tools MCP Server Card -->
344
+ <div class="mcp-section mb-6">
345
+ <div class="ccw-tools-card bg-gradient-to-br from-orange-500/10 to-orange-500/5 border-2 ${isCcwToolsInstalled ? 'border-success' : 'border-orange-500/30'} rounded-lg p-6 hover:shadow-lg transition-all">
346
+ <div class="flex items-start justify-between gap-4">
347
+ <div class="flex items-start gap-4 flex-1">
348
+ <div class="shrink-0 w-12 h-12 bg-orange-500 rounded-lg flex items-center justify-center">
349
+ <i data-lucide="wrench" class="w-6 h-6 text-white"></i>
350
+ </div>
351
+ <div class="flex-1 min-w-0">
352
+ <div class="flex items-center gap-2 mb-2">
353
+ <h3 class="text-lg font-bold text-foreground">CCW Tools MCP</h3>
354
+ ${isCcwToolsInstalled ? `
355
+ <span class="inline-flex items-center gap-1 px-2 py-0.5 text-xs font-semibold rounded-full bg-success-light text-success">
356
+ <i data-lucide="check" class="w-3 h-3"></i>
357
+ ${enabledTools.length} tools
358
+ </span>
359
+ ` : `
360
+ <span class="inline-flex items-center gap-1 px-2 py-0.5 text-xs font-semibold rounded-full bg-orange-500/20 text-orange-600 dark:text-orange-400">
361
+ <i data-lucide="package" class="w-3 h-3"></i>
362
+ Available
363
+ </span>
364
+ `}
365
+ </div>
366
+ <!-- Tool Selection Grid -->
367
+ <div class="grid grid-cols-3 sm:grid-cols-5 gap-2 mb-3">
368
+ ${CCW_MCP_TOOLS.map(tool => `
369
+ <label class="flex items-center gap-1.5 text-xs cursor-pointer hover:bg-muted/50 rounded px-1.5 py-1 transition-colors">
370
+ <input type="checkbox" class="ccw-tool-checkbox w-3 h-3"
371
+ data-tool="${tool.name}"
372
+ ${enabledTools.includes(tool.name) ? 'checked' : ''}>
373
+ <span class="${tool.core ? 'font-medium' : 'text-muted-foreground'}">${tool.desc}</span>
374
+ </label>
375
+ `).join('')}
376
+ </div>
377
+ <div class="flex items-center gap-3 text-xs">
378
+ <button class="text-orange-500 hover:underline" onclick="selectCcwTools('core')">Core only</button>
379
+ <button class="text-orange-500 hover:underline" onclick="selectCcwTools('all')">All</button>
380
+ <button class="text-muted-foreground hover:underline" onclick="selectCcwTools('none')">None</button>
381
+ </div>
382
+ </div>
383
+ </div>
384
+ <div class="shrink-0 flex gap-2">
385
+ ${isCcwToolsInstalled ? `
386
+ <button class="px-4 py-2 text-sm bg-orange-500 text-white rounded-lg hover:opacity-90 transition-opacity flex items-center gap-1"
387
+ onclick="updateCcwToolsMcp('workspace')"
388
+ title="${t('mcp.updateInWorkspace')}">
389
+ <i data-lucide="folder" class="w-4 h-4"></i>
390
+ ${t('mcp.updateInWorkspace')}
391
+ </button>
392
+ <button class="px-4 py-2 text-sm bg-success text-success-foreground rounded-lg hover:opacity-90 transition-opacity flex items-center gap-1"
393
+ onclick="updateCcwToolsMcp('global')"
394
+ title="${t('mcp.updateInGlobal')}">
395
+ <i data-lucide="globe" class="w-4 h-4"></i>
396
+ ${t('mcp.updateInGlobal')}
397
+ </button>
398
+ ` : `
399
+ <button class="px-4 py-2 text-sm bg-orange-500 text-white rounded-lg hover:opacity-90 transition-opacity flex items-center gap-1"
400
+ onclick="installCcwToolsMcp('workspace')"
401
+ title="${t('mcp.installToWorkspace')}">
402
+ <i data-lucide="folder" class="w-4 h-4"></i>
403
+ ${t('mcp.installToWorkspace')}
404
+ </button>
405
+ <button class="px-4 py-2 text-sm bg-success text-success-foreground rounded-lg hover:opacity-90 transition-opacity flex items-center gap-1"
406
+ onclick="installCcwToolsMcp('global')"
407
+ title="${t('mcp.installToGlobal')}">
408
+ <i data-lucide="globe" class="w-4 h-4"></i>
409
+ ${t('mcp.installToGlobal')}
410
+ </button>
411
+ `}
412
+ </div>
413
+ </div>
414
+ </div>
415
+ </div>
416
+
417
+ <!-- Project Available MCP Servers -->
418
+ <div class="mcp-section mb-6">
419
+ <div class="flex items-center justify-between mb-4">
420
+ <div class="flex items-center gap-3">
421
+ <h3 class="text-lg font-semibold text-foreground">${t('mcp.projectAvailable')}</h3>
422
+ <button class="px-3 py-1.5 text-sm bg-primary text-primary-foreground rounded-lg hover:opacity-90 transition-opacity flex items-center gap-1"
423
+ onclick="openMcpCreateModal('project')">
424
+ <span>+</span> ${t('mcp.newProjectServer')}
425
+ </button>
426
+ ${hasMcpJson ? `
427
+ <span class="inline-flex items-center gap-1.5 px-2 py-1 text-xs bg-success/10 text-success rounded-md border border-success/20">
428
+ <i data-lucide="file-check" class="w-3.5 h-3.5"></i>
429
+ .mcp.json
430
+ </span>
431
+ ` : `
432
+ <span class="inline-flex items-center gap-1.5 px-2 py-1 text-xs bg-muted text-muted-foreground rounded-md border border-border" title="New servers will create .mcp.json">
433
+ <i data-lucide="file-plus" class="w-3.5 h-3.5"></i>
434
+ Will use .mcp.json
435
+ </span>
436
+ `}
437
+ </div>
438
+ <span class="text-sm text-muted-foreground">${projectAvailableEntries.length} ${t('mcp.serversAvailable')}</span>
439
+ </div>
440
+
441
+ ${projectAvailableEntries.length === 0 ? `
442
+ <div class="mcp-empty-state bg-card border border-border rounded-lg p-6 text-center">
443
+ <div class="text-muted-foreground mb-3"><i data-lucide="plug" class="w-10 h-10 mx-auto"></i></div>
444
+ <p class="text-muted-foreground">${t('empty.noMcpServers')}</p>
445
+ <p class="text-sm text-muted-foreground mt-1">${t('empty.addMcpServersHint')}</p>
446
+ </div>
447
+ ` : `
448
+ <div class="mcp-server-grid grid gap-3">
449
+ ${projectAvailableEntries.map(entry => {
450
+ return renderProjectAvailableServerCard(entry);
451
+ }).join('')}
452
+ </div>
453
+ `}
454
+ </div>
455
+
456
+ <!-- Global Available MCP Servers -->
457
+ <div class="mcp-section mb-6">
458
+ <div class="flex items-center justify-between mb-4">
459
+ <div class="flex items-center gap-3">
460
+ <div class="flex items-center gap-2">
461
+ <i data-lucide="globe" class="w-5 h-5 text-success"></i>
462
+ <h3 class="text-lg font-semibold text-foreground">${t('mcp.globalAvailable')}</h3>
463
+ </div>
464
+ <button class="px-3 py-1.5 text-sm bg-success text-success-foreground rounded-lg hover:opacity-90 transition-opacity flex items-center gap-1"
465
+ onclick="openMcpCreateModal('global')">
466
+ <span>+</span> ${t('mcp.newGlobalServer')}
467
+ </button>
468
+ </div>
469
+ <span class="text-sm text-muted-foreground">${globalManagementEntries.length} ${t('mcp.globalServersFrom')}</span>
470
+ </div>
471
+
472
+ ${globalManagementEntries.length === 0 ? `
473
+ <div class="mcp-empty-state bg-card border border-border rounded-lg p-6 text-center">
474
+ <div class="text-muted-foreground mb-3"><i data-lucide="globe" class="w-10 h-10 mx-auto"></i></div>
475
+ <p class="text-muted-foreground">${t('empty.noGlobalMcpServers')}</p>
476
+ <p class="text-sm text-muted-foreground mt-1">${t('empty.globalServersHint')}</p>
477
+ </div>
478
+ ` : `
479
+ <div class="mcp-server-grid grid gap-3">
480
+ ${globalManagementEntries.map(([serverName, serverConfig]) => {
481
+ return renderGlobalManagementCard(serverName, serverConfig);
482
+ }).join('')}
483
+ </div>
484
+ `}
485
+ </div>
486
+
487
+ <!-- Available MCP Servers from Other Projects -->
488
+ <div class="mcp-section">
489
+ <div class="flex items-center justify-between mb-4">
490
+ <h3 class="text-lg font-semibold text-foreground">${t('mcp.availableOther')}</h3>
491
+ <span class="text-sm text-muted-foreground">${otherProjectServers.length} ${t('mcp.serversAvailable')}</span>
492
+ </div>
493
+
494
+ ${otherProjectServers.length === 0 ? `
495
+ <div class="mcp-empty-state bg-card border border-border rounded-lg p-6 text-center">
496
+ <p class="text-muted-foreground">${t('empty.noAdditionalMcp')}</p>
497
+ </div>
498
+ ` : `
499
+ <div class="mcp-server-grid grid gap-3">
500
+ ${otherProjectServers.map(([serverName, serverInfo]) => {
501
+ return renderAvailableServerCard(serverName, serverInfo);
502
+ }).join('')}
503
+ </div>
504
+ `}
505
+ </div>
506
+
507
+ <!-- MCP Templates Section -->
508
+ ${mcpTemplates.length > 0 ? `
509
+ <div class="mcp-section mt-6">
510
+ <div class="flex items-center justify-between mb-4">
511
+ <h3 class="text-lg font-semibold text-foreground flex items-center gap-2">
512
+ <i data-lucide="layout-template" class="w-5 h-5"></i>
513
+ ${t('mcp.templates')}
514
+ </h3>
515
+ <span class="text-sm text-muted-foreground">${mcpTemplates.length} ${t('mcp.savedTemplates')}</span>
516
+ </div>
517
+
518
+ <div class="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-4">
519
+ ${mcpTemplates.map(template => `
520
+ <div class="mcp-template-card bg-card border border-border rounded-lg p-4 hover:shadow-md transition-all">
521
+ <div class="flex items-start justify-between mb-3">
522
+ <div class="flex-1 min-w-0">
523
+ <h4 class="font-semibold text-foreground truncate flex items-center gap-2">
524
+ <i data-lucide="layout-template" class="w-4 h-4 shrink-0"></i>
525
+ <span class="truncate">${escapeHtml(template.name)}</span>
526
+ </h4>
527
+ ${template.description ? `
528
+ <p class="text-xs text-muted-foreground mt-1 line-clamp-2">${escapeHtml(template.description)}</p>
529
+ ` : ''}
530
+ </div>
531
+ </div>
532
+
533
+ <div class="mcp-server-details text-sm space-y-1 mb-3">
534
+ <div class="flex items-center gap-2 text-muted-foreground">
535
+ <span class="font-mono text-xs bg-muted px-1.5 py-0.5 rounded">${t('mcp.cmd')}</span>
536
+ <span class="truncate text-xs" title="${escapeHtml(template.serverConfig.command)}">${escapeHtml(template.serverConfig.command)}</span>
537
+ </div>
538
+ ${template.serverConfig.args && template.serverConfig.args.length > 0 ? `
539
+ <div class="flex items-start gap-2 text-muted-foreground">
540
+ <span class="font-mono text-xs bg-muted px-1.5 py-0.5 rounded shrink-0">${t('mcp.args')}</span>
541
+ <span class="text-xs font-mono truncate" title="${escapeHtml(template.serverConfig.args.join(' '))}">${escapeHtml(template.serverConfig.args.slice(0, 2).join(' '))}${template.serverConfig.args.length > 2 ? '...' : ''}</span>
542
+ </div>
543
+ ` : ''}
544
+ </div>
545
+
546
+ <div class="mt-3 pt-3 border-t border-border flex items-center justify-between gap-2">
547
+ <div class="flex items-center gap-2">
548
+ <button class="text-xs text-primary hover:text-primary/80 transition-colors flex items-center gap-1"
549
+ data-template-name="${escapeHtml(template.name)}"
550
+ data-scope="project"
551
+ data-action="install-template"
552
+ title="${t('mcp.installToProject')}">
553
+ <i data-lucide="download" class="w-3 h-3"></i>
554
+ ${t('mcp.toProject')}
555
+ </button>
556
+ <button class="text-xs text-success hover:text-success/80 transition-colors flex items-center gap-1"
557
+ data-template-name="${escapeHtml(template.name)}"
558
+ data-scope="global"
559
+ data-action="install-template"
560
+ title="${t('mcp.installToGlobal')}">
561
+ <i data-lucide="globe" class="w-3 h-3"></i>
562
+ ${t('mcp.toGlobal')}
563
+ </button>
564
+ </div>
565
+ <button class="text-xs text-destructive hover:text-destructive/80 transition-colors"
566
+ data-template-name="${escapeHtml(template.name)}"
567
+ data-action="delete-template"
568
+ title="${t('mcp.deleteTemplate')}">
569
+ <i data-lucide="trash-2" class="w-3 h-3"></i>
570
+ </button>
571
+ </div>
572
+ </div>
573
+ `).join('')}
574
+ </div>
575
+ </div>
576
+ ` : ''}
577
+
578
+ <!-- Copy Codex Servers to Claude (Claude mode only) -->
579
+ ${currentCliMode === 'claude' && Object.keys(codexMcpServers || {}).length > 0 ? `
580
+ <div class="mcp-section mb-6">
581
+ <div class="flex items-center justify-between mb-4">
582
+ <h3 class="text-lg font-semibold text-foreground flex items-center gap-2">
583
+ <i data-lucide="copy" class="w-5 h-5"></i>
584
+ ${t('mcp.claude.copyFromCodex')}
585
+ </h3>
586
+ <span class="text-sm text-muted-foreground">${Object.keys(codexMcpServers || {}).length} ${t('mcp.serversAvailable')}</span>
587
+ </div>
588
+ <div class="mcp-server-grid grid gap-3">
589
+ ${Object.entries(codexMcpServers || {}).map(([serverName, serverConfig]) => {
590
+ const alreadyInClaude = mcpUserServers && mcpUserServers[serverName];
591
+ const isStdio = !!serverConfig.command;
592
+ const isHttp = !!serverConfig.url;
593
+ return `
594
+ <div class="mcp-server-card bg-card border ${alreadyInClaude ? 'border-success/50' : 'border-orange-200 dark:border-orange-800'} border-dashed rounded-lg p-4 hover:shadow-md transition-all">
595
+ <div class="flex items-start justify-between mb-3">
596
+ <div class="flex items-center gap-2 flex-wrap">
597
+ <i data-lucide="code-2" class="w-5 h-5 text-orange-500"></i>
598
+ <h4 class="font-semibold text-foreground">${escapeHtml(serverName)}</h4>
599
+ <span class="text-xs px-2 py-0.5 bg-orange-100 text-orange-700 dark:bg-orange-900/30 dark:text-orange-300 rounded-full">Codex</span>
600
+ ${isHttp
601
+ ? '<span class="text-xs px-2 py-0.5 bg-blue-100 text-blue-700 dark:bg-blue-900/30 dark:text-blue-300 rounded-full">HTTP</span>'
602
+ : '<span class="text-xs px-2 py-0.5 bg-green-100 text-green-700 dark:bg-green-900/30 dark:text-green-300 rounded-full">STDIO</span>'
603
+ }
604
+ ${alreadyInClaude ? '<span class="text-xs px-2 py-0.5 bg-success/10 text-success rounded-full">' + t('mcp.claude.alreadyAdded') + '</span>' : ''}
605
+ </div>
606
+ ${!alreadyInClaude ? `
607
+ <button class="px-3 py-1 text-xs bg-primary text-primary-foreground rounded hover:opacity-90 transition-opacity"
608
+ onclick="copyCodexServerToClaude('${escapeHtml(serverName)}', ${JSON.stringify(serverConfig).replace(/'/g, "&#39;")})"
609
+ title="${t('mcp.claude.copyToClaude')}">
610
+ <i data-lucide="arrow-right" class="w-3.5 h-3.5 inline"></i> Claude
611
+ </button>
612
+ ` : ''}
613
+ </div>
614
+ <div class="mcp-server-details text-sm space-y-1">
615
+ <div class="flex items-center gap-2 text-muted-foreground">
616
+ <span class="font-mono text-xs bg-muted px-1.5 py-0.5 rounded">${isHttp ? t('mcp.url') : t('mcp.cmd')}</span>
617
+ <span class="truncate" title="${escapeHtml(serverConfig.command || serverConfig.url || 'N/A')}">${escapeHtml(serverConfig.command || serverConfig.url || 'N/A')}</span>
618
+ </div>
619
+ </div>
620
+ </div>
621
+ `;
622
+ }).join('')}
623
+ </div>
624
+ </div>
625
+ ` : ''}
626
+
627
+ <!-- All Projects MCP Overview Table (Claude mode only) -->
628
+ ${currentCliMode === 'claude' ? `
629
+ <div class="mcp-section mt-6">
630
+ <div class="flex items-center justify-between mb-4">
631
+ <h3 class="text-lg font-semibold text-foreground">${t('mcp.allProjects')}</h3>
632
+ <span class="text-sm text-muted-foreground">${Object.keys(mcpAllProjects).length} ${t('mcp.projects')}</span>
633
+ </div>
634
+
635
+ <div class="mcp-projects-table bg-card border border-border rounded-lg overflow-hidden">
636
+ <table class="w-full">
637
+ <thead class="bg-muted/50">
638
+ <tr>
639
+ <th class="text-left px-4 py-3 text-sm font-semibold text-foreground border-b border-border">${t('mcp.project')}</th>
640
+ <th class="text-left px-4 py-3 text-sm font-semibold text-foreground border-b border-border">${t('mcp.servers')}</th>
641
+ <th class="text-center px-4 py-3 text-sm font-semibold text-foreground border-b border-border w-24">${t('mcp.status')}</th>
642
+ </tr>
643
+ </thead>
644
+ <tbody>
645
+ ${Object.entries(mcpAllProjects).map(([path, config]) => {
646
+ const servers = config.mcpServers || {};
647
+ const projectDisabled = config.disabledMcpServers || [];
648
+ const serverNames = Object.keys(servers);
649
+ const isCurrentProject = path === currentPath;
650
+ const enabledCount = serverNames.filter(s => !projectDisabled.includes(s)).length;
651
+ const projectHasMcpJson = config.hasMcpJson || false;
652
+
653
+ return `
654
+ <tr class="border-b border-border last:border-b-0 ${isCurrentProject ? 'bg-primary/5' : 'hover:bg-hover/50'}">
655
+ <td class="px-4 py-3">
656
+ <div class="flex items-center gap-2 min-w-0">
657
+ <span class="shrink-0">${isCurrentProject ? '<i data-lucide="map-pin" class="w-4 h-4 text-primary"></i>' : '<i data-lucide="folder" class="w-4 h-4"></i>'}</span>
658
+ <div class="min-w-0">
659
+ <div class="font-medium text-foreground truncate text-sm flex items-center gap-2" title="${escapeHtml(path)}">
660
+ <span class="truncate">${escapeHtml(path.split('\\').pop() || path)}</span>
661
+ ${isCurrentProject ? `<span class="text-xs text-primary font-medium shrink-0">${t('mcp.current')}</span>` : ''}
662
+ ${projectHasMcpJson ? `<span class="shrink-0 inline-flex items-center gap-1 px-1.5 py-0.5 text-xs bg-success/10 text-success rounded" title=".mcp.json detected"><i data-lucide="file-check" class="w-3 h-3"></i></span>` : ''}
663
+ </div>
664
+ <div class="text-xs text-muted-foreground truncate">${escapeHtml(path)}</div>
665
+ </div>
666
+ </div>
667
+ </td>
668
+ <td class="px-4 py-3">
669
+ <div class="flex flex-wrap gap-1.5">
670
+ ${serverNames.length === 0
671
+ ? `<span class="text-xs text-muted-foreground italic">${t('mcp.noMcpServers')}</span>`
672
+ : serverNames.map(serverName => {
673
+ const isEnabled = !projectDisabled.includes(serverName);
674
+ return `
675
+ <span class="inline-flex items-center gap-1 px-2 py-0.5 text-xs font-medium rounded-full ${isEnabled ? 'bg-success-light text-success' : 'bg-hover text-muted-foreground'}">
676
+ <span class="w-1.5 h-1.5 rounded-full ${isEnabled ? 'bg-success' : 'bg-muted-foreground'}"></span>
677
+ ${escapeHtml(serverName)}
678
+ </span>
679
+ `;
680
+ }).join('')
681
+ }
682
+ </div>
683
+ </td>
684
+ <td class="px-4 py-3 text-center">
685
+ <span class="inline-flex items-center px-2 py-1 text-xs font-semibold rounded-full ${serverNames.length > 0 ? 'bg-success-light text-success' : 'bg-hover text-muted-foreground'}">
686
+ ${enabledCount}/${serverNames.length}
687
+ </span>
688
+ </td>
689
+ </tr>
690
+ `;
691
+ }).join('')}
692
+ </tbody>
693
+ </table>
694
+ </div>
695
+ </div>
696
+ ` : ''}
697
+ `}
698
+
699
+ <!-- MCP Server Details Modal -->
700
+ <div id="mcpDetailsModal" class="fixed inset-0 bg-black/50 flex items-center justify-center z-50 hidden">
701
+ <div class="bg-card border border-border rounded-lg shadow-xl max-w-2xl w-full mx-4 max-h-[80vh] overflow-hidden flex flex-col">
702
+ <!-- Modal Header -->
703
+ <div class="flex items-center justify-between px-6 py-4 border-b border-border">
704
+ <h2 class="text-lg font-semibold text-foreground">${t('mcp.detailsModal.title')}</h2>
705
+ <button id="mcpDetailsModalClose" class="text-muted-foreground hover:text-foreground transition-colors">
706
+ <i data-lucide="x" class="w-5 h-5"></i>
707
+ </button>
708
+ </div>
709
+
710
+ <!-- Modal Body -->
711
+ <div id="mcpDetailsModalBody" class="px-6 py-4 overflow-y-auto flex-1">
712
+ <!-- Content will be dynamically filled -->
713
+ </div>
714
+ </div>
715
+ </div>
716
+ </div>
717
+ `;
718
+
719
+ // Attach event listeners for toggle switches
720
+ attachMcpEventListeners();
721
+
722
+ // Initialize Lucide icons
723
+ if (typeof lucide !== 'undefined') lucide.createIcons();
724
+ }
725
+
726
+ // Render card for Project Available MCP (current project can use)
727
+ function renderProjectAvailableServerCard(entry) {
728
+ const { name, config, source, canRemove, canToggle, isEnabled } = entry;
729
+ const command = config.command || 'N/A';
730
+ const args = config.args || [];
731
+ const hasEnv = config.env && Object.keys(config.env).length > 0;
732
+
733
+ // Source badge
734
+ let sourceBadge = '';
735
+ if (source === 'enterprise') {
736
+ sourceBadge = `<span class="text-xs px-2 py-0.5 bg-warning/20 text-warning rounded-full">${t('mcp.sourceEnterprise')}</span>`;
737
+ } else if (source === 'global') {
738
+ sourceBadge = `<span class="text-xs px-2 py-0.5 bg-success/10 text-success rounded-full">${t('mcp.sourceGlobal')}</span>`;
739
+ } else if (source === 'project') {
740
+ sourceBadge = `<span class="text-xs px-2 py-0.5 bg-primary/10 text-primary rounded-full">${t('mcp.sourceProject')}</span>`;
741
+ }
742
+
743
+ return `
744
+ <div class="mcp-server-card bg-card border border-border rounded-lg p-4 hover:shadow-md transition-all cursor-pointer ${canToggle && !isEnabled ? 'opacity-60' : ''}"
745
+ data-server-name="${escapeHtml(name)}"
746
+ data-server-config="${escapeHtml(JSON.stringify(config))}"
747
+ data-server-source="${source}"
748
+ data-action="view-details"
749
+ title="${t('mcp.clickToViewDetails')}">
750
+ <div class="flex items-start justify-between mb-3">
751
+ <div class="flex items-center gap-2">
752
+ <span>${canToggle && isEnabled ? '<i data-lucide="check-circle" class="w-5 h-5 text-success"></i>' : '<i data-lucide="circle" class="w-5 h-5 text-muted-foreground"></i>'}</span>
753
+ <h4 class="font-semibold text-foreground">${escapeHtml(name)}</h4>
754
+ ${sourceBadge}
755
+ </div>
756
+ ${canToggle ? `
757
+ <label class="mcp-toggle relative inline-flex items-center cursor-pointer" onclick="event.stopPropagation()">
758
+ <input type="checkbox" class="sr-only peer"
759
+ ${isEnabled ? 'checked' : ''}
760
+ data-server-name="${escapeHtml(name)}"
761
+ data-action="toggle">
762
+ <div class="w-9 h-5 bg-hover peer-focus:outline-none rounded-full peer peer-checked:after:translate-x-full peer-checked:after:border-white after:content-[''] after:absolute after:top-[2px] after:left-[2px] after:bg-white after:border-gray-300 after:border after:rounded-full after:h-4 after:w-4 after:transition-all peer-checked:bg-success"></div>
763
+ </label>
764
+ ` : ''}
765
+ </div>
766
+
767
+ <div class="mcp-server-details text-sm space-y-1">
768
+ <div class="flex items-center gap-2 text-muted-foreground">
769
+ <span class="font-mono text-xs bg-muted px-1.5 py-0.5 rounded">${t('mcp.cmd')}</span>
770
+ <span class="truncate" title="${escapeHtml(command)}">${escapeHtml(command)}</span>
771
+ </div>
772
+ ${args.length > 0 ? `
773
+ <div class="flex items-start gap-2 text-muted-foreground">
774
+ <span class="font-mono text-xs bg-muted px-1.5 py-0.5 rounded shrink-0">${t('mcp.args')}</span>
775
+ <span class="text-xs font-mono truncate" title="${escapeHtml(args.join(' '))}">${escapeHtml(args.slice(0, 3).join(' '))}${args.length > 3 ? '...' : ''}</span>
776
+ </div>
777
+ ` : ''}
778
+ ${hasEnv ? `
779
+ <div class="flex items-center gap-2 text-muted-foreground">
780
+ <span class="font-mono text-xs bg-muted px-1.5 py-0.5 rounded">${t('mcp.env')}</span>
781
+ <span class="text-xs">${Object.keys(config.env).length} ${t('mcp.variables')}</span>
782
+ </div>
783
+ ` : ''}
784
+ </div>
785
+
786
+ <div class="mt-3 pt-3 border-t border-border flex items-center justify-between gap-2" onclick="event.stopPropagation()">
787
+ <div class="flex items-center gap-2">
788
+ <button class="text-xs text-success hover:text-success/80 transition-colors flex items-center gap-1"
789
+ data-server-name="${escapeHtml(name)}"
790
+ data-server-config="${escapeHtml(JSON.stringify(config))}"
791
+ data-action="save-as-template"
792
+ title="${t('mcp.saveAsTemplate')}">
793
+ <i data-lucide="save" class="w-3 h-3"></i>
794
+ ${t('mcp.saveAsTemplate')}
795
+ </button>
796
+ </div>
797
+ ${canRemove ? `
798
+ <button class="text-xs text-destructive hover:text-destructive/80 transition-colors"
799
+ data-server-name="${escapeHtml(name)}"
800
+ data-action="remove">
801
+ ${t('mcp.removeFromProject')}
802
+ </button>
803
+ ` : ''}
804
+ </div>
805
+ </div>
806
+ `;
807
+ }
808
+
809
+ // Render card for Global Management (manage global servers)
810
+ function renderGlobalManagementCard(serverName, serverConfig) {
811
+ const command = serverConfig.command || serverConfig.url || 'N/A';
812
+ const args = serverConfig.args || [];
813
+ const hasEnv = serverConfig.env && Object.keys(serverConfig.env).length > 0;
814
+ const serverType = serverConfig.type || 'stdio';
815
+
816
+ return `
817
+ <div class="mcp-server-card mcp-server-global bg-card border border-success/30 rounded-lg p-4 hover:shadow-md transition-all cursor-pointer"
818
+ data-server-name="${escapeHtml(serverName)}"
819
+ data-server-config="${escapeHtml(JSON.stringify(serverConfig))}"
820
+ data-server-source="global"
821
+ data-action="view-details"
822
+ title="${t('mcp.clickToEdit')}">
823
+ <div class="flex items-start justify-between mb-3">
824
+ <div class="flex items-center gap-2">
825
+ <i data-lucide="globe" class="w-5 h-5 text-success"></i>
826
+ <h4 class="font-semibold text-foreground">${escapeHtml(serverName)}</h4>
827
+ </div>
828
+ </div>
829
+
830
+ <div class="mcp-server-details text-sm space-y-1">
831
+ <div class="flex items-center gap-2 text-muted-foreground">
832
+ <span class="font-mono text-xs bg-muted px-1.5 py-0.5 rounded">${serverType === 'stdio' ? t('mcp.cmd') : t('mcp.url')}</span>
833
+ <span class="truncate" title="${escapeHtml(command)}">${escapeHtml(command)}</span>
834
+ </div>
835
+ ${args.length > 0 ? `
836
+ <div class="flex items-start gap-2 text-muted-foreground">
837
+ <span class="font-mono text-xs bg-muted px-1.5 py-0.5 rounded shrink-0">${t('mcp.args')}</span>
838
+ <span class="text-xs font-mono truncate" title="${escapeHtml(args.join(' '))}">${escapeHtml(args.slice(0, 3).join(' '))}${args.length > 3 ? '...' : ''}</span>
839
+ </div>
840
+ ` : ''}
841
+ ${hasEnv ? `
842
+ <div class="flex items-center gap-2 text-muted-foreground">
843
+ <span class="font-mono text-xs bg-muted px-1.5 py-0.5 rounded">${t('mcp.env')}</span>
844
+ <span class="text-xs">${Object.keys(serverConfig.env).length} ${t('mcp.variables')}</span>
845
+ </div>
846
+ ` : ''}
847
+ <div class="flex items-center gap-2 text-muted-foreground mt-1">
848
+ <span class="text-xs italic">${t('mcp.availableToAll')}</span>
849
+ </div>
850
+ </div>
851
+
852
+ <div class="mt-3 pt-3 border-t border-border flex items-center justify-end" onclick="event.stopPropagation()">
853
+ <button class="text-xs text-destructive hover:text-destructive/80 transition-colors"
854
+ data-server-name="${escapeHtml(serverName)}"
855
+ data-action="remove-global">
856
+ ${t('mcp.removeGlobal')}
857
+ </button>
858
+ </div>
859
+ </div>
860
+ `;
861
+ }
862
+
863
+ function renderAvailableServerCard(serverName, serverInfo) {
864
+ const serverConfig = serverInfo.config;
865
+ const usedIn = serverInfo.usedIn || [];
866
+ const command = serverConfig.command || 'N/A';
867
+ const args = serverConfig.args || [];
868
+
869
+ // Get the actual name to use when adding (original name if different from display key)
870
+ const originalName = serverInfo.originalName || serverName;
871
+ const hasVariant = serverInfo.originalName && serverInfo.originalName !== serverName;
872
+
873
+ // Get source project info
874
+ const sourceProject = serverInfo.sourceProject;
875
+ const sourceProjectName = sourceProject ? (sourceProject.split('\\').pop() || sourceProject.split('/').pop()) : null;
876
+
877
+ // Generate args preview
878
+ const argsPreview = args.length > 0 ? args.slice(0, 3).join(' ') + (args.length > 3 ? '...' : '') : '';
879
+
880
+ return `
881
+ <div class="mcp-server-card mcp-server-available bg-card border border-border border-dashed rounded-lg p-4 hover:shadow-md hover:border-solid transition-all">
882
+ <div class="flex items-start justify-between mb-3">
883
+ <div class="flex items-center gap-2 flex-wrap">
884
+ <span><i data-lucide="circle-dashed" class="w-5 h-5 text-muted-foreground"></i></span>
885
+ <h4 class="font-semibold text-foreground">${escapeHtml(originalName)}</h4>
886
+ ${hasVariant ? `
887
+ <span class="text-xs px-2 py-0.5 bg-warning/20 text-warning rounded-full" title="Different config from: ${escapeHtml(sourceProject || '')}">
888
+ ${escapeHtml(sourceProjectName || 'variant')}
889
+ </span>
890
+ ` : ''}
891
+ </div>
892
+ <div class="flex gap-2">
893
+ <button class="px-3 py-1 text-xs bg-primary text-primary-foreground rounded hover:opacity-90 transition-opacity"
894
+ data-server-name="${escapeHtml(originalName)}"
895
+ data-server-key="${escapeHtml(serverName)}"
896
+ data-server-config='${JSON.stringify(serverConfig).replace(/'/g, "&#39;")}'
897
+ data-scope="project"
898
+ data-action="add-from-other"
899
+ title="${t('mcp.installToProject')}">
900
+ <i data-lucide="folder-plus" class="w-3.5 h-3.5 inline"></i>
901
+ </button>
902
+ <button class="px-3 py-1 text-xs bg-success text-success-foreground rounded hover:opacity-90 transition-opacity"
903
+ data-server-name="${escapeHtml(originalName)}"
904
+ data-server-key="${escapeHtml(serverName)}"
905
+ data-server-config='${JSON.stringify(serverConfig).replace(/'/g, "&#39;")}'
906
+ data-scope="global"
907
+ data-action="add-from-other"
908
+ title="${t('mcp.installToGlobal')}">
909
+ <i data-lucide="globe" class="w-3.5 h-3.5 inline"></i>
910
+ </button>
911
+ </div>
912
+ </div>
913
+
914
+ <div class="mcp-server-details text-sm space-y-1">
915
+ <div class="flex items-center gap-2 text-muted-foreground">
916
+ <span class="font-mono text-xs bg-muted px-1.5 py-0.5 rounded">${t('mcp.cmd')}</span>
917
+ <span class="truncate" title="${escapeHtml(command)}">${escapeHtml(command)}</span>
918
+ </div>
919
+ ${argsPreview ? `
920
+ <div class="flex items-start gap-2 text-muted-foreground">
921
+ <span class="font-mono text-xs bg-muted px-1.5 py-0.5 rounded shrink-0">${t('mcp.args')}</span>
922
+ <span class="text-xs font-mono truncate" title="${escapeHtml(args.join(' '))}">${escapeHtml(argsPreview)}</span>
923
+ </div>
924
+ ` : ''}
925
+ <div class="flex items-center gap-2 text-muted-foreground">
926
+ <span class="text-xs">${t('mcp.usedInCount').replace('{count}', usedIn.length).replace('{s}', usedIn.length !== 1 ? 's' : '')}</span>
927
+ ${sourceProjectName ? `<span class="text-xs text-muted-foreground/70">• ${t('mcp.from')} ${escapeHtml(sourceProjectName)}</span>` : ''}
928
+ </div>
929
+ </div>
930
+
931
+ <div class="mt-3 pt-3 border-t border-border flex items-center gap-2">
932
+ <button class="text-xs text-primary hover:text-primary/80 transition-colors flex items-center gap-1"
933
+ data-server-name="${escapeHtml(originalName)}"
934
+ data-server-config="${escapeHtml(JSON.stringify(serverConfig))}"
935
+ data-action="install-to-project"
936
+ title="${t('mcp.installToProject')}">
937
+ <i data-lucide="download" class="w-3 h-3"></i>
938
+ ${t('mcp.installToProject')}
939
+ </button>
940
+ <button class="text-xs text-success hover:text-success/80 transition-colors flex items-center gap-1"
941
+ data-server-name="${escapeHtml(originalName)}"
942
+ data-server-config="${escapeHtml(JSON.stringify(serverConfig))}"
943
+ data-action="install-to-global"
944
+ title="${t('mcp.installToGlobal')}">
945
+ <i data-lucide="globe" class="w-3 h-3"></i>
946
+ ${t('mcp.installToGlobal')}
947
+ </button>
948
+ </div>
949
+ </div>
950
+ `;
951
+ }
952
+
953
+ // Render available server card for Codex mode (with Claude badge and copy to Codex button)
954
+ function renderAvailableServerCardForCodex(serverName, serverInfo) {
955
+ const serverConfig = serverInfo.config;
956
+ const usedIn = serverInfo.usedIn || [];
957
+ const command = serverConfig.command || serverConfig.url || 'N/A';
958
+ const args = serverConfig.args || [];
959
+
960
+ // Get the actual name to use when adding
961
+ const originalName = serverInfo.originalName || serverName;
962
+ const hasVariant = serverInfo.originalName && serverInfo.originalName !== serverName;
963
+
964
+ // Get source project info
965
+ const sourceProject = serverInfo.sourceProject;
966
+ const sourceProjectName = sourceProject ? (sourceProject.split('\\').pop() || sourceProject.split('/').pop()) : null;
967
+
968
+ // Generate args preview
969
+ const argsPreview = args.length > 0 ? args.slice(0, 3).join(' ') + (args.length > 3 ? '...' : '') : '';
970
+
971
+ // Check if already in Codex
972
+ const alreadyInCodex = codexMcpServers && codexMcpServers[originalName];
973
+
974
+ return `
975
+ <div class="mcp-server-card mcp-server-available bg-card border ${alreadyInCodex ? 'border-success/50' : 'border-border'} border-dashed rounded-lg p-4 hover:shadow-md hover:border-solid transition-all">
976
+ <div class="flex items-start justify-between mb-3">
977
+ <div class="flex items-center gap-2 flex-wrap">
978
+ <span><i data-lucide="circle-dashed" class="w-5 h-5 text-muted-foreground"></i></span>
979
+ <h4 class="font-semibold text-foreground">${escapeHtml(originalName)}</h4>
980
+ <span class="text-xs px-2 py-0.5 bg-primary/10 text-primary rounded-full">Claude</span>
981
+ ${hasVariant ? `
982
+ <span class="text-xs px-2 py-0.5 bg-warning/20 text-warning rounded-full" title="Different config from: ${escapeHtml(sourceProject || '')}">
983
+ ${escapeHtml(sourceProjectName || 'variant')}
984
+ </span>
985
+ ` : ''}
986
+ ${alreadyInCodex ? `<span class="text-xs px-2 py-0.5 bg-success/10 text-success rounded-full">${t('mcp.codex.alreadyAdded')}</span>` : ''}
987
+ </div>
988
+ ${!alreadyInCodex ? `
989
+ <button class="px-3 py-1 text-xs bg-orange-500 text-white rounded hover:opacity-90 transition-opacity"
990
+ onclick="copyClaudeServerToCodex('${escapeHtml(originalName)}', ${JSON.stringify(serverConfig).replace(/'/g, "&#39;")})"
991
+ title="${t('mcp.codex.copyToCodex')}">
992
+ <i data-lucide="arrow-right" class="w-3.5 h-3.5 inline"></i> Codex
993
+ </button>
994
+ ` : ''}
995
+ </div>
996
+
997
+ <div class="mcp-server-details text-sm space-y-1">
998
+ <div class="flex items-center gap-2 text-muted-foreground">
999
+ <span class="font-mono text-xs bg-muted px-1.5 py-0.5 rounded">${t('mcp.cmd')}</span>
1000
+ <span class="truncate" title="${escapeHtml(command)}">${escapeHtml(command)}</span>
1001
+ </div>
1002
+ ${argsPreview ? `
1003
+ <div class="flex items-start gap-2 text-muted-foreground">
1004
+ <span class="font-mono text-xs bg-muted px-1.5 py-0.5 rounded shrink-0">${t('mcp.args')}</span>
1005
+ <span class="text-xs font-mono truncate" title="${escapeHtml(args.join(' '))}">${escapeHtml(argsPreview)}</span>
1006
+ </div>
1007
+ ` : ''}
1008
+ <div class="flex items-center gap-2 text-muted-foreground">
1009
+ <span class="text-xs">${t('mcp.usedInCount').replace('{count}', usedIn.length).replace('{s}', usedIn.length !== 1 ? 's' : '')}</span>
1010
+ ${sourceProjectName ? `<span class="text-xs text-muted-foreground/70">• ${t('mcp.from')} ${escapeHtml(sourceProjectName)}</span>` : ''}
1011
+ </div>
1012
+ </div>
1013
+
1014
+ <div class="mt-3 pt-3 border-t border-border flex items-center gap-2">
1015
+ <button class="text-xs text-orange-500 hover:text-orange-600 transition-colors flex items-center gap-1"
1016
+ onclick="copyClaudeServerToCodex('${escapeHtml(originalName)}', ${JSON.stringify(serverConfig).replace(/'/g, "&#39;")})"
1017
+ title="${t('mcp.codex.copyToCodex')}">
1018
+ <i data-lucide="download" class="w-3 h-3"></i>
1019
+ ${t('mcp.codex.install')}
1020
+ </button>
1021
+ </div>
1022
+ </div>
1023
+ `;
1024
+ }
1025
+
1026
+ // ========================================
1027
+ // Codex MCP Server Card Renderer
1028
+ // ========================================
1029
+
1030
+ function renderCodexServerCard(serverName, serverConfig) {
1031
+ const isStdio = !!serverConfig.command;
1032
+ const isHttp = !!serverConfig.url;
1033
+ const isEnabled = serverConfig.enabled !== false; // Default to enabled
1034
+ const command = serverConfig.command || serverConfig.url || 'N/A';
1035
+ const args = serverConfig.args || [];
1036
+ const hasEnv = serverConfig.env && Object.keys(serverConfig.env).length > 0;
1037
+
1038
+ // Server type badge
1039
+ const typeBadge = isHttp
1040
+ ? `<span class="text-xs px-2 py-0.5 bg-blue-100 text-blue-700 dark:bg-blue-900/30 dark:text-blue-300 rounded-full">HTTP</span>`
1041
+ : `<span class="text-xs px-2 py-0.5 bg-green-100 text-green-700 dark:bg-green-900/30 dark:text-green-300 rounded-full">STDIO</span>`;
1042
+
1043
+ return `
1044
+ <div class="mcp-server-card bg-card border border-orange-200 dark:border-orange-800 rounded-lg p-4 hover:shadow-md transition-all cursor-pointer ${!isEnabled ? 'opacity-60' : ''}"
1045
+ data-server-name="${escapeHtml(serverName)}"
1046
+ data-server-config="${escapeHtml(JSON.stringify(serverConfig))}"
1047
+ data-cli-type="codex"
1048
+ data-action="view-details-codex"
1049
+ title="${t('mcp.clickToEdit')}">
1050
+ <div class="flex items-start justify-between mb-3">
1051
+ <div class="flex items-center gap-2 flex-wrap">
1052
+ <span>${isEnabled ? '<i data-lucide="check-circle" class="w-5 h-5 text-orange-500"></i>' : '<i data-lucide="circle" class="w-5 h-5 text-muted-foreground"></i>'}</span>
1053
+ <h4 class="font-semibold text-foreground">${escapeHtml(serverName)}</h4>
1054
+ <span class="text-xs px-2 py-0.5 bg-orange-100 text-orange-700 dark:bg-orange-900/30 dark:text-orange-300 rounded-full">Codex</span>
1055
+ ${typeBadge}
1056
+ </div>
1057
+ <label class="mcp-toggle relative inline-flex items-center cursor-pointer" onclick="event.stopPropagation()">
1058
+ <input type="checkbox" class="sr-only peer"
1059
+ ${isEnabled ? 'checked' : ''}
1060
+ data-server-name="${escapeHtml(serverName)}"
1061
+ data-action="toggle-codex">
1062
+ <div class="w-9 h-5 bg-hover peer-focus:outline-none rounded-full peer peer-checked:after:translate-x-full peer-checked:after:border-white after:content-[''] after:absolute after:top-[2px] after:left-[2px] after:bg-white after:border-gray-300 after:border after:rounded-full after:h-4 after:w-4 after:transition-all peer-checked:bg-orange-500"></div>
1063
+ </label>
1064
+ </div>
1065
+
1066
+ <div class="mcp-server-details text-sm space-y-1">
1067
+ <div class="flex items-center gap-2 text-muted-foreground">
1068
+ <span class="font-mono text-xs bg-muted px-1.5 py-0.5 rounded">${isHttp ? t('mcp.url') : t('mcp.cmd')}</span>
1069
+ <span class="truncate" title="${escapeHtml(command)}">${escapeHtml(command)}</span>
1070
+ </div>
1071
+ ${args.length > 0 ? `
1072
+ <div class="flex items-start gap-2 text-muted-foreground">
1073
+ <span class="font-mono text-xs bg-muted px-1.5 py-0.5 rounded shrink-0">${t('mcp.args')}</span>
1074
+ <span class="text-xs font-mono truncate" title="${escapeHtml(args.join(' '))}">${escapeHtml(args.slice(0, 3).join(' '))}${args.length > 3 ? '...' : ''}</span>
1075
+ </div>
1076
+ ` : ''}
1077
+ ${hasEnv ? `
1078
+ <div class="flex items-center gap-2 text-muted-foreground">
1079
+ <span class="font-mono text-xs bg-muted px-1.5 py-0.5 rounded">${t('mcp.env')}</span>
1080
+ <span class="text-xs">${Object.keys(serverConfig.env).length} ${t('mcp.variables')}</span>
1081
+ </div>
1082
+ ` : ''}
1083
+ ${serverConfig.enabled_tools ? `
1084
+ <div class="flex items-center gap-2 text-muted-foreground">
1085
+ <span class="font-mono text-xs bg-muted px-1.5 py-0.5 rounded">${t('mcp.codex.enabledTools')}</span>
1086
+ <span class="text-xs">${serverConfig.enabled_tools.length} ${t('mcp.codex.tools')}</span>
1087
+ </div>
1088
+ ` : ''}
1089
+ </div>
1090
+
1091
+ <div class="mt-3 pt-3 border-t border-border flex items-center justify-between gap-2" onclick="event.stopPropagation()">
1092
+ <div class="flex items-center gap-2">
1093
+ <button class="text-xs text-primary hover:text-primary/80 transition-colors flex items-center gap-1"
1094
+ onclick="copyCodexServerToClaude('${escapeHtml(serverName)}', ${JSON.stringify(serverConfig).replace(/'/g, "&#39;")})"
1095
+ title="${t('mcp.codex.copyToClaude')}">
1096
+ <i data-lucide="copy" class="w-3 h-3"></i>
1097
+ ${t('mcp.codex.copyToClaude')}
1098
+ </button>
1099
+ </div>
1100
+ <button class="text-xs text-destructive hover:text-destructive/80 transition-colors"
1101
+ data-server-name="${escapeHtml(serverName)}"
1102
+ data-action="remove-codex">
1103
+ ${t('mcp.codex.remove')}
1104
+ </button>
1105
+ </div>
1106
+ </div>
1107
+ `;
1108
+ }
1109
+
1110
+ // ========================================
1111
+ // Codex MCP Create Modal
1112
+ // ========================================
1113
+
1114
+ function openCodexMcpCreateModal() {
1115
+ // Reuse the existing modal with different settings
1116
+ const modal = document.getElementById('mcpCreateModal');
1117
+ if (modal) {
1118
+ modal.classList.remove('hidden');
1119
+ // Reset to form mode
1120
+ mcpCreateMode = 'form';
1121
+ switchMcpCreateTab('form');
1122
+ // Clear form
1123
+ document.getElementById('mcpServerName').value = '';
1124
+ document.getElementById('mcpServerCommand').value = '';
1125
+ document.getElementById('mcpServerArgs').value = '';
1126
+ document.getElementById('mcpServerEnv').value = '';
1127
+ // Clear JSON input
1128
+ document.getElementById('mcpServerJson').value = '';
1129
+ document.getElementById('mcpJsonPreview').classList.add('hidden');
1130
+ // Set scope to codex
1131
+ const scopeSelect = document.getElementById('mcpServerScope');
1132
+ if (scopeSelect) {
1133
+ // Add codex option if not exists
1134
+ if (!scopeSelect.querySelector('option[value="codex"]')) {
1135
+ const codexOption = document.createElement('option');
1136
+ codexOption.value = 'codex';
1137
+ codexOption.textContent = t('mcp.codex.scopeCodex');
1138
+ scopeSelect.appendChild(codexOption);
1139
+ }
1140
+ scopeSelect.value = 'codex';
1141
+ }
1142
+ // Focus on name input
1143
+ document.getElementById('mcpServerName').focus();
1144
+ // Setup JSON input listener
1145
+ setupMcpJsonListener();
1146
+ }
1147
+ }
1148
+
1149
+ function attachMcpEventListeners() {
1150
+ // Toggle switches
1151
+ document.querySelectorAll('.mcp-server-card input[data-action="toggle"]').forEach(input => {
1152
+ input.addEventListener('change', async (e) => {
1153
+ const serverName = e.target.dataset.serverName;
1154
+ const enable = e.target.checked;
1155
+ await toggleMcpServer(serverName, enable);
1156
+ });
1157
+ });
1158
+
1159
+ // Add from other projects (with scope selection)
1160
+ document.querySelectorAll('.mcp-server-card button[data-action="add-from-other"]').forEach(btn => {
1161
+ btn.addEventListener('click', async (e) => {
1162
+ const serverName = btn.dataset.serverName;
1163
+ const serverConfig = JSON.parse(btn.dataset.serverConfig);
1164
+ const scope = btn.dataset.scope; // 'project' or 'global'
1165
+
1166
+ if (scope === 'global') {
1167
+ await addGlobalMcpServer(serverName, serverConfig);
1168
+ } else {
1169
+ await copyMcpServerToProject(serverName, serverConfig);
1170
+ }
1171
+ });
1172
+ });
1173
+
1174
+ // Remove buttons (project-level)
1175
+ document.querySelectorAll('.mcp-server-card button[data-action="remove"]').forEach(btn => {
1176
+ btn.addEventListener('click', async (e) => {
1177
+ const serverName = btn.dataset.serverName;
1178
+ if (confirm(t('mcp.removeConfirm', { name: serverName }))) {
1179
+ await removeMcpServerFromProject(serverName);
1180
+ }
1181
+ });
1182
+ });
1183
+
1184
+ // Remove buttons (global-level)
1185
+ document.querySelectorAll('.mcp-server-card button[data-action="remove-global"]').forEach(btn => {
1186
+ btn.addEventListener('click', async (e) => {
1187
+ const serverName = btn.dataset.serverName;
1188
+ if (confirm(t('mcp.removeGlobalConfirm', { name: serverName }))) {
1189
+ await removeGlobalMcpServer(serverName);
1190
+ }
1191
+ });
1192
+ });
1193
+
1194
+ // Install to project buttons
1195
+ document.querySelectorAll('.mcp-server-card button[data-action="install-to-project"]').forEach(btn => {
1196
+ btn.addEventListener('click', async (e) => {
1197
+ const serverName = btn.dataset.serverName;
1198
+ const serverConfig = JSON.parse(btn.dataset.serverConfig);
1199
+ await installMcpToProject(serverName, serverConfig);
1200
+ });
1201
+ });
1202
+
1203
+ // Install to global buttons
1204
+ document.querySelectorAll('.mcp-server-card button[data-action="install-to-global"]').forEach(btn => {
1205
+ btn.addEventListener('click', async (e) => {
1206
+ const serverName = btn.dataset.serverName;
1207
+ const serverConfig = JSON.parse(btn.dataset.serverConfig);
1208
+ await addGlobalMcpServer(serverName, serverConfig);
1209
+ });
1210
+ });
1211
+
1212
+ // Save as template buttons
1213
+ document.querySelectorAll('.mcp-server-card button[data-action="save-as-template"]').forEach(btn => {
1214
+ btn.addEventListener('click', async (e) => {
1215
+ const serverName = btn.dataset.serverName;
1216
+ const serverConfig = JSON.parse(btn.dataset.serverConfig);
1217
+ await saveMcpAsTemplate(serverName, serverConfig);
1218
+ });
1219
+ });
1220
+
1221
+ // Install from template buttons
1222
+ document.querySelectorAll('.mcp-template-card button[data-action="install-template"]').forEach(btn => {
1223
+ btn.addEventListener('click', async (e) => {
1224
+ const templateName = btn.dataset.templateName;
1225
+ const scope = btn.dataset.scope || 'project';
1226
+ await installFromTemplate(templateName, scope);
1227
+ });
1228
+ });
1229
+
1230
+ // Delete template buttons
1231
+ document.querySelectorAll('.mcp-template-card button[data-action="delete-template"]').forEach(btn => {
1232
+ btn.addEventListener('click', async (e) => {
1233
+ const templateName = btn.dataset.templateName;
1234
+ if (confirm(t('mcp.deleteTemplateConfirm', { name: templateName }))) {
1235
+ await deleteMcpTemplate(templateName);
1236
+ }
1237
+ });
1238
+ });
1239
+
1240
+ // ========================================
1241
+ // Codex MCP Event Listeners
1242
+ // ========================================
1243
+
1244
+ // Toggle Codex MCP servers
1245
+ document.querySelectorAll('.mcp-server-card input[data-action="toggle-codex"]').forEach(input => {
1246
+ input.addEventListener('change', async (e) => {
1247
+ const serverName = e.target.dataset.serverName;
1248
+ const enable = e.target.checked;
1249
+ await toggleCodexMcpServer(serverName, enable);
1250
+ });
1251
+ });
1252
+
1253
+ // Remove Codex MCP servers
1254
+ document.querySelectorAll('.mcp-server-card button[data-action="remove-codex"]').forEach(btn => {
1255
+ btn.addEventListener('click', async (e) => {
1256
+ const serverName = btn.dataset.serverName;
1257
+ if (confirm(t('mcp.codex.removeConfirm', { name: serverName }))) {
1258
+ await removeCodexMcpServer(serverName);
1259
+ }
1260
+ });
1261
+ });
1262
+
1263
+ // View details / Edit - click on Claude server card
1264
+ document.querySelectorAll('.mcp-server-card[data-action="view-details"]').forEach(card => {
1265
+ card.addEventListener('click', (e) => {
1266
+ const serverName = card.dataset.serverName;
1267
+ const serverConfig = JSON.parse(card.dataset.serverConfig);
1268
+ const serverSource = card.dataset.serverSource;
1269
+ showMcpEditModal(serverName, serverConfig, serverSource, 'claude');
1270
+ });
1271
+ });
1272
+
1273
+ // View details / Edit - click on Codex server card
1274
+ document.querySelectorAll('.mcp-server-card[data-action="view-details-codex"]').forEach(card => {
1275
+ card.addEventListener('click', (e) => {
1276
+ const serverName = card.dataset.serverName;
1277
+ const serverConfig = JSON.parse(card.dataset.serverConfig);
1278
+ showMcpEditModal(serverName, serverConfig, 'codex', 'codex');
1279
+ });
1280
+ });
1281
+
1282
+ // Modal close button
1283
+ const closeBtn = document.getElementById('mcpDetailsModalClose');
1284
+ const modal = document.getElementById('mcpDetailsModal');
1285
+ if (closeBtn && modal) {
1286
+ closeBtn.addEventListener('click', () => {
1287
+ modal.classList.add('hidden');
1288
+ });
1289
+ // Close on background click
1290
+ modal.addEventListener('click', (e) => {
1291
+ if (e.target === modal) {
1292
+ modal.classList.add('hidden');
1293
+ }
1294
+ });
1295
+ }
1296
+ }
1297
+
1298
+ // ========================================
1299
+ // MCP Edit Modal (replaces Details Modal)
1300
+ // ========================================
1301
+
1302
+ // Store current editing context
1303
+ let mcpEditContext = {
1304
+ serverName: null,
1305
+ serverConfig: null,
1306
+ serverSource: null,
1307
+ cliType: 'claude'
1308
+ };
1309
+
1310
+ function showMcpDetails(serverName, serverConfig, serverSource, cliType = 'claude') {
1311
+ showMcpEditModal(serverName, serverConfig, serverSource, cliType);
1312
+ }
1313
+
1314
+ function showMcpEditModal(serverName, serverConfig, serverSource, cliType = 'claude') {
1315
+ const modal = document.getElementById('mcpDetailsModal');
1316
+ const modalBody = document.getElementById('mcpDetailsModalBody');
1317
+
1318
+ if (!modal || !modalBody) return;
1319
+
1320
+ // Store editing context
1321
+ mcpEditContext = {
1322
+ serverName,
1323
+ serverConfig: JSON.parse(JSON.stringify(serverConfig)), // Deep clone
1324
+ serverSource,
1325
+ cliType
1326
+ };
1327
+
1328
+ // Check if editable (enterprise is read-only)
1329
+ const isReadOnly = serverSource === 'enterprise';
1330
+ const isCodex = cliType === 'codex';
1331
+
1332
+ // Build source badge
1333
+ let sourceBadge = '';
1334
+ if (serverSource === 'enterprise') {
1335
+ sourceBadge = `<span class="inline-flex items-center px-2 py-1 text-xs font-semibold rounded-full bg-warning/20 text-warning">${t('mcp.sourceEnterprise')}</span>`;
1336
+ } else if (serverSource === 'global') {
1337
+ sourceBadge = `<span class="inline-flex items-center px-2 py-1 text-xs font-semibold rounded-full bg-success/10 text-success">${t('mcp.sourceGlobal')}</span>`;
1338
+ } else if (serverSource === 'project') {
1339
+ sourceBadge = `<span class="inline-flex items-center px-2 py-1 text-xs font-semibold rounded-full bg-primary/10 text-primary">${t('mcp.sourceProject')}</span>`;
1340
+ } else if (isCodex) {
1341
+ sourceBadge = `<span class="inline-flex items-center px-2 py-1 text-xs font-semibold rounded-full bg-orange-100 text-orange-700 dark:bg-orange-900/30 dark:text-orange-300">Codex</span>`;
1342
+ }
1343
+
1344
+ // Format args and env for textarea
1345
+ const argsText = (serverConfig.args || []).join('\n');
1346
+ const envText = Object.entries(serverConfig.env || {}).map(([k, v]) => `${k}=${v}`).join('\n');
1347
+
1348
+ // Build edit form HTML
1349
+ modalBody.innerHTML = `
1350
+ <div class="space-y-4">
1351
+ <!-- Server Name and Source -->
1352
+ <div>
1353
+ <label class="text-xs font-semibold text-muted-foreground uppercase tracking-wide">${t('mcp.detailsModal.serverName')}</label>
1354
+ <div class="mt-1 flex items-center gap-2">
1355
+ <input type="text" id="mcpEditName" value="${escapeHtml(serverName)}"
1356
+ class="text-lg font-bold text-foreground bg-transparent border-b border-border focus:border-primary outline-none px-1 py-0.5 flex-1"
1357
+ ${isReadOnly ? 'disabled' : ''}
1358
+ placeholder="${t('mcp.editModal.serverNamePlaceholder')}">
1359
+ ${sourceBadge}
1360
+ </div>
1361
+ </div>
1362
+
1363
+ <!-- Command/URL -->
1364
+ <div>
1365
+ <label class="text-xs font-semibold text-muted-foreground uppercase tracking-wide mb-1 block">
1366
+ ${serverConfig.url ? t('mcp.url') : t('mcp.cmd')}
1367
+ </label>
1368
+ <input type="text" id="mcpEditCommand" value="${escapeHtml(serverConfig.command || serverConfig.url || '')}"
1369
+ class="w-full px-3 py-2 text-sm font-mono bg-muted border border-border rounded-lg focus:border-primary outline-none"
1370
+ ${isReadOnly ? 'disabled' : ''}
1371
+ placeholder="${serverConfig.url ? 'https://...' : 'npx, node, python...'}">
1372
+ </div>
1373
+
1374
+ <!-- Arguments -->
1375
+ <div>
1376
+ <label class="text-xs font-semibold text-muted-foreground uppercase tracking-wide mb-1 block">
1377
+ ${t('mcp.args')} <span class="font-normal">(${t('mcp.editModal.onePerLine')})</span>
1378
+ </label>
1379
+ <textarea id="mcpEditArgs" rows="3"
1380
+ class="w-full px-3 py-2 text-sm font-mono bg-muted border border-border rounded-lg focus:border-primary outline-none resize-none"
1381
+ ${isReadOnly ? 'disabled' : ''}
1382
+ placeholder="-y&#10;package-name">${escapeHtml(argsText)}</textarea>
1383
+ </div>
1384
+
1385
+ <!-- Environment Variables -->
1386
+ <div>
1387
+ <label class="text-xs font-semibold text-muted-foreground uppercase tracking-wide mb-1 block">
1388
+ ${t('mcp.env')} <span class="font-normal">(KEY=VALUE ${t('mcp.editModal.onePerLine')})</span>
1389
+ </label>
1390
+ <textarea id="mcpEditEnv" rows="3"
1391
+ class="w-full px-3 py-2 text-sm font-mono bg-muted border border-border rounded-lg focus:border-primary outline-none resize-none"
1392
+ ${isReadOnly ? 'disabled' : ''}
1393
+ placeholder="API_KEY=your-key&#10;DEBUG=true">${escapeHtml(envText)}</textarea>
1394
+ </div>
1395
+
1396
+ ${isCodex ? `
1397
+ <!-- Codex-specific: enabled_tools -->
1398
+ <div>
1399
+ <label class="text-xs font-semibold text-muted-foreground uppercase tracking-wide mb-1 block">
1400
+ ${t('mcp.codex.enabledTools')} <span class="font-normal">(${t('mcp.editModal.onePerLine')})</span>
1401
+ </label>
1402
+ <textarea id="mcpEditEnabledTools" rows="2"
1403
+ class="w-full px-3 py-2 text-sm font-mono bg-muted border border-border rounded-lg focus:border-primary outline-none resize-none"
1404
+ ${isReadOnly ? 'disabled' : ''}
1405
+ placeholder="tool1&#10;tool2">${escapeHtml((serverConfig.enabled_tools || []).join('\n'))}</textarea>
1406
+ </div>
1407
+ ` : ''}
1408
+
1409
+ <!-- Raw JSON Preview (collapsible) -->
1410
+ <details class="group">
1411
+ <summary class="text-xs font-semibold text-muted-foreground uppercase tracking-wide cursor-pointer flex items-center gap-1">
1412
+ <i data-lucide="chevron-right" class="w-3 h-3 transition-transform group-open:rotate-90"></i>
1413
+ Raw JSON
1414
+ </summary>
1415
+ <pre id="mcpEditJsonPreview" class="mt-2 bg-muted rounded-lg p-3 text-xs font-mono overflow-x-auto">${escapeHtml(JSON.stringify(serverConfig, null, 2))}</pre>
1416
+ </details>
1417
+
1418
+ <!-- Action Buttons -->
1419
+ ${!isReadOnly ? `
1420
+ <div class="flex items-center justify-between pt-4 border-t border-border">
1421
+ <div class="flex items-center gap-2">
1422
+ ${serverSource === 'project' || isCodex ? `
1423
+ <button onclick="deleteMcpFromEdit()" class="px-4 py-2 text-sm text-destructive hover:bg-destructive/10 rounded-lg transition-colors flex items-center gap-1.5">
1424
+ <i data-lucide="trash-2" class="w-4 h-4"></i>
1425
+ ${t('mcp.editModal.delete')}
1426
+ </button>
1427
+ ` : ''}
1428
+ </div>
1429
+ <div class="flex items-center gap-2">
1430
+ <button onclick="closeMcpEditModal()" class="px-4 py-2 text-sm text-muted-foreground hover:bg-muted rounded-lg transition-colors">
1431
+ ${t('common.cancel')}
1432
+ </button>
1433
+ <button onclick="saveMcpEdit()" class="px-4 py-2 text-sm bg-primary text-primary-foreground rounded-lg hover:opacity-90 transition-opacity flex items-center gap-1.5">
1434
+ <i data-lucide="check" class="w-4 h-4"></i>
1435
+ ${t('mcp.editModal.save')}
1436
+ </button>
1437
+ </div>
1438
+ </div>
1439
+ ` : `
1440
+ <div class="flex items-center justify-end pt-4 border-t border-border">
1441
+ <button onclick="closeMcpEditModal()" class="px-4 py-2 text-sm bg-muted text-foreground rounded-lg hover:bg-muted/80 transition-colors">
1442
+ ${t('common.close')}
1443
+ </button>
1444
+ </div>
1445
+ `}
1446
+ </div>
1447
+ `;
1448
+
1449
+ // Update modal title
1450
+ const modalTitle = modal.querySelector('h2');
1451
+ if (modalTitle) {
1452
+ modalTitle.textContent = isReadOnly ? t('mcp.detailsModal.title') : t('mcp.editModal.title');
1453
+ }
1454
+
1455
+ // Show modal
1456
+ modal.classList.remove('hidden');
1457
+
1458
+ // Re-initialize Lucide icons in modal
1459
+ if (typeof lucide !== 'undefined') lucide.createIcons();
1460
+
1461
+ // Add input listeners to update JSON preview
1462
+ if (!isReadOnly) {
1463
+ ['mcpEditCommand', 'mcpEditArgs', 'mcpEditEnv', 'mcpEditEnabledTools'].forEach(id => {
1464
+ const el = document.getElementById(id);
1465
+ if (el) {
1466
+ el.addEventListener('input', updateMcpEditJsonPreview);
1467
+ }
1468
+ });
1469
+ }
1470
+ }
1471
+
1472
+ function closeMcpEditModal() {
1473
+ const modal = document.getElementById('mcpDetailsModal');
1474
+ if (modal) {
1475
+ modal.classList.add('hidden');
1476
+ }
1477
+ mcpEditContext = { serverName: null, serverConfig: null, serverSource: null, cliType: 'claude' };
1478
+ }
1479
+
1480
+ function updateMcpEditJsonPreview() {
1481
+ const preview = document.getElementById('mcpEditJsonPreview');
1482
+ if (!preview) return;
1483
+
1484
+ const config = buildConfigFromEditForm();
1485
+ preview.textContent = JSON.stringify(config, null, 2);
1486
+ }
1487
+
1488
+ function buildConfigFromEditForm() {
1489
+ const command = document.getElementById('mcpEditCommand')?.value.trim() || '';
1490
+ const argsText = document.getElementById('mcpEditArgs')?.value.trim() || '';
1491
+ const envText = document.getElementById('mcpEditEnv')?.value.trim() || '';
1492
+ const enabledToolsEl = document.getElementById('mcpEditEnabledTools');
1493
+
1494
+ // Build config
1495
+ const config = {};
1496
+
1497
+ // Command or URL
1498
+ if (mcpEditContext.serverConfig?.url) {
1499
+ config.url = command;
1500
+ } else {
1501
+ config.command = command;
1502
+ }
1503
+
1504
+ // Args
1505
+ if (argsText) {
1506
+ config.args = argsText.split('\n').map(a => a.trim()).filter(a => a);
1507
+ }
1508
+
1509
+ // Env
1510
+ if (envText) {
1511
+ config.env = {};
1512
+ envText.split('\n').forEach(line => {
1513
+ const trimmed = line.trim();
1514
+ if (trimmed && trimmed.includes('=')) {
1515
+ const eqIndex = trimmed.indexOf('=');
1516
+ const key = trimmed.substring(0, eqIndex).trim();
1517
+ const value = trimmed.substring(eqIndex + 1).trim();
1518
+ if (key) {
1519
+ config.env[key] = value;
1520
+ }
1521
+ }
1522
+ });
1523
+ }
1524
+
1525
+ // Codex-specific: enabled_tools
1526
+ if (enabledToolsEl) {
1527
+ const toolsText = enabledToolsEl.value.trim();
1528
+ if (toolsText) {
1529
+ config.enabled_tools = toolsText.split('\n').map(t => t.trim()).filter(t => t);
1530
+ }
1531
+ }
1532
+
1533
+ return config;
1534
+ }
1535
+
1536
+ async function saveMcpEdit() {
1537
+ const newName = document.getElementById('mcpEditName')?.value.trim();
1538
+ if (!newName) {
1539
+ showRefreshToast(t('mcp.editModal.nameRequired'), 'error');
1540
+ return;
1541
+ }
1542
+
1543
+ const newConfig = buildConfigFromEditForm();
1544
+
1545
+ if (!newConfig.command && !newConfig.url) {
1546
+ showRefreshToast(t('mcp.editModal.commandRequired'), 'error');
1547
+ return;
1548
+ }
1549
+
1550
+ const { serverName, serverSource, cliType } = mcpEditContext;
1551
+ const nameChanged = newName !== serverName;
1552
+
1553
+ try {
1554
+ if (cliType === 'codex') {
1555
+ // Codex MCP update
1556
+ // If name changed, remove old and add new
1557
+ if (nameChanged) {
1558
+ await removeCodexMcpServer(serverName);
1559
+ }
1560
+ await addCodexMcpServer(newName, newConfig);
1561
+ } else if (serverSource === 'global') {
1562
+ // Global MCP update
1563
+ if (nameChanged) {
1564
+ await removeGlobalMcpServer(serverName);
1565
+ }
1566
+ await addGlobalMcpServer(newName, newConfig);
1567
+ } else if (serverSource === 'project') {
1568
+ // Project MCP update
1569
+ if (nameChanged) {
1570
+ await removeMcpServerFromProject(serverName);
1571
+ }
1572
+ await copyMcpServerToProject(newName, newConfig, 'mcp');
1573
+ }
1574
+
1575
+ closeMcpEditModal();
1576
+ showRefreshToast(t('mcp.editModal.saved', { name: newName }), 'success');
1577
+ } catch (err) {
1578
+ console.error('Failed to save MCP edit:', err);
1579
+ showRefreshToast(t('mcp.editModal.saveFailed') + ': ' + err.message, 'error');
1580
+ }
1581
+ }
1582
+
1583
+ async function deleteMcpFromEdit() {
1584
+ const { serverName, serverSource, cliType } = mcpEditContext;
1585
+
1586
+ if (!confirm(t('mcp.editModal.deleteConfirm', { name: serverName }))) {
1587
+ return;
1588
+ }
1589
+
1590
+ try {
1591
+ if (cliType === 'codex') {
1592
+ await removeCodexMcpServer(serverName);
1593
+ } else if (serverSource === 'global') {
1594
+ await removeGlobalMcpServer(serverName);
1595
+ } else if (serverSource === 'project') {
1596
+ await removeMcpServerFromProject(serverName);
1597
+ }
1598
+
1599
+ closeMcpEditModal();
1600
+ showRefreshToast(t('mcp.editModal.deleted', { name: serverName }), 'success');
1601
+ } catch (err) {
1602
+ console.error('Failed to delete MCP:', err);
1603
+ showRefreshToast(t('mcp.editModal.deleteFailed') + ': ' + err.message, 'error');
1604
+ }
1605
+ }
1606
+
1607
+ // ========================================
1608
+ // MCP Template Management Functions
1609
+ // ========================================
1610
+
1611
+ let mcpTemplates = [];
1612
+
1613
+ /**
1614
+ * Load all MCP templates from API
1615
+ */
1616
+ async function loadMcpTemplates() {
1617
+ try {
1618
+ const response = await fetch('/api/mcp-templates');
1619
+ const data = await response.json();
1620
+
1621
+ if (data.success) {
1622
+ mcpTemplates = data.templates || [];
1623
+ console.log('[MCP Templates] Loaded', mcpTemplates.length, 'templates');
1624
+ } else {
1625
+ console.error('[MCP Templates] Failed to load:', data.error);
1626
+ mcpTemplates = [];
1627
+ }
1628
+
1629
+ return mcpTemplates;
1630
+ } catch (error) {
1631
+ console.error('[MCP Templates] Error loading templates:', error);
1632
+ mcpTemplates = [];
1633
+ return [];
1634
+ }
1635
+ }
1636
+
1637
+ /**
1638
+ * Save MCP server configuration as a template
1639
+ */
1640
+ async function saveMcpAsTemplate(serverName, serverConfig) {
1641
+ try {
1642
+ // Prompt for template name and description
1643
+ const templateName = prompt(t('mcp.enterTemplateName'), serverName);
1644
+ if (!templateName) return;
1645
+
1646
+ const description = prompt(t('mcp.enterTemplateDesc'), `Template for ${serverName}`);
1647
+
1648
+ const payload = {
1649
+ name: templateName,
1650
+ description: description || '',
1651
+ serverConfig: serverConfig,
1652
+ category: 'user'
1653
+ };
1654
+
1655
+ const response = await fetch('/api/mcp-templates', {
1656
+ method: 'POST',
1657
+ headers: { 'Content-Type': 'application/json' },
1658
+ body: JSON.stringify(payload)
1659
+ });
1660
+
1661
+ const data = await response.json();
1662
+
1663
+ if (data.success) {
1664
+ showRefreshToast(t('mcp.templateSaved', { name: templateName }), 'success');
1665
+ await loadMcpTemplates();
1666
+ await renderMcpManager(); // Refresh view
1667
+ } else {
1668
+ showRefreshToast(t('mcp.templateSaveFailed', { error: data.error }), 'error');
1669
+ }
1670
+ } catch (error) {
1671
+ console.error('[MCP] Save template error:', error);
1672
+ showRefreshToast(t('mcp.templateSaveFailed', { error: error.message }), 'error');
1673
+ }
1674
+ }
1675
+
1676
+ /**
1677
+ * Install MCP server from template
1678
+ */
1679
+ async function installFromTemplate(templateName, scope = 'project') {
1680
+ try {
1681
+ // Find template
1682
+ const template = mcpTemplates.find(t => t.name === templateName);
1683
+ if (!template) {
1684
+ showRefreshToast(t('mcp.templateNotFound', { name: templateName }), 'error');
1685
+ return;
1686
+ }
1687
+
1688
+ // Prompt for server name (default to template name)
1689
+ const serverName = prompt(t('mcp.enterServerName'), templateName);
1690
+ if (!serverName) return;
1691
+
1692
+ // Install based on scope
1693
+ if (scope === 'project') {
1694
+ await installMcpToProject(serverName, template.serverConfig);
1695
+ } else if (scope === 'global') {
1696
+ await addGlobalMcpServer(serverName, template.serverConfig);
1697
+ }
1698
+
1699
+ showRefreshToast(t('mcp.templateInstalled', { name: serverName }), 'success');
1700
+ await renderMcpManager();
1701
+ } catch (error) {
1702
+ console.error('[MCP] Install from template error:', error);
1703
+ showRefreshToast(t('mcp.templateInstallFailed', { error: error.message }), 'error');
1704
+ }
1705
+ }
1706
+
1707
+ /**
1708
+ * Delete MCP template
1709
+ */
1710
+ async function deleteMcpTemplate(templateName) {
1711
+ try {
1712
+ const response = await fetch(`/api/mcp-templates/${encodeURIComponent(templateName)}`, {
1713
+ method: 'DELETE'
1714
+ });
1715
+
1716
+ const data = await response.json();
1717
+
1718
+ if (data.success) {
1719
+ showRefreshToast(t('mcp.templateDeleted', { name: templateName }), 'success');
1720
+ await loadMcpTemplates();
1721
+ await renderMcpManager();
1722
+ } else {
1723
+ showRefreshToast(t('mcp.templateDeleteFailed', { error: data.error }), 'error');
1724
+ }
1725
+ } catch (error) {
1726
+ console.error('[MCP] Delete template error:', error);
1727
+ showRefreshToast(t('mcp.templateDeleteFailed', { error: error.message }), 'error');
1728
+ }
1729
+ }