cli-jaw 2.0.4 → 2.0.5

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 (624) hide show
  1. package/README.ja.md +17 -0
  2. package/README.ko.md +17 -0
  3. package/README.md +17 -0
  4. package/README.zh-CN.md +16 -1
  5. package/dist/bin/cli-jaw.js +7 -1
  6. package/dist/bin/cli-jaw.js.map +1 -1
  7. package/dist/bin/commands/browser.js +5 -0
  8. package/dist/bin/commands/browser.js.map +1 -1
  9. package/dist/bin/commands/connector.js +227 -0
  10. package/dist/bin/commands/connector.js.map +1 -0
  11. package/dist/bin/commands/dashboard-memory.js +243 -0
  12. package/dist/bin/commands/dashboard-memory.js.map +1 -0
  13. package/dist/bin/commands/dashboard.js +5 -0
  14. package/dist/bin/commands/dashboard.js.map +1 -1
  15. package/dist/bin/commands/dispatch-helpers.js +17 -0
  16. package/dist/bin/commands/dispatch-helpers.js.map +1 -0
  17. package/dist/bin/commands/dispatch.js +8 -3
  18. package/dist/bin/commands/dispatch.js.map +1 -1
  19. package/dist/bin/commands/doctor.js +34 -8
  20. package/dist/bin/commands/doctor.js.map +1 -1
  21. package/dist/bin/commands/reminders.js +105 -0
  22. package/dist/bin/commands/reminders.js.map +1 -0
  23. package/dist/bin/postinstall.js +26 -0
  24. package/dist/bin/postinstall.js.map +1 -1
  25. package/dist/server.js +71 -9
  26. package/dist/server.js.map +1 -1
  27. package/dist/src/agent/events.js +58 -30
  28. package/dist/src/agent/events.js.map +1 -1
  29. package/dist/src/agent/spawn.js +20 -3
  30. package/dist/src/agent/spawn.js.map +1 -1
  31. package/dist/src/agent/tool-timeout.js +27 -0
  32. package/dist/src/agent/tool-timeout.js.map +1 -0
  33. package/dist/src/agent/watchdog.js +20 -3
  34. package/dist/src/agent/watchdog.js.map +1 -1
  35. package/dist/src/browser/launch-policy.js +1 -1
  36. package/dist/src/browser/launch-policy.js.map +1 -1
  37. package/dist/src/browser/web-ai/chatgpt-model.js +102 -19
  38. package/dist/src/browser/web-ai/chatgpt-model.js.map +1 -1
  39. package/dist/src/browser/web-ai/chatgpt-response.js +30 -1
  40. package/dist/src/browser/web-ai/chatgpt-response.js.map +1 -1
  41. package/dist/src/browser/web-ai/chatgpt.js +55 -8
  42. package/dist/src/browser/web-ai/chatgpt.js.map +1 -1
  43. package/dist/src/browser/web-ai/diagnostics.js +1 -0
  44. package/dist/src/browser/web-ai/diagnostics.js.map +1 -1
  45. package/dist/src/browser/web-ai/gemini-live.js +24 -3
  46. package/dist/src/browser/web-ai/gemini-live.js.map +1 -1
  47. package/dist/src/browser/web-ai/interstitial.js +64 -0
  48. package/dist/src/browser/web-ai/interstitial.js.map +1 -0
  49. package/dist/src/browser/web-ai/session.js +12 -0
  50. package/dist/src/browser/web-ai/session.js.map +1 -1
  51. package/dist/src/browser/web-ai/watcher.js +25 -7
  52. package/dist/src/browser/web-ai/watcher.js.map +1 -1
  53. package/dist/src/cli/connector.js +60 -0
  54. package/dist/src/cli/connector.js.map +1 -0
  55. package/dist/src/cli/handlers-runtime.js +39 -0
  56. package/dist/src/cli/handlers-runtime.js.map +1 -1
  57. package/dist/src/cli/reminders.js +29 -0
  58. package/dist/src/cli/reminders.js.map +1 -0
  59. package/dist/src/core/cli-detect.js +161 -0
  60. package/dist/src/core/cli-detect.js.map +1 -0
  61. package/dist/src/core/config.js +9 -22
  62. package/dist/src/core/config.js.map +1 -1
  63. package/dist/src/core/db.js +14 -1
  64. package/dist/src/core/db.js.map +1 -1
  65. package/dist/src/core/settings-merge.js +2 -2
  66. package/dist/src/core/settings-merge.js.map +1 -1
  67. package/dist/src/jaw-ceo/completion.js +101 -0
  68. package/dist/src/jaw-ceo/completion.js.map +1 -0
  69. package/dist/src/jaw-ceo/confirmations.js +39 -0
  70. package/dist/src/jaw-ceo/confirmations.js.map +1 -0
  71. package/dist/src/jaw-ceo/coordinator-admin.js +295 -0
  72. package/dist/src/jaw-ceo/coordinator-admin.js.map +1 -0
  73. package/dist/src/jaw-ceo/coordinator-completions.js +201 -0
  74. package/dist/src/jaw-ceo/coordinator-completions.js.map +1 -0
  75. package/dist/src/jaw-ceo/coordinator-realtime-tools.js +108 -0
  76. package/dist/src/jaw-ceo/coordinator-realtime-tools.js.map +1 -0
  77. package/dist/src/jaw-ceo/coordinator-types.js +2 -0
  78. package/dist/src/jaw-ceo/coordinator-types.js.map +1 -0
  79. package/dist/src/jaw-ceo/coordinator-utils.js +86 -0
  80. package/dist/src/jaw-ceo/coordinator-utils.js.map +1 -0
  81. package/dist/src/jaw-ceo/coordinator-workers.js +234 -0
  82. package/dist/src/jaw-ceo/coordinator-workers.js.map +1 -0
  83. package/dist/src/jaw-ceo/coordinator.js +116 -0
  84. package/dist/src/jaw-ceo/coordinator.js.map +1 -0
  85. package/dist/src/jaw-ceo/docs-edit.js +142 -0
  86. package/dist/src/jaw-ceo/docs-edit.js.map +1 -0
  87. package/dist/src/jaw-ceo/openai-key.js +22 -0
  88. package/dist/src/jaw-ceo/openai-key.js.map +1 -0
  89. package/dist/src/jaw-ceo/policy.js +92 -0
  90. package/dist/src/jaw-ceo/policy.js.map +1 -0
  91. package/dist/src/jaw-ceo/realtime-sideband.js +372 -0
  92. package/dist/src/jaw-ceo/realtime-sideband.js.map +1 -0
  93. package/dist/src/jaw-ceo/store.js +206 -0
  94. package/dist/src/jaw-ceo/store.js.map +1 -0
  95. package/dist/src/jaw-ceo/transcript-persistence.js +28 -0
  96. package/dist/src/jaw-ceo/transcript-persistence.js.map +1 -0
  97. package/dist/src/jaw-ceo/types.js +2 -0
  98. package/dist/src/jaw-ceo/types.js.map +1 -0
  99. package/dist/src/manager/connector/audit-log.js +95 -0
  100. package/dist/src/manager/connector/audit-log.js.map +1 -0
  101. package/dist/src/manager/connector/routes.js +251 -0
  102. package/dist/src/manager/connector/routes.js.map +1 -0
  103. package/dist/src/manager/connector/types.js +2 -0
  104. package/dist/src/manager/connector/types.js.map +1 -0
  105. package/dist/src/manager/memory/embedding/hybrid-search.js +61 -0
  106. package/dist/src/manager/memory/embedding/hybrid-search.js.map +1 -0
  107. package/dist/src/manager/memory/embedding/index.js +6 -0
  108. package/dist/src/manager/memory/embedding/index.js.map +1 -0
  109. package/dist/src/manager/memory/embedding/provider.js +187 -0
  110. package/dist/src/manager/memory/embedding/provider.js.map +1 -0
  111. package/dist/src/manager/memory/embedding/state-machine.js +68 -0
  112. package/dist/src/manager/memory/embedding/state-machine.js.map +1 -0
  113. package/dist/src/manager/memory/embedding/sync.js +156 -0
  114. package/dist/src/manager/memory/embedding/sync.js.map +1 -0
  115. package/dist/src/manager/memory/embedding/vec-store.js +166 -0
  116. package/dist/src/manager/memory/embedding/vec-store.js.map +1 -0
  117. package/dist/src/manager/memory/federation.js +56 -0
  118. package/dist/src/manager/memory/federation.js.map +1 -0
  119. package/dist/src/manager/memory/index.js +5 -0
  120. package/dist/src/manager/memory/index.js.map +1 -0
  121. package/dist/src/manager/memory/instance-discovery.js +83 -0
  122. package/dist/src/manager/memory/instance-discovery.js.map +1 -0
  123. package/dist/src/manager/memory/result-rerank.js +28 -0
  124. package/dist/src/manager/memory/result-rerank.js.map +1 -0
  125. package/dist/src/manager/memory/types.js +2 -0
  126. package/dist/src/manager/memory/types.js.map +1 -0
  127. package/dist/src/manager/observability.js.map +1 -1
  128. package/dist/src/manager/process-verify.js +4 -1
  129. package/dist/src/manager/process-verify.js.map +1 -1
  130. package/dist/src/manager/registry.js +59 -1
  131. package/dist/src/manager/registry.js.map +1 -1
  132. package/dist/src/manager/reminders/api.js +16 -0
  133. package/dist/src/manager/reminders/api.js.map +1 -0
  134. package/dist/src/manager/reminders/dispatcher.js +40 -0
  135. package/dist/src/manager/reminders/dispatcher.js.map +1 -0
  136. package/dist/src/manager/reminders/due-time.js +16 -0
  137. package/dist/src/manager/reminders/due-time.js.map +1 -0
  138. package/dist/src/manager/reminders/instance-link.js +22 -0
  139. package/dist/src/manager/reminders/instance-link.js.map +1 -0
  140. package/dist/src/manager/reminders/routes.js +170 -0
  141. package/dist/src/manager/reminders/routes.js.map +1 -0
  142. package/dist/src/manager/reminders/scheduler.js +67 -0
  143. package/dist/src/manager/reminders/scheduler.js.map +1 -0
  144. package/dist/src/manager/reminders/store.js +311 -0
  145. package/dist/src/manager/reminders/store.js.map +1 -0
  146. package/dist/src/manager/routes/dashboard-memory.js +380 -0
  147. package/dist/src/manager/routes/dashboard-memory.js.map +1 -0
  148. package/dist/src/manager/server.js +209 -4
  149. package/dist/src/manager/server.js.map +1 -1
  150. package/dist/src/manager/worker-messages.js +60 -0
  151. package/dist/src/manager/worker-messages.js.map +1 -0
  152. package/dist/src/manager/workspace/routes.js +240 -0
  153. package/dist/src/manager/workspace/routes.js.map +1 -0
  154. package/dist/src/manager/workspace/store.js +369 -0
  155. package/dist/src/manager/workspace/store.js.map +1 -0
  156. package/dist/src/manager/workspace/types.js +5 -0
  157. package/dist/src/manager/workspace/types.js.map +1 -0
  158. package/dist/src/memory/indexing.js +38 -11
  159. package/dist/src/memory/indexing.js.map +1 -1
  160. package/dist/src/memory/shared.js +6 -0
  161. package/dist/src/memory/shared.js.map +1 -1
  162. package/dist/src/orchestrator/parser.js +3 -8
  163. package/dist/src/orchestrator/parser.js.map +1 -1
  164. package/dist/src/prompt/builder.js +35 -16
  165. package/dist/src/prompt/builder.js.map +1 -1
  166. package/dist/src/prompt/templates/a1-system.md +23 -1
  167. package/dist/src/reminders/jaw-reminders-bridge.js +313 -0
  168. package/dist/src/reminders/jaw-reminders-bridge.js.map +1 -0
  169. package/dist/src/reminders/types.js +7 -0
  170. package/dist/src/reminders/types.js.map +1 -0
  171. package/dist/src/routes/jaw-ceo.js +290 -0
  172. package/dist/src/routes/jaw-ceo.js.map +1 -0
  173. package/dist/src/routes/jaw-memory.js +37 -1
  174. package/dist/src/routes/jaw-memory.js.map +1 -1
  175. package/dist/src/routes/settings.js +47 -12
  176. package/dist/src/routes/settings.js.map +1 -1
  177. package/package.json +4 -1
  178. package/public/css/chat.css +90 -12
  179. package/public/css/diagram.css +9 -9
  180. package/public/css/modals.css +60 -1
  181. package/public/css/sidebar.css +18 -0
  182. package/public/dist/assets/{AdvancedExport-DJZ2VmBR.js → AdvancedExport-BAdZUC-6.js} +1 -1
  183. package/public/dist/assets/{Agent-CgpLT8IY.js → Agent-DNpehKB2.js} +1 -1
  184. package/public/dist/assets/{Browser-CrkiQoB8.js → Browser-BsdxDVgM.js} +1 -1
  185. package/public/dist/assets/{ChannelsDiscord-CVUC22D4.js → ChannelsDiscord-Dg_jto6l.js} +1 -1
  186. package/public/dist/assets/{ChannelsTelegram-DEatIQNM.js → ChannelsTelegram-DGyZfJGS.js} +1 -1
  187. package/public/dist/assets/DashboardEmbeddingSection-BYdGgqg7.js +2 -0
  188. package/public/dist/assets/{DashboardMeta-BKoxRc7i.js → DashboardMeta-CAH9ONTb.js} +1 -1
  189. package/public/dist/assets/{Display-DnNGV9Km.js → Display-DM_yvyKL.js} +1 -1
  190. package/public/dist/assets/{Employees-DZ2iJYKy.js → Employees-CuYuTy0R.js} +1 -1
  191. package/public/dist/assets/{HealthBadge-Cq2c7G9s.js → HealthBadge-Dtr-dDnw.js} +1 -1
  192. package/public/dist/assets/{Heartbeat-BML6eTXZ.js → Heartbeat-C-vq02MW.js} +1 -1
  193. package/public/dist/assets/{InlineWarn-BooBRm7o.js → InlineWarn-E64UaKFh.js} +1 -1
  194. package/public/dist/assets/{Mcp-BlEviQ3h.js → Mcp-Dlp2X7X7.js} +1 -1
  195. package/public/dist/assets/{Memory-BRyH80He.js → Memory-BPKWJDXK.js} +1 -1
  196. package/public/dist/assets/MilkdownWysiwygEditor-Ctww8i0L.js +160 -0
  197. package/public/dist/assets/{ModelProvider-DxyR7EL9.js → ModelProvider-Bd6vGkXT.js} +1 -1
  198. package/public/dist/assets/{Network-DDOOESh1.js → Network-Df1R2YcQ.js} +1 -1
  199. package/public/dist/assets/{Permissions-Br0eSbKb.js → Permissions-BKJ5K6EL.js} +1 -1
  200. package/public/dist/assets/{Permissions-QHkzStqQ.js → Permissions-CcWZoOVP.js} +1 -1
  201. package/public/dist/assets/{Profile-C79NKumk.js → Profile-DZ7xf1WZ.js} +1 -1
  202. package/public/dist/assets/{Prompts-BmiIDiXW.js → Prompts-Bh5DYt8e.js} +1 -1
  203. package/public/dist/assets/{SpeechKeys-B8304XJK.js → SpeechKeys-CQwtVxOP.js} +1 -1
  204. package/public/dist/assets/app-BxsIleo0.js +32 -0
  205. package/public/dist/assets/app-CB9n5A77.css +1 -0
  206. package/public/dist/assets/architecture-YZFGNWBL-BRI-0IaU.js +1 -0
  207. package/public/dist/assets/{architectureDiagram-Q4EWVU46-D0VjH4Hz.js → architectureDiagram-Q4EWVU46-z0JCgZrJ.js} +1 -1
  208. package/public/dist/assets/{blockDiagram-DXYQGD6D-CY4ROZPD.js → blockDiagram-DXYQGD6D-KaOz3aFS.js} +1 -1
  209. package/public/dist/assets/{c4Diagram-AHTNJAMY-CXSKBhA5.js → c4Diagram-AHTNJAMY-Bp2QLC-s.js} +1 -1
  210. package/public/dist/assets/channel-jLopKIgm.js +1 -0
  211. package/public/dist/assets/{chunk-2KRD3SAO-DKaML5Aw.js → chunk-2KRD3SAO-C8hJZPJu.js} +1 -1
  212. package/public/dist/assets/{chunk-336JU56O-CU3sZss9.js → chunk-336JU56O-CAt5aBOe.js} +1 -1
  213. package/public/dist/assets/chunk-426QAEUC-atJRhvKN.js +1 -0
  214. package/public/dist/assets/{chunk-4BX2VUAB-X0ZSlark.js → chunk-4BX2VUAB-Cm3KqBkS.js} +1 -1
  215. package/public/dist/assets/{chunk-4TB4RGXK-DxI0GIVy.js → chunk-4TB4RGXK-BD7UhIN8.js} +1 -1
  216. package/public/dist/assets/chunk-55IACEB6-CAWcPDdr.js +1 -0
  217. package/public/dist/assets/{chunk-5FUZZQ4R-BqlA0aLQ.js → chunk-5FUZZQ4R-Cg1pzs6p.js} +1 -1
  218. package/public/dist/assets/{chunk-5PVQY5BW-BVW1DG4s.js → chunk-5PVQY5BW-otv3HwE_.js} +1 -1
  219. package/public/dist/assets/{chunk-67CJDMHE-Blm2TZ-J.js → chunk-67CJDMHE-AU2iOKLT.js} +1 -1
  220. package/public/dist/assets/{chunk-7N4EOEYR-BoIOnnji.js → chunk-7N4EOEYR-BTIS4jBw.js} +1 -1
  221. package/public/dist/assets/{chunk-AA7GKIK3-Dsfi7oEW.js → chunk-AA7GKIK3-c9UUJO_T.js} +1 -1
  222. package/public/dist/assets/{chunk-BSJP7CBP-DMGz9IOn.js → chunk-BSJP7CBP-DnSYKjii.js} +1 -1
  223. package/public/dist/assets/{chunk-CIAEETIT-T5_zEp7h.js → chunk-CIAEETIT-Dwbln6rM.js} +1 -1
  224. package/public/dist/assets/{chunk-EDXVE4YY-DjfvQwok.js → chunk-EDXVE4YY-COudQKkJ.js} +1 -1
  225. package/public/dist/assets/{chunk-ENJZ2VHE-LawplD1x.js → chunk-ENJZ2VHE-PT_jAckw.js} +1 -1
  226. package/public/dist/assets/{chunk-FMBD7UC4-C7tckXtF.js → chunk-FMBD7UC4-6yXYicjw.js} +1 -1
  227. package/public/dist/assets/{chunk-FOC6F5B3-xPGbUzqY.js → chunk-FOC6F5B3-BqDmFFeg.js} +1 -1
  228. package/public/dist/assets/{chunk-ICPOFSXX-Cdp_Bt1o.js → chunk-ICPOFSXX-P6V6jqiT.js} +2 -2
  229. package/public/dist/assets/{chunk-K5T4RW27-BwIFil23.js → chunk-K5T4RW27-C42sOmZR.js} +1 -1
  230. package/public/dist/assets/{chunk-KGLVRYIC-BsNQ1OkZ.js → chunk-KGLVRYIC-K-BVa8b2.js} +1 -1
  231. package/public/dist/assets/{chunk-LIHQZDEY-DbaX7k3n.js → chunk-LIHQZDEY-Xw9ZX8of.js} +1 -1
  232. package/public/dist/assets/{chunk-ORNJ4GCN-C6D0ZHHr.js → chunk-ORNJ4GCN-CEXU6WaD.js} +1 -1
  233. package/public/dist/assets/{chunk-OYMX7WX6-C1WV9uUT.js → chunk-OYMX7WX6-DmJG2z1S.js} +1 -1
  234. package/public/dist/assets/chunk-QZHKN3VN-BkVLii_3.js +1 -0
  235. package/public/dist/assets/{chunk-U2HBQHQK-B-5yzlzA.js → chunk-U2HBQHQK-Ny6UwUgK.js} +1 -1
  236. package/public/dist/assets/{chunk-X2U36JSP-yL0Tpsl1.js → chunk-X2U36JSP-CoMFMVNF.js} +1 -1
  237. package/public/dist/assets/{chunk-XPW4576I-CeaUaSCU.js → chunk-XPW4576I-CmAz5C-j.js} +1 -1
  238. package/public/dist/assets/{chunk-YZCP3GAM-CASMOrQq.js → chunk-YZCP3GAM-c7FYtZME.js} +1 -1
  239. package/public/dist/assets/{chunk-ZZ45TVLE-B4c7TOr9.js → chunk-ZZ45TVLE-D5oxCC1O.js} +1 -1
  240. package/public/dist/assets/classDiagram-6PBFFD2Q-BgOyAynm.js +1 -0
  241. package/public/dist/assets/classDiagram-v2-HSJHXN6E-5gBdoVH9.js +1 -0
  242. package/public/dist/assets/{cose-bilkent-S5V4N54A-B21AcnwY.js → cose-bilkent-S5V4N54A-CvH6qY_r.js} +1 -1
  243. package/public/dist/assets/{dagre-DzTKkA3a.js → dagre--20B2-ZQ.js} +1 -1
  244. package/public/dist/assets/{dagre-KV5264BT-fAyzMxIJ.js → dagre-KV5264BT-CQ0wo0ma.js} +1 -1
  245. package/public/dist/assets/{diagram-5BDNPKRD-O30SB9_x.js → diagram-5BDNPKRD-Cl4PzCD0.js} +1 -1
  246. package/public/dist/assets/{diagram-G4DWMVQ6-DZ1S54sq.js → diagram-G4DWMVQ6-_aP6kMTh.js} +1 -1
  247. package/public/dist/assets/{diagram-MMDJMWI5-DRO_GSWW.js → diagram-MMDJMWI5-J1l6Q0JG.js} +1 -1
  248. package/public/dist/assets/{diagram-TYMM5635-BxWgQj5r.js → diagram-TYMM5635-B_Xoa1Gp.js} +1 -1
  249. package/public/dist/assets/{dist-W51oDoeA.js → dist-2oDfqE98.js} +1 -1
  250. package/public/dist/assets/{dist-BZosRD2u.js → dist-55MYVjjj.js} +1 -1
  251. package/public/dist/assets/dist-B1p80u1b.js +1 -0
  252. package/public/dist/assets/{dist-Bd9PlnQm.js → dist-B1rKu9eP.js} +1 -1
  253. package/public/dist/assets/{dist-Ch5VAlV9.js → dist-B6G8pbap.js} +1 -1
  254. package/public/dist/assets/{dist-CpUK8ypo.js → dist-BA7sRne4.js} +1 -1
  255. package/public/dist/assets/{dist-eU7TyC86.js → dist-BDMNMdPF.js} +1 -1
  256. package/public/dist/assets/{dist-DFYRUAjN.js → dist-BGzHP3f8.js} +1 -1
  257. package/public/dist/assets/dist-BKyzWv22.js +1 -0
  258. package/public/dist/assets/{dist-BNfXO3Yr.js → dist-BTp_Oy_x.js} +1 -1
  259. package/public/dist/assets/{dist-BD0UXfgF2.js → dist-BhzKO6nt2.js} +1 -1
  260. package/public/dist/assets/{dist-BUnPYbK3.js → dist-BjqyutOM.js} +1 -1
  261. package/public/dist/assets/dist-BoG5I6U5.js +1 -0
  262. package/public/dist/assets/{dist-yHP6L0Ty.js → dist-BuaQLcgQ.js} +1 -1
  263. package/public/dist/assets/{dist-DZsFVYF4.js → dist-C0sOT_UM.js} +5 -5
  264. package/public/dist/assets/{dist-CIuXW-sc.js → dist-C5S-Rbvc.js} +1 -1
  265. package/public/dist/assets/{dist-Db16ogVk.js → dist-C9LWf2uC.js} +1 -1
  266. package/public/dist/assets/{dist-urPTQzXL.js → dist-CCKktDoF.js} +1 -1
  267. package/public/dist/assets/{dist-D7bCuS3f.js → dist-CEDX2HGI.js} +1 -1
  268. package/public/dist/assets/dist-Ch6JG5jE.js +23 -0
  269. package/public/dist/assets/{dist-SU-YTAIg.js → dist-CkMC2PPt.js} +1 -1
  270. package/public/dist/assets/dist-CxJpXP6s.js +1 -0
  271. package/public/dist/assets/dist-DNLFuTrS.js +1 -0
  272. package/public/dist/assets/{dist-CL4PTYWf.js → dist-DdhWu7OM.js} +1 -1
  273. package/public/dist/assets/{dist-D6cUXP7K.js → dist-G7QUHtDS.js} +1 -1
  274. package/public/dist/assets/{dist-UYn7T-GH.js → dist-W7IGn2ug.js} +1 -1
  275. package/public/dist/assets/{dist-Cp42cMcI.js → dist-wdLr2dSH.js} +1 -1
  276. package/public/dist/assets/{dockerfile-zC7EsydA.js → dockerfile-CrC2HXHP.js} +1 -1
  277. package/public/dist/assets/employees-CIkIyvtL.js +33 -0
  278. package/public/dist/assets/{erDiagram-SMLLAGMA-oYlCN-2d.js → erDiagram-SMLLAGMA-BvtqhMsS.js} +1 -1
  279. package/public/dist/assets/{error-normalize-5n-zlEQ3.js → error-normalize-CkhPeQYx.js} +1 -1
  280. package/public/dist/assets/esm-BmJkIat2.js +4 -0
  281. package/public/dist/assets/{factor-DSufIRGh.js → factor-1CttFx2G.js} +1 -1
  282. package/public/dist/assets/{fields-BtncIZYA.js → fields-BISouxp2.js} +1 -1
  283. package/public/dist/assets/{flowDiagram-DWJPFMVM-er6Wfb34.js → flowDiagram-DWJPFMVM-C_F0rqqT.js} +1 -1
  284. package/public/dist/assets/{ganttDiagram-T4ZO3ILL-Co2j1qn3.js → ganttDiagram-T4ZO3ILL-B5S8EZRg.js} +1 -1
  285. package/public/dist/assets/gitGraph-7Q5UKJZL-DTTW6pxr.js +1 -0
  286. package/public/dist/assets/{gitGraphDiagram-UUTBAWPF-clYndUgV.js → gitGraphDiagram-UUTBAWPF-DMGzYJeb.js} +1 -1
  287. package/public/dist/assets/{graphlib-BjOif-SH.js → graphlib-CZk_Ii16.js} +1 -1
  288. package/public/dist/assets/idb-cache-BnZfG5FD.js +1 -0
  289. package/public/dist/assets/info-OMHHGYJF-BU-iuaSm.js +1 -0
  290. package/public/dist/assets/{infoDiagram-42DDH7IO-uYs7OoSA.js → infoDiagram-42DDH7IO-dzSaYFFK.js} +1 -1
  291. package/public/dist/assets/{ishikawaDiagram-UXIWVN3A-32bIEtCb.js → ishikawaDiagram-UXIWVN3A-DZW-Nqsf.js} +1 -1
  292. package/public/dist/assets/javascript-BhB45e0W.js +1 -0
  293. package/public/dist/assets/{journeyDiagram-VCZTEJTY-hBNsU7mK.js → journeyDiagram-VCZTEJTY-ByVRJxVF.js} +1 -1
  294. package/public/dist/assets/{kanban-definition-6JOO6SKY-DoepMHBI.js → kanban-definition-6JOO6SKY-xwN0YCW2.js} +1 -1
  295. package/public/dist/assets/katex-DamPUmTE.js +1 -0
  296. package/public/dist/assets/manager-B84u-pcn.js +25 -0
  297. package/public/dist/assets/manager-DMg_sTEP.css +1 -0
  298. package/public/dist/assets/memory-BV62wlsG.js +1 -0
  299. package/public/dist/assets/{memory-dJGp6QBv.js → memory-CnBc2_Va.js} +1 -1
  300. package/public/dist/assets/mermaid-loader-DxFGz4EE.js +1 -0
  301. package/public/dist/assets/{mermaid-parser.core-CrXy-45O.js → mermaid-parser.core-CMIZ0my_.js} +1 -1
  302. package/public/dist/assets/mermaid-preprocess-p7OIpOon.js +5 -0
  303. package/public/dist/assets/mermaid.core-CicVBD9l.js +1 -0
  304. package/public/dist/assets/{mermaid.core-DYApH8sc.js → mermaid.core-Dj6viEzN.js} +2 -2
  305. package/public/dist/assets/{mindmap-definition-QFDTVHPH-C-QWWjaf.js → mindmap-definition-QFDTVHPH-Chqgsh1N.js} +1 -1
  306. package/public/dist/assets/{nsis-DLskXEqR.js → nsis-_CjIiUyF.js} +1 -1
  307. package/public/dist/assets/packet-4T2RLAQJ-DzmBsAf0.js +1 -0
  308. package/public/dist/assets/{page-shell-D5tbivHH.js → page-shell-CTxVDJ8Y.js} +1 -1
  309. package/public/dist/assets/pie-ZZUOXDRM-DyG1KzTn.js +1 -0
  310. package/public/dist/assets/{pieDiagram-DEJITSTG-DvMWxH5R.js → pieDiagram-DEJITSTG-DJLiSymi.js} +1 -1
  311. package/public/dist/assets/{pug-B2rz0Rz1.js → pug-CbR8lCtK.js} +1 -1
  312. package/public/dist/assets/{quadrantDiagram-34T5L4WZ-DjoQDyzO.js → quadrantDiagram-34T5L4WZ-qcnWmcRr.js} +1 -1
  313. package/public/dist/assets/radar-PYXPWWZC-CEBZARUu.js +1 -0
  314. package/public/dist/assets/render-lpTpN1El.js +28 -0
  315. package/public/dist/assets/{requirementDiagram-MS252O5E-sQav3xIA.js → requirementDiagram-MS252O5E-wnPtv0LG.js} +1 -1
  316. package/public/dist/assets/{sankeyDiagram-XADWPNL6-DkdEFMeo.js → sankeyDiagram-XADWPNL6-Bn3pUd_K.js} +1 -1
  317. package/public/dist/assets/{sequenceDiagram-FGHM5R23-kaplo282.js → sequenceDiagram-FGHM5R23-DfxcHSdj.js} +1 -1
  318. package/public/dist/assets/settings-CfAXLa8a.js +1 -0
  319. package/public/dist/assets/{settings-C7QWaUHB.js → settings-FWgmcubl.js} +8 -8
  320. package/public/dist/assets/sidebar-Cwt0FxQl.js +14 -0
  321. package/public/dist/assets/skills-D-qUVJ-e.js +1 -0
  322. package/public/dist/assets/{skills-CHkTgM7L.js → skills-DDmTyywh.js} +2 -2
  323. package/public/dist/assets/slash-commands-BfI19vIz.js +1 -0
  324. package/public/dist/assets/{slash-commands-2ThyUGvX.js → slash-commands-D5y5AVvS.js} +1 -1
  325. package/public/dist/assets/{stateDiagram-FHFEXIEX-B2L3h2aE.js → stateDiagram-FHFEXIEX-CiR7P2OG.js} +1 -1
  326. package/public/dist/assets/stateDiagram-v2-QKLJ7IA2-BaWWpysK.js +1 -0
  327. package/public/dist/assets/{timeline-definition-GMOUNBTQ-_Nnp2LU-.js → timeline-definition-GMOUNBTQ-DUz1-iOx.js} +1 -1
  328. package/public/dist/assets/{trace-drawer-Dis80M6X.js → trace-drawer-DVF5F1df.js} +1 -1
  329. package/public/dist/assets/treeView-SZITEDCU-D5-TP7Qp.js +1 -0
  330. package/public/dist/assets/treemap-W4RFUUIX-BQhm_ZSp.js +1 -0
  331. package/public/dist/assets/ui-BJXY16Dk.js +140 -0
  332. package/public/dist/assets/ui-BXm3OPPh.js +1 -0
  333. package/public/dist/assets/{vendor-render-984tPmpy.js → vendor-render-DEStnpJ8.js} +22 -22
  334. package/public/dist/assets/{vennDiagram-DHZGUBPP-vNU8EDoW.js → vennDiagram-DHZGUBPP-CYu7SDdG.js} +1 -1
  335. package/public/dist/assets/wardley-RL74JXVD-CpnBMRe0.js +1 -0
  336. package/public/dist/assets/{wardleyDiagram-NUSXRM2D-DB2wK0zr.js → wardleyDiagram-NUSXRM2D-CLAhOzap.js} +1 -1
  337. package/public/dist/assets/wiki-link-suggestions-LQuYT22G.js +23 -0
  338. package/public/dist/assets/{xychartDiagram-5P7HB3ND-CisgDiJM.js → xychartDiagram-5P7HB3ND-DLnY9i8z.js} +1 -1
  339. package/public/dist/index.html +36 -12
  340. package/public/dist/manager/index.html +2 -2
  341. package/public/index.html +34 -10
  342. package/public/js/features/chat-messages.ts +132 -0
  343. package/public/js/features/chat-scroll.ts +226 -0
  344. package/public/js/features/employees.ts +4 -1
  345. package/public/js/features/help-content.ts +11 -1
  346. package/public/js/features/help-dialog.ts +78 -17
  347. package/public/js/features/message-actions.ts +157 -0
  348. package/public/js/features/message-history.ts +138 -0
  349. package/public/js/features/message-item-html.ts +34 -0
  350. package/public/js/features/preview-shortcut-bridge.ts +29 -0
  351. package/public/js/features/process-block-dom.ts +175 -0
  352. package/public/js/features/process-log-adapter.ts +104 -0
  353. package/public/js/features/settings-cli-status.ts +52 -1
  354. package/public/js/features/settings.ts +1 -0
  355. package/public/js/features/sidebar.ts +7 -1
  356. package/public/js/features/ui-status.ts +47 -0
  357. package/public/js/main.ts +10 -3
  358. package/public/js/render/code-copy.ts +47 -0
  359. package/public/js/render/delegations.ts +10 -0
  360. package/public/js/render/file-links.ts +150 -0
  361. package/public/js/render/highlight.ts +83 -0
  362. package/public/js/render/html.ts +26 -0
  363. package/public/js/render/markdown.ts +84 -0
  364. package/public/js/render/math.ts +76 -0
  365. package/public/js/render/mermaid-preprocess.ts +77 -0
  366. package/public/js/render/mermaid.ts +267 -0
  367. package/public/js/render/post-render.ts +35 -0
  368. package/public/js/render/sanitize.ts +51 -0
  369. package/public/js/render/svg-actions.ts +291 -0
  370. package/public/js/render.ts +17 -1081
  371. package/public/js/ui.ts +23 -841
  372. package/public/js/virtual-scroll-bootstrap.ts +18 -5
  373. package/public/js/virtual-scroll.ts +17 -0
  374. package/public/js/ws.ts +5 -0
  375. package/public/locales/en.json +76 -1
  376. package/public/locales/ja.json +74 -1
  377. package/public/locales/ko.json +76 -1
  378. package/public/locales/zh.json +74 -1
  379. package/public/manager/src/App.tsx +143 -258
  380. package/public/manager/src/AppChrome.tsx +144 -0
  381. package/public/manager/src/InstancePreview.tsx +18 -0
  382. package/public/manager/src/SidebarRailRouter.tsx +229 -0
  383. package/public/manager/src/components/ActivityTimeline.tsx +9 -1
  384. package/public/manager/src/components/ProcessControlPanel.tsx +3 -0
  385. package/public/manager/src/components/SidebarRail.tsx +22 -0
  386. package/public/manager/src/components/Workbench.tsx +17 -13
  387. package/public/manager/src/components/WorkbenchHeader.tsx +44 -33
  388. package/public/manager/src/components/WorkspaceLayout.tsx +3 -0
  389. package/public/manager/src/dashboard-board/DashboardBoardSidebar.tsx +1 -1
  390. package/public/manager/src/dashboard-board/DashboardBoardWorkspace.tsx +6 -0
  391. package/public/manager/src/dashboard-features.ts +4 -1
  392. package/public/manager/src/dashboard-reminders/DashboardRemindersSidebar.tsx +157 -0
  393. package/public/manager/src/dashboard-reminders/DashboardRemindersWorkspace.tsx +502 -0
  394. package/public/manager/src/dashboard-reminders/InlineReminderTitle.tsx +78 -0
  395. package/public/manager/src/dashboard-reminders/ReminderDetailPopover.tsx +118 -0
  396. package/public/manager/src/dashboard-reminders/reminder-order.ts +45 -0
  397. package/public/manager/src/dashboard-reminders/reminders-api.ts +126 -0
  398. package/public/manager/src/dashboard-reminders/reminders-view-model.ts +63 -0
  399. package/public/manager/src/dashboard-reminders/useDashboardReminderDrag.ts +55 -0
  400. package/public/manager/src/dashboard-reminders/useRemindersFeed.ts +72 -0
  401. package/public/manager/src/dashboard-schedule/DashboardScheduleWorkspace.tsx +10 -4
  402. package/public/manager/src/dashboard-settings/DashboardEmbeddingSection.tsx +350 -0
  403. package/public/manager/src/dashboard-settings/DashboardSettingsSidebar.tsx +6 -2
  404. package/public/manager/src/dashboard-settings/DashboardSettingsWorkspace.tsx +217 -6
  405. package/public/manager/src/dashboard-settings/dashboard-settings-ui.ts +2 -0
  406. package/public/manager/src/dashboard-url-state.ts +15 -0
  407. package/public/manager/src/help/HelpDrawer.tsx +5 -6
  408. package/public/manager/src/help/HelpTopicButton.tsx +23 -0
  409. package/public/manager/src/help/help-shortcuts.ts +7 -0
  410. package/public/manager/src/help/helpContent.tsx +155 -5
  411. package/public/manager/src/hooks/useDashboardView.ts +12 -1
  412. package/public/manager/src/hooks/useInstanceMessageEvents.ts +3 -1
  413. package/public/manager/src/jaw-ceo/JawCeoConsole.tsx +108 -0
  414. package/public/manager/src/jaw-ceo/JawCeoConsolePanels.tsx +331 -0
  415. package/public/manager/src/jaw-ceo/JawCeoSettingsPanel.tsx +94 -0
  416. package/public/manager/src/jaw-ceo/JawCeoTabs.tsx +28 -0
  417. package/public/manager/src/jaw-ceo/JawCeoVoiceOverlay.tsx +34 -0
  418. package/public/manager/src/jaw-ceo/JawCeoWorkbenchButton.tsx +61 -0
  419. package/public/manager/src/jaw-ceo/api.ts +169 -0
  420. package/public/manager/src/jaw-ceo/jaw-ceo-console.css +481 -0
  421. package/public/manager/src/jaw-ceo/jaw-ceo-virtual.css +24 -0
  422. package/public/manager/src/jaw-ceo/jaw-ceo.css +429 -0
  423. package/public/manager/src/jaw-ceo/types.ts +164 -0
  424. package/public/manager/src/jaw-ceo/useJawCeo.ts +235 -0
  425. package/public/manager/src/jaw-ceo/useJawCeoConsoleModel.ts +67 -0
  426. package/public/manager/src/jaw-ceo/useJawCeoDashboardBridge.tsx +76 -0
  427. package/public/manager/src/jaw-ceo/useJawCeoVirtualTimeline.ts +109 -0
  428. package/public/manager/src/jaw-ceo/useJawCeoVoice.ts +219 -0
  429. package/public/manager/src/jaw-ceo/voice-cues.ts +34 -0
  430. package/public/manager/src/jaw-ceo/voice-session.ts +87 -0
  431. package/public/manager/src/main.tsx +10 -0
  432. package/public/manager/src/manager-components.css +7 -0
  433. package/public/manager/src/manager-dashboard-board-sidebar-scroll.css +64 -0
  434. package/public/manager/src/manager-dashboard-reminders-parity.css +156 -0
  435. package/public/manager/src/manager-dashboard-reminders-priority.css +164 -0
  436. package/public/manager/src/manager-dashboard-reminders.css +482 -0
  437. package/public/manager/src/manager-dashboard-schedule.css +12 -0
  438. package/public/manager/src/manager-dashboard-settings.css +16 -2
  439. package/public/manager/src/manager-help.css +27 -0
  440. package/public/manager/src/manager-layout.css +19 -0
  441. package/public/manager/src/manager-notes.css +13 -4
  442. package/public/manager/src/manager-p0-1-1.css +3 -3
  443. package/public/manager/src/manager-polish.css +7 -3
  444. package/public/manager/src/manager-shortcuts.ts +116 -0
  445. package/public/manager/src/notes/MarkdownEditor.tsx +20 -3
  446. package/public/manager/src/notes/MarkdownPreview.tsx +10 -1
  447. package/public/manager/src/notes/NotesFrontmatterStrip.tsx +61 -0
  448. package/public/manager/src/notes/NotesQuickSwitcher.tsx +187 -0
  449. package/public/manager/src/notes/NotesSidebar.tsx +36 -14
  450. package/public/manager/src/notes/NotesWorkspace.tsx +63 -4
  451. package/public/manager/src/notes/frontmatter-preview.ts +16 -0
  452. package/public/manager/src/notes/notes-quick-switcher.css +151 -0
  453. package/public/manager/src/notes/notes-tags.css +292 -0
  454. package/public/manager/src/notes/notes-types.ts +4 -0
  455. package/public/manager/src/notes/rendering/MarkdownRenderer.tsx +67 -3
  456. package/public/manager/src/notes/rendering/MermaidBlock.tsx +11 -1
  457. package/public/manager/src/notes/useNoteDocument.ts +14 -2
  458. package/public/manager/src/notes/useNotesModel.ts +39 -2
  459. package/public/manager/src/notes/wiki-link-codemirror-completion.ts +55 -0
  460. package/public/manager/src/notes/wiki-link-rendering.ts +129 -0
  461. package/public/manager/src/notes/wiki-link-resolver.ts +220 -0
  462. package/public/manager/src/notes/wiki-link-suggestions.ts +153 -0
  463. package/public/manager/src/notes/wysiwyg/MilkdownWysiwygEditor.tsx +126 -79
  464. package/public/manager/src/notes/wysiwyg/WysiwygFrontmatterPanel.tsx +93 -0
  465. package/public/manager/src/notes/wysiwyg/milkdown-code-block-view.ts +11 -0
  466. package/public/manager/src/notes/wysiwyg/milkdown-editor-utils.ts +32 -0
  467. package/public/manager/src/notes/wysiwyg/milkdown-math.ts +9 -0
  468. package/public/manager/src/notes/wysiwyg/milkdown-wikilink-completion.ts +195 -0
  469. package/public/manager/src/notes/wysiwyg/milkdown-wikilink-plugin.ts +152 -0
  470. package/public/manager/src/notes/wysiwyg/milkdown-wysiwyg-types.ts +13 -0
  471. package/public/manager/src/notes/wysiwyg/wysiwyg-fixtures.ts +4 -0
  472. package/public/manager/src/notes/wysiwyg/wysiwyg-frontmatter.ts +155 -0
  473. package/public/manager/src/settings/pages/Embedding.tsx +382 -0
  474. package/public/manager/src/settings-embedding.css +168 -0
  475. package/public/manager/src/types.ts +12 -2
  476. package/scripts/install-officecli.sh +3 -1
  477. package/scripts/install-wsl.sh +65 -11
  478. package/public/dist/assets/MilkdownWysiwygEditor-DIebNZF7.js +0 -52
  479. package/public/dist/assets/app-CphocJzo.css +0 -1
  480. package/public/dist/assets/app-DJ8ys0j5.js +0 -32
  481. package/public/dist/assets/architecture-YZFGNWBL-VpdnYwVC.js +0 -1
  482. package/public/dist/assets/channel-xhp7drfS.js +0 -1
  483. package/public/dist/assets/chunk-426QAEUC-DadS7Aoc.js +0 -1
  484. package/public/dist/assets/chunk-55IACEB6-Cr281urY.js +0 -1
  485. package/public/dist/assets/chunk-QZHKN3VN-BYRU3VjK.js +0 -1
  486. package/public/dist/assets/classDiagram-6PBFFD2Q-Brk35lpl.js +0 -1
  487. package/public/dist/assets/classDiagram-v2-HSJHXN6E-CVZHK0Hg.js +0 -1
  488. package/public/dist/assets/dist-BsT5UdNP.js +0 -1
  489. package/public/dist/assets/dist-ClqO40BE.js +0 -1
  490. package/public/dist/assets/dist-CxeLAw2Y.js +0 -1
  491. package/public/dist/assets/dist-D2SH8nxa.js +0 -1
  492. package/public/dist/assets/dist-DfodGES_.js +0 -23
  493. package/public/dist/assets/dist-l9HH00ip.js +0 -1
  494. package/public/dist/assets/employees-RJ_wRL09.js +0 -33
  495. package/public/dist/assets/gitGraph-7Q5UKJZL-CBKRRUZJ.js +0 -1
  496. package/public/dist/assets/idb-cache-C5ilDI6r.js +0 -1
  497. package/public/dist/assets/info-OMHHGYJF-BqzRlF_E.js +0 -1
  498. package/public/dist/assets/insert-image-markdown-kk053MvN.js +0 -22
  499. package/public/dist/assets/javascript-DT-6B2IU.js +0 -1
  500. package/public/dist/assets/katex-CS4w7U2j.js +0 -1
  501. package/public/dist/assets/manager-DAe38I94.js +0 -25
  502. package/public/dist/assets/manager-fQR46YFa.css +0 -1
  503. package/public/dist/assets/memory-w3yQettQ.js +0 -1
  504. package/public/dist/assets/mermaid-loader-BEFIOoJn.js +0 -1
  505. package/public/dist/assets/mermaid.core-D0tXHgVC.js +0 -1
  506. package/public/dist/assets/packet-4T2RLAQJ-D7VqJs5z.js +0 -1
  507. package/public/dist/assets/pie-ZZUOXDRM-FbISwuBg.js +0 -1
  508. package/public/dist/assets/radar-PYXPWWZC-CLzM3m4R.js +0 -1
  509. package/public/dist/assets/render-KVGsbWj1.js +0 -31
  510. package/public/dist/assets/settings-DmUCo6lz.js +0 -1
  511. package/public/dist/assets/skills-SxG_nfwn.js +0 -1
  512. package/public/dist/assets/slash-commands-BxJkKdhB.js +0 -1
  513. package/public/dist/assets/stateDiagram-v2-QKLJ7IA2-DScAhoyh.js +0 -1
  514. package/public/dist/assets/treeView-SZITEDCU-HJajkeQK.js +0 -1
  515. package/public/dist/assets/treemap-W4RFUUIX-CsRQ896G.js +0 -1
  516. package/public/dist/assets/ui-LhD1VfQs.js +0 -1
  517. package/public/dist/assets/ui-kS1ZJfez.js +0 -143
  518. package/public/dist/assets/wardley-RL74JXVD-DMQFnlJp.js +0 -1
  519. package/public/dist/assets/ws-DVE3eWRj.js +0 -14
  520. /package/public/dist/assets/{agent-meta-DHddpWHQ.js → agent-meta-C1pQzExf.js} +0 -0
  521. /package/public/dist/assets/{apl-1H3fJ-n9.js → apl-HYRstREL.js} +0 -0
  522. /package/public/dist/assets/{asciiarmor-_1f4lqIZ.js → asciiarmor-BDXCrhAK.js} +0 -0
  523. /package/public/dist/assets/{asn1-B_wp1Pdt.js → asn1-CnSBhb0M.js} +0 -0
  524. /package/public/dist/assets/{asterisk-9Hxmlnbz.js → asterisk-AqV7rnIi.js} +0 -0
  525. /package/public/dist/assets/{brainfuck-C-PAVNT8.js → brainfuck-Dv46-iL9.js} +0 -0
  526. /package/public/dist/assets/{chunk-AGHRB4JF-EroDCcEL.js → chunk-AGHRB4JF-DIMn8T3_.js} +0 -0
  527. /package/public/dist/assets/{clike-DAVXDhrz.js → clike-GRffz5hY.js} +0 -0
  528. /package/public/dist/assets/{clojure-BIpQu4v4.js → clojure-DzgT_fqE.js} +0 -0
  529. /package/public/dist/assets/{cmake-YlOBEvgc.js → cmake-Co1237r5.js} +0 -0
  530. /package/public/dist/assets/{cobol-CU8lw6mH.js → cobol-rUaLketb.js} +0 -0
  531. /package/public/dist/assets/{coffeescript-DKG8mABL.js → coffeescript-DQyfHE86.js} +0 -0
  532. /package/public/dist/assets/{commonlisp-CbWboHR_.js → commonlisp-Buue1PGW.js} +0 -0
  533. /package/public/dist/assets/{constants-CYjOMbjf.js → constants-4A2GptQT.js} +0 -0
  534. /package/public/dist/assets/{crystal-WWs3Njra.js → crystal-BMWO0kOJ.js} +0 -0
  535. /package/public/dist/assets/{css-C9qNtc8P.js → css-DaxibPo5.js} +0 -0
  536. /package/public/dist/assets/{cypher-DglnWVZO.js → cypher-CmpGfiBR.js} +0 -0
  537. /package/public/dist/assets/{cytoscape.esm-BH6rhG9A.js → cytoscape.esm-zT8GwLFm.js} +0 -0
  538. /package/public/dist/assets/{d-DiNW6jcC.js → d-qmdtoIzU.js} +0 -0
  539. /package/public/dist/assets/{diff-PxsoX0eK.js → diff-B40m6u1h.js} +0 -0
  540. /package/public/dist/assets/{dist-BRFvH5ne.js → dist-BJyDhGpS.js} +0 -0
  541. /package/public/dist/assets/{dtd-BSiaE0hE.js → dtd-569ynYVj.js} +0 -0
  542. /package/public/dist/assets/{dylan-GcUJt0a5.js → dylan-Dxolp30i.js} +0 -0
  543. /package/public/dist/assets/{ebnf-DSIN9Ycp.js → ebnf-BZiX9Iq3.js} +0 -0
  544. /package/public/dist/assets/{ecl-LP9l5g0J.js → ecl-B8F1Q0oZ.js} +0 -0
  545. /package/public/dist/assets/{eiffel-CqpH5qaY.js → eiffel-Bv8Kvgh1.js} +0 -0
  546. /package/public/dist/assets/{elm-Dw3cJ7hz.js → elm-QWXQaIis.js} +0 -0
  547. /package/public/dist/assets/{erlang-DL9rpJep.js → erlang-CuDGzTGm.js} +0 -0
  548. /package/public/dist/assets/{fcl-DUmfTiJ7.js → fcl-BiFeqtHf.js} +0 -0
  549. /package/public/dist/assets/{forth-BSp-YOKj.js → forth-R7Uc2VcL.js} +0 -0
  550. /package/public/dist/assets/{fortran-BtIRmo6s.js → fortran-kDRG6BzW.js} +0 -0
  551. /package/public/dist/assets/{gas-okqIRy8I.js → gas-CyRkuC5T.js} +0 -0
  552. /package/public/dist/assets/{gherkin-DIlSAX9i.js → gherkin-BE0p00ey.js} +0 -0
  553. /package/public/dist/assets/{groovy-Dn6VQ7le.js → groovy-B_Sh3D1t.js} +0 -0
  554. /package/public/dist/assets/{haskell-DxXJRRAf.js → haskell-DWXgCy4o.js} +0 -0
  555. /package/public/dist/assets/{haxe-Cf0nItU4.js → haxe-xGxZ54Hv.js} +0 -0
  556. /package/public/dist/assets/{http-C3zQeWFV.js → http-CKv9cSBA.js} +0 -0
  557. /package/public/dist/assets/{idb-cache-CjnC_K0x.js → idb-cache-CZ3JdK8r.js} +0 -0
  558. /package/public/dist/assets/{idl-BvqkcZq5.js → idl-DmlI3XzS.js} +0 -0
  559. /package/public/dist/assets/{javascript-D1nkUbtD.js → javascript-j6r5uf_k.js} +0 -0
  560. /package/public/dist/assets/{jsx-runtime-DOs-BiPY.js → jsx-runtime-BjL7qHh8.js} +0 -0
  561. /package/public/dist/assets/{julia-CCUCEuMJ.js → julia-CQSp9Qa0.js} +0 -0
  562. /package/public/dist/assets/{livescript-DJ-6Zja2.js → livescript-CqvtVTlb.js} +0 -0
  563. /package/public/dist/assets/{locale-f397XJgr.js → locale-BHMJIzyw.js} +0 -0
  564. /package/public/dist/assets/{lua-CxE7YJaJ.js → lua-DazQKUZ0.js} +0 -0
  565. /package/public/dist/assets/{mathematica-oNEp3xtK.js → mathematica-DqhJVCs9.js} +0 -0
  566. /package/public/dist/assets/{mbox-AvjHxuWl.js → mbox-Dsyo1_UL.js} +0 -0
  567. /package/public/dist/assets/{mirc-CiJGiXcS.js → mirc-D7ThhoF-.js} +0 -0
  568. /package/public/dist/assets/{mllike-CV8oQtvy.js → mllike-EPZ6pqQD.js} +0 -0
  569. /package/public/dist/assets/{modelica-M4BAagfL.js → modelica-PU45hbqg.js} +0 -0
  570. /package/public/dist/assets/{mscgen-3QXoaqEl.js → mscgen-DEnh2Ojr.js} +0 -0
  571. /package/public/dist/assets/{mumps-BZoVj2VG.js → mumps-8GhR7rRa.js} +0 -0
  572. /package/public/dist/assets/{nginx-CBycrar_.js → nginx-CXwagpwp.js} +0 -0
  573. /package/public/dist/assets/{ntriples-BzBBWYSl.js → ntriples-BDQxxstw.js} +0 -0
  574. /package/public/dist/assets/{octave-BeIsGSMy.js → octave-2c90WnyU.js} +0 -0
  575. /package/public/dist/assets/{oz-DdNR63iD.js → oz-CO0rQ8EL.js} +0 -0
  576. /package/public/dist/assets/{pascal-BvmM4Y7z.js → pascal-CcVwOGeN.js} +0 -0
  577. /package/public/dist/assets/{path-utils-BuEEtj9w.js → path-utils-DySmCVZK.js} +0 -0
  578. /package/public/dist/assets/{perl-DFibaVbf.js → perl-B9UvGoAV.js} +0 -0
  579. /package/public/dist/assets/{pig-DyLcrln-.js → pig-Cd1f86ZJ.js} +0 -0
  580. /package/public/dist/assets/{powershell-CDOpEwhN.js → powershell-DV4cOnkJ.js} +0 -0
  581. /package/public/dist/assets/{preload-helper-CMqhPRCH.js → preload-helper-DuH3-WbD.js} +0 -0
  582. /package/public/dist/assets/{properties-Ju2yMwi3.js → properties-DaSVPhrZ.js} +0 -0
  583. /package/public/dist/assets/{protobuf-CfKx2SfO.js → protobuf-CEtFnY22.js} +0 -0
  584. /package/public/dist/assets/{puppet-dmjzqANu.js → puppet-Ca0DHfcW.js} +0 -0
  585. /package/public/dist/assets/{python-D9BZyYi1.js → python-ydzCwWG-.js} +0 -0
  586. /package/public/dist/assets/{q-C68SC4ma.js → q-Cl8kzQmk.js} +0 -0
  587. /package/public/dist/assets/{r-DP6KYC7N.js → r-DKZPJFrD.js} +0 -0
  588. /package/public/dist/assets/{rough.esm-BRGGwqOp.js → rough.esm-Dnk3WN4D.js} +0 -0
  589. /package/public/dist/assets/{rpm-DH2Z4z_v.js → rpm-D0lqeHoZ.js} +0 -0
  590. /package/public/dist/assets/{ruby-CwiqsW2a.js → ruby-RVmtmNDw.js} +0 -0
  591. /package/public/dist/assets/{sas-Btq5K1IU.js → sas-CEzodAB7.js} +0 -0
  592. /package/public/dist/assets/{scheme-BvG_RNGf.js → scheme-Pqmt6dh3.js} +0 -0
  593. /package/public/dist/assets/{settings-client-ajlwI-oK.js → settings-client-BN3XzwEo.js} +0 -0
  594. /package/public/dist/assets/{shell-waWIPbLL.js → shell-CLG3zy-u.js} +0 -0
  595. /package/public/dist/assets/{sieve-wIycBENd.js → sieve-C95IWC3O.js} +0 -0
  596. /package/public/dist/assets/{simple-mode-lHvrAolQ.js → simple-mode-B9ErAHMD.js} +0 -0
  597. /package/public/dist/assets/{smalltalk-SHoYQOtw.js → smalltalk-4cL-gRJb.js} +0 -0
  598. /package/public/dist/assets/{solr-D1Apesak.js → solr-CEA7oOx_.js} +0 -0
  599. /package/public/dist/assets/{sparql-BJozOrmN.js → sparql-w8kHjP6I.js} +0 -0
  600. /package/public/dist/assets/{spreadsheet-71jLW2fq.js → spreadsheet-C8bDfTgC.js} +0 -0
  601. /package/public/dist/assets/{sql-BaHU6gLz.js → sql-BrKz8968.js} +0 -0
  602. /package/public/dist/assets/{stex-BuVGLRtt.js → stex-xFw1nXeT.js} +0 -0
  603. /package/public/dist/assets/{stylus-Cqweyusm.js → stylus-gS_68vPk.js} +0 -0
  604. /package/public/dist/assets/{swift-Dx3m4XfV.js → swift-D5lXmY2d.js} +0 -0
  605. /package/public/dist/assets/{tcl-5yXjCZ8e.js → tcl-D-41REtC.js} +0 -0
  606. /package/public/dist/assets/{textile-9HVfsYV6.js → textile-D-Znao0j.js} +0 -0
  607. /package/public/dist/assets/{tiddlywiki-DlSf8L__.js → tiddlywiki-Co8lXJKd.js} +0 -0
  608. /package/public/dist/assets/{tiki-CDZRH0A4.js → tiki-DsiNRYMP.js} +0 -0
  609. /package/public/dist/assets/{toml-BgjfE0Y7.js → toml-CaVZGou4.js} +0 -0
  610. /package/public/dist/assets/{troff-B4ZrJIrZ.js → troff-DJvDVr-d.js} +0 -0
  611. /package/public/dist/assets/{ttcn-BJIiaYhP.js → ttcn-DAC92l4d.js} +0 -0
  612. /package/public/dist/assets/{ttcn-cfg-PZp_QKQg.js → ttcn-cfg-C0X-mYlr.js} +0 -0
  613. /package/public/dist/assets/{turtle--ft5y7IN.js → turtle-D6WHyCrL.js} +0 -0
  614. /package/public/dist/assets/{vb-DL6cecZs.js → vb-CEo_ccq3.js} +0 -0
  615. /package/public/dist/assets/{vbscript-Bb1xiv7E.js → vbscript-B_PKDV1v.js} +0 -0
  616. /package/public/dist/assets/{velocity-DKS9ZfbJ.js → velocity-ClKnKBLH.js} +0 -0
  617. /package/public/dist/assets/{vendor-utils-CWelFt6C.js → vendor-utils-IVTPs69f.js} +0 -0
  618. /package/public/dist/assets/{verilog-DSfhYFaT.js → verilog-Pnuspgw6.js} +0 -0
  619. /package/public/dist/assets/{vhdl-CFTSUmh9.js → vhdl-CZnyY4fs.js} +0 -0
  620. /package/public/dist/assets/{w3c-keyname-IiiZScED.js → w3c-keyname-DJlC6DTY.js} +0 -0
  621. /package/public/dist/assets/{webidl-C-IpKmuN.js → webidl-DHr762Dm.js} +0 -0
  622. /package/public/dist/assets/{xquery-9pXYEith.js → xquery-5-FiyUtN.js} +0 -0
  623. /package/public/dist/assets/{yacas-CqSYoSTv.js → yacas-GlqsXIeT.js} +0 -0
  624. /package/public/dist/assets/{z80-BltHQfm4.js → z80-CI7N40Oo.js} +0 -0
package/public/js/ui.ts CHANGED
@@ -1,6 +1,6 @@
1
1
  // ── UI Utilities ──
2
2
  import { state } from './state.js';
3
- import { renderMarkdown, escapeHtml, sanitizeHtml, stripOrchestration, linkifyFilePaths } from './render.js';
3
+ import { renderMarkdown, escapeHtml, stripOrchestration, linkifyFilePaths } from './render.js';
4
4
  import { renderMermaidBlocks } from './render.js';
5
5
  import { generateId } from './uuid.js';
6
6
  import { getAppName } from './features/appname.js';
@@ -12,7 +12,15 @@ import { getVirtualScroll, VS_THRESHOLD, type RestoreReason, type VirtualItem }
12
12
  import { bootstrapVirtualHistory, type VirtualHistoryBootstrapDeps } from './virtual-scroll-bootstrap.js';
13
13
  import { createStreamRenderer, appendChunk, finalizeStream, hydrateStreamRenderer, type StreamState } from './streaming-render.js';
14
14
  import { activateWidgets } from './diagram/iframe-renderer.js';
15
- import { renderLiveToolActivity, cleanupToolElements, bindToolItemInteractions, type ToolLogEntry } from './features/tool-ui.js';
15
+ import { renderLiveToolActivity, cleanupToolElements, type ToolLogEntry } from './features/tool-ui.js';
16
+ import { initMessageActions } from './features/message-actions.js';
17
+ import { addSystemMsg, addMessage, removeSkeleton } from './features/chat-messages.js';
18
+ import { buildLazyVirtualMessageItem } from './features/message-item-html.js';
19
+ import { loadMessages } from './features/message-history.js';
20
+ import { isChatNearBottom, reconcileChatBottomAfterLayout, showChatRestoreIndicator, hideChatRestoreIndicator, hideChatRestoreIndicatorAfterSettle, reconcileChatBottomAfterRestore, scrollToBottom, ensureScrollTracking, canFollowAfterRestore, markFollowingBottom } from './features/chat-scroll.js';
21
+ import { currentProcessBlockFromDom, hasAgentToolBlock, normalizeAgentToolBlocks, removeAgentToolBlocks, serializeProcessStepsForToolLog } from './features/process-block-dom.js';
22
+ import { mergeExplicitAndLiveToolLogs, normalizeMessageToolLog, parseToolLog, sanitizedToolLogEntries, sanitizedToolLogJson, sanitizedToolLogJsonFromEntries, toProcessSteps, type ActiveRunSnapshot, type MessageItem, type QueuedOverlayItem } from './features/process-log-adapter.js';
23
+ import { setStatus, updateQueueBadge, updateStatMsgs, loadStats } from './features/ui-status.js';
16
24
  import { ICONS, emojiToIcon, emojiToStatus, isCompletionEmoji } from './icons.js';
17
25
  import { providerIcon } from './provider-icons.js';
18
26
  import { findRunningProcessStepMatch } from './features/process-step-match.js';
@@ -30,7 +38,6 @@ import {
30
38
  collapseBlock,
31
39
  stopBlockTicker,
32
40
  buildProcessBlockHtml,
33
- bindProcessBlockInteractions,
34
41
  getStoredProcessStepDetail,
35
42
  mergeStoredProcessStepDetail,
36
43
  processStepMetaFromStore,
@@ -38,363 +45,24 @@ import {
38
45
  type ProcessStep,
39
46
  type ProcessBlockState,
40
47
  } from './features/process-block.js';
41
- interface MessageItem { role: string; content: string; tool_log?: string | null; trace_run_id?: string | null; cli?: string | null; }
42
- interface QueuedOverlayItem { id: string; prompt: string; source?: string; ts?: number; }
43
- interface ActiveRunSnapshot { running?: boolean; cli?: string; text?: string; toolLog?: ToolLogEntry[]; startedAt?: number; }
44
48
 
45
- declare global {
46
- interface Window {
47
- __jawProcessBlockLayoutMutation?: (anchor: Element | null, mutate: () => void) => void;
48
- }
49
- }
50
-
51
- function processStepType(toolType?: string): ProcessStep['type'] {
52
- return toolType === 'thinking' || toolType === 'search' || toolType === 'subagent'
53
- ? toolType
54
- : 'tool';
55
- }
56
-
57
- function processStepStatus(status?: string): ProcessStep['status'] {
58
- return status === 'running' || status === 'done' || status === 'error' ? status : 'done';
59
- }
60
-
61
- function fallbackToolLabel(tool: ToolLogEntry): string {
62
- if (tool.label) return tool.label;
63
- const named = tool as ToolLogEntry & { name?: unknown };
64
- return typeof named.name === 'string' && named.name ? named.name : 'tool';
65
- }
66
-
67
- function parseToolLog(toolLog?: string | null): ToolLogEntry[] {
68
- return parseToolLogBounded(toolLog) as ToolLogEntry[];
69
- }
70
49
 
71
- function sanitizedToolLogJson(toolLog?: string | null): string | null {
72
- return serializeSanitizedToolLog(parseToolLog(toolLog));
73
- }
74
-
75
- function normalizeMessageToolLog<T extends MessageItem>(message: T): T {
76
- if (message.role !== 'assistant' || !message.tool_log) return { ...message, tool_log: null };
77
- return { ...message, tool_log: sanitizedToolLogJson(message.tool_log) };
78
- }
50
+ export {
51
+ setStatus, updateQueueBadge, loadStats, loadMessages,
52
+ addSystemMsg, addMessage,
53
+ isChatNearBottom, reconcileChatBottomAfterLayout, showChatRestoreIndicator,
54
+ hideChatRestoreIndicator, hideChatRestoreIndicatorAfterSettle, reconcileChatBottomAfterRestore,
55
+ scrollToBottom,
56
+ };
79
57
 
80
58
  function getAgentIcon(_cli?: string | null): string {
81
59
  return getAgentAvatarMarkup();
82
60
  }
83
61
 
84
- function toProcessSteps(tools: ToolLogEntry[], runStartedAt?: number): ProcessStep[] {
85
- const baseTime = runStartedAt && runStartedAt > 0 ? runStartedAt : Date.now();
86
- return tools.map((tool) => ({
87
- id: generateId(),
88
- icon: tool.icon ? emojiToIcon(tool.icon) : ICONS.tool,
89
- rawIcon: tool.rawIcon || tool.icon || '',
90
- label: fallbackToolLabel(tool),
91
- isEmployee: tool.isEmployee === true,
92
- type: processStepType(tool.toolType),
93
- detail: tool.detail || '',
94
- stepRef: tool.stepRef || '',
95
- traceRunId: tool.traceRunId || '',
96
- traceSeq: tool.traceSeq,
97
- detailAvailable: tool.detailAvailable,
98
- detailBytes: tool.detailBytes,
99
- rawRetentionStatus: tool.rawRetentionStatus,
100
- status: processStepStatus(tool.status),
101
- startTime: baseTime,
102
- }));
103
- }
104
-
105
- const TOOL_BLOCK_SELECTOR =
106
- ':scope > .process-block, :scope > .tool-group, ' +
107
- ':scope > .msg-content > .process-block, :scope > .msg-content > .tool-group';
108
-
109
- function agentBody(agentMsg: HTMLElement): HTMLElement | null {
110
- return agentMsg.querySelector('.agent-body') as HTMLElement | null;
111
- }
112
-
113
- function agentToolBlocks(agentMsg: HTMLElement): HTMLElement[] {
114
- const body = agentBody(agentMsg);
115
- return body ? Array.from(body.querySelectorAll<HTMLElement>(TOOL_BLOCK_SELECTOR)) : [];
116
- }
117
-
118
- function preferredAgentToolBlock(body: HTMLElement): HTMLElement | null {
119
- const content = body.querySelector(':scope > .msg-content') as HTMLElement | null;
120
- return body.querySelector(':scope > .process-block')
121
- ?? body.querySelector(':scope > .tool-group')
122
- ?? content?.querySelector(':scope > .process-block')
123
- ?? content?.querySelector(':scope > .tool-group')
124
- ?? null;
125
- }
126
-
127
- function normalizeAgentToolBlocks(agentMsg: HTMLElement): void {
128
- const body = agentBody(agentMsg);
129
- if (!body) return;
130
-
131
- const content = body.querySelector('.msg-content') as HTMLElement | null;
132
- const blocks = agentToolBlocks(agentMsg);
133
- if (blocks.length === 0) return;
134
-
135
- const keep = preferredAgentToolBlock(body) ?? blocks[0];
136
- if (content && keep.parentElement !== body) {
137
- body.insertBefore(keep, content);
138
- }
139
-
140
- for (const block of blocks) {
141
- if (block !== keep) {
142
- releaseProcessBlockDetails(block);
143
- block.remove();
144
- }
145
- }
146
- }
147
-
148
- function hasAgentToolBlock(agentMsg: HTMLElement): boolean {
149
- return agentToolBlocks(agentMsg).length > 0;
150
- }
151
-
152
- function processStepTypeFromDom(type?: string): ProcessStep['type'] {
153
- return type === 'thinking' || type === 'search' || type === 'subagent'
154
- ? type
155
- : 'tool';
156
- }
157
-
158
- function processStepStatusFromDom(status?: string): ProcessStep['status'] {
159
- return status === 'done' || status === 'error' ? status : 'running';
160
- }
161
-
162
- function processStepFromDom(row: HTMLElement): ProcessStep | null {
163
- const id = row.dataset['stepId'] || '';
164
- if (!id) return null;
165
- const storedMeta = processStepMetaFromStore(id);
166
- const label = row.querySelector('.process-step-label')?.textContent?.trim() || '';
167
- const pre = row.querySelector('.process-step-full') as HTMLElement | null;
168
- const detail = pre?.dataset['detailLazy'] === 'true'
169
- ? getStoredProcessStepDetail(id) || storedMeta?.preview || ''
170
- : pre?.textContent || getStoredProcessStepDetail(id) || storedMeta?.preview || '';
171
- const iconEl = row.querySelector('.process-step-icon') as HTMLElement | null;
172
- const icon = iconEl?.innerHTML || ICONS.tool;
173
- const startTime = Number(row.dataset['startTime'] || '');
174
- return {
175
- id,
176
- type: storedMeta?.type || processStepTypeFromDom(row.dataset['type']),
177
- icon: storedMeta?.icon || icon,
178
- rawIcon: storedMeta?.rawIcon,
179
- label: storedMeta?.label || label,
180
- isEmployee: storedMeta?.isEmployee === true || row.dataset['isEmployee'] === 'true',
181
- detail,
182
- detailPreview: storedMeta?.preview,
183
- detailLength: storedMeta?.detailLength,
184
- detailTruncated: storedMeta?.detailTruncated,
185
- stepRef: storedMeta?.stepRef || row.dataset['stepRef'] || '',
186
- traceRunId: storedMeta?.traceRunId || row.dataset['traceRunId'] || '',
187
- traceSeq: storedMeta?.traceSeq || Number(row.dataset['traceSeq'] || 0) || undefined,
188
- detailAvailable: storedMeta?.detailAvailable,
189
- detailBytes: storedMeta?.detailBytes,
190
- rawRetentionStatus: storedMeta?.rawRetentionStatus,
191
- status: storedMeta?.status || processStepStatusFromDom(row.dataset['status']),
192
- startTime: Number.isFinite(startTime) && startTime > 0 ? startTime : Date.now(),
193
- };
194
- }
195
-
196
- function currentProcessBlockFromDom(agentMsg: HTMLElement): ProcessBlockState | null {
197
- const block = agentBody(agentMsg)?.querySelector(':scope > .process-block') as HTMLElement | null;
198
- if (!block) return null;
199
- const steps = Array.from(block.querySelectorAll<HTMLElement>('.process-step'))
200
- .map(processStepFromDom)
201
- .filter((step): step is ProcessStep => Boolean(step));
202
- return {
203
- element: block,
204
- steps,
205
- collapsed: block.classList.contains('collapsed'),
206
- };
207
- }
208
-
209
- function processStepToToolLog(step: ProcessStep, finalize = false): ToolLogEntry {
210
- const detail = getStoredProcessStepDetail(step.id) || step.detail || step.detailPreview || '';
211
- const status = finalize && step.status === 'running' ? 'done' : step.status;
212
- return {
213
- icon: step.rawIcon || step.icon || ICONS.tool,
214
- rawIcon: step.rawIcon || step.icon || '',
215
- label: step.label || 'tool',
216
- isEmployee: step.isEmployee === true,
217
- detail,
218
- toolType: step.type,
219
- stepRef: step.stepRef || '',
220
- status,
221
- traceRunId: step.traceRunId || '',
222
- traceSeq: step.traceSeq,
223
- detailAvailable: step.detailAvailable,
224
- detailBytes: step.detailBytes,
225
- rawRetentionStatus: step.rawRetentionStatus,
226
- };
227
- }
228
-
229
- function processStepFromMeta(stepId: string, finalize = false): ToolLogEntry | null {
230
- const meta = processStepMetaFromStore(stepId);
231
- if (!meta) return null;
232
- const status = finalize && meta.status === 'running' ? 'done' : meta.status;
233
- return {
234
- icon: meta.rawIcon || meta.icon || ICONS.tool,
235
- rawIcon: meta.rawIcon || meta.icon || '',
236
- label: meta.label || 'tool',
237
- isEmployee: meta.isEmployee === true,
238
- detail: getStoredProcessStepDetail(stepId) || meta.preview || '',
239
- toolType: meta.type,
240
- stepRef: meta.stepRef || '',
241
- status,
242
- traceRunId: meta.traceRunId || '',
243
- traceSeq: meta.traceSeq,
244
- detailAvailable: meta.detailAvailable,
245
- detailBytes: meta.detailBytes,
246
- rawRetentionStatus: meta.rawRetentionStatus,
247
- };
248
- }
249
-
250
- function serializeProcessStepsForToolLog(source: ProcessBlockState | HTMLElement | null, finalize = false): ToolLogEntry[] {
251
- if (!source) return [];
252
- if ('steps' in source) return source.steps.map(step => processStepToToolLog(step, finalize));
253
-
254
- const ids = new Set<string>();
255
- source.querySelectorAll<HTMLElement>('.process-block[data-process-step-ids]').forEach(block => {
256
- (block.dataset['processStepIds'] || '').split(/\s+/).filter(Boolean).forEach(id => ids.add(id));
257
- });
258
- source.querySelectorAll<HTMLElement>('.process-step[data-step-id]').forEach(row => {
259
- const id = row.dataset['stepId'];
260
- if (id) ids.add(id);
261
- });
262
- const entries: ToolLogEntry[] = [];
263
- ids.forEach(id => {
264
- const fromMeta = processStepFromMeta(id, finalize);
265
- if (fromMeta) {
266
- entries.push(fromMeta);
267
- return;
268
- }
269
- const row = source.querySelector<HTMLElement>(`.process-step[data-step-id="${CSS.escape(id)}"]`);
270
- const step = row ? processStepFromDom(row) : null;
271
- if (step) entries.push(processStepToToolLog(step, finalize));
272
- });
273
- return entries;
274
- }
275
-
276
- function identityKey(entry: ToolLogEntry, ordinal: number): string {
277
- const stepRef = String(entry.stepRef || '').trim();
278
- if (stepRef) return `ref:${stepRef}`;
279
- return `ord:${entry.toolType || 'tool'}:${entry.label || 'tool'}:${ordinal}`;
280
- }
281
-
282
- function mergeExplicitAndLiveToolLogs(explicit: ToolLogEntry[], live: ToolLogEntry[]): ToolLogEntry[] {
283
- if (explicit.length === 0) return live;
284
- const merged = new Map<string, ToolLogEntry>();
285
- const ordinalCounts = new Map<string, number>();
286
- const keyFor = (entry: ToolLogEntry): string => {
287
- const base = `${entry.toolType || 'tool'}:${entry.label || 'tool'}`;
288
- const next = (ordinalCounts.get(base) || 0) + 1;
289
- ordinalCounts.set(base, next);
290
- return identityKey(entry, next);
291
- };
292
- live.forEach(entry => merged.set(keyFor(entry), entry));
293
- ordinalCounts.clear();
294
- explicit.forEach(entry => {
295
- const key = keyFor(entry);
296
- const liveEntry = merged.get(key);
297
- const liveDetail = liveEntry?.detail || '';
298
- const explicitDetail = entry.detail || '';
299
- merged.set(key, {
300
- ...(liveEntry || {}),
301
- ...entry,
302
- detail: explicitDetail.length >= liveDetail.length ? explicitDetail : liveDetail,
303
- status: entry.status || liveEntry?.status || 'done',
304
- });
305
- });
306
- return Array.from(merged.values());
307
- }
308
-
309
- function sanitizedToolLogEntries(entries: ToolLogEntry[]): ToolLogEntry[] {
310
- return sanitizeToolLogForDurableStorage(entries) as ToolLogEntry[];
311
- }
312
-
313
- function sanitizedToolLogJsonFromEntries(entries: ToolLogEntry[]): string | null {
314
- return serializeSanitizedToolLog(entries);
315
- }
316
-
317
-
318
- function removeAgentToolBlocks(agentMsg: HTMLElement): void {
319
- for (const block of agentToolBlocks(agentMsg)) {
320
- releaseProcessBlockDetails(block);
321
- block.remove();
322
- }
323
- }
324
-
325
- export function setStatus(s: string): void {
326
- const badge = document.getElementById('statusBadge');
327
- const btn = document.getElementById('btnSend');
328
- state.agentBusy = s === 'running';
329
- document.getElementById('typingIndicator')?.classList.toggle('active', state.agentBusy);
330
- if (s === 'running') {
331
- if (badge) { badge.className = 'status-badge status-running'; badge.textContent = 'running'; }
332
- if (btn) { btn.innerHTML = ICONS.stop; btn.title = t('btn.stop'); btn.classList.add('stop-mode'); }
333
- showSkeleton();
334
- } else {
335
- if (badge) { badge.className = 'status-badge status-idle'; badge.textContent = 'idle'; }
336
- if (btn) { btn.innerHTML = ICONS.send; btn.title = 'Send'; btn.classList.remove('stop-mode'); }
337
- removeSkeleton();
338
- }
339
- }
340
-
341
- export function updateQueueBadge(count: number): void {
342
- let el = document.getElementById('queueBadge');
343
- if (!el) {
344
- el = document.createElement('span');
345
- el.id = 'queueBadge';
346
- el.className = 'queue-badge';
347
- const sendBtn = document.getElementById('btnSend');
348
- if (sendBtn?.parentElement) sendBtn.parentElement.style.position = 'relative';
349
- if (sendBtn) { sendBtn.style.position = 'relative'; sendBtn.appendChild(el); }
350
- }
351
- el.textContent = count > 0 ? String(count) : '';
352
- el.style.display = count > 0 ? 'flex' : 'none';
353
- }
354
-
355
- function showSkeleton(): void {
356
- const container = document.getElementById('chatMessages');
357
- if (!container || container.querySelector('.skeleton-msg')) return;
358
- if (state.currentAgentDiv && state.currentAgentDiv.isConnected) return;
359
- // No flushToDOM — skeleton goes directly into container as overlay
360
- hideEmptyState();
361
- const skel = document.createElement('div');
362
- skel.className = 'skeleton-msg';
363
- skel.innerHTML = '<div class="skeleton-line"></div><div class="skeleton-line"></div><div class="skeleton-line"></div>';
364
- container.appendChild(skel);
365
- scrollToBottom();
366
- }
367
-
368
- function removeSkeleton(): void {
369
- document.querySelectorAll('.skeleton-msg').forEach(el => el.remove());
370
- }
371
-
372
- function hideEmptyState(): void {
373
- document.getElementById('emptyState')?.classList.remove('visible');
374
- }
375
-
376
- function showEmptyState(): void {
377
- const container = document.getElementById('chatMessages');
378
- if (container && container.children.length === 0) {
379
- document.getElementById('emptyState')?.classList.add('visible');
380
- }
381
- }
382
-
383
- export function addSystemMsg(text: string, extraClass?: string, type?: string): void {
384
- const container = document.getElementById('chatMessages');
385
- if (!container) return;
386
- const vs = getVirtualScroll();
387
- hideEmptyState();
388
- const div = document.createElement('div');
389
- const typeClass = type ? ` msg-type-${type}` : '';
390
- div.className = 'msg msg-system' + typeClass + (extraClass ? ' ' + extraClass : '');
391
- div.innerHTML = sanitizeHtml(text);
392
- if (vs.active) {
393
- vs.appendLiveItem(div);
394
- } else {
395
- container.appendChild(div);
62
+ declare global {
63
+ interface Window {
64
+ __jawProcessBlockLayoutMutation?: (anchor: Element | null, mutate: () => void) => void;
396
65
  }
397
- scrollToBottom();
398
66
  }
399
67
 
400
68
  export function cleanupToolActivity(): void {
@@ -663,7 +331,7 @@ export function finalizeAgent(text: string, toolLog?: ToolLogEntry[]): void {
663
331
  content: finalText,
664
332
  cli: null,
665
333
  tool_log: durableToolLogJson,
666
- }));
334
+ }, vs.count));
667
335
  releaseProcessBlockDetails(div);
668
336
  } else {
669
337
  vs.appendLiveItem(div);
@@ -686,287 +354,6 @@ export function finalizeAgent(text: string, toolLog?: ToolLogEntry[]): void {
686
354
  loadStats();
687
355
  }
688
356
 
689
- /** Convert server-stored prompts back to display format on reload.
690
- * Handles: file uploads, multi-file, voice+file combos (ko + en patterns) */
691
- function formatUserPrompt(text: string): string {
692
- // Multi-file: "[사용자가 파일 3개를 보냈습니다]" or "[User sent 3 files]"
693
- const multiMatch = text.match(/^\[(?:사용자가 파일 (\d+)개를 보냈습니다|User sent (\d+) files)\]/);
694
- if (multiMatch) {
695
- const count = multiMatch[1] || multiMatch[2];
696
- const userMsgMatch = text.match(/(?:사용자 메시지|User message): (.+)$/s);
697
- const userMsg = userMsgMatch ? ' ' + userMsgMatch[1].trim() : '';
698
- return `📎 [${count} files]${userMsg}`;
699
- }
700
-
701
- // Single file: "[사용자가 파일을 보냈습니다: /path/to/file.md]"
702
- const fileMatch = text.match(/^\[(?:사용자가 파일을 보냈습니다|User sent a file): ([^\]]+)\]/);
703
- if (fileMatch) {
704
- const fileName = fileMatch[1].split('/').pop() || fileMatch[1];
705
- // Check if voice is also present (🎤 after file block)
706
- const voiceMatch = text.match(/🎤\s*(.{0,80})/);
707
- const voicePart = voiceMatch ? `${t('chat.voice.label')} ` : '';
708
- const userMsgMatch = text.match(/(?:사용자 메시지|User message): (.+)$/s);
709
- const userMsg = userMsgMatch ? ' ' + userMsgMatch[1].trim() : '';
710
- return `${voicePart}📎 [${fileName}]${userMsg}`;
711
- }
712
-
713
- return text;
714
- }
715
-
716
- export function addMessage(role: string, text: string, cli?: string | null): HTMLDivElement {
717
- const container = document.getElementById('chatMessages');
718
- const vs = getVirtualScroll();
719
- hideEmptyState();
720
- removeSkeleton();
721
-
722
- // For user messages: convert file-upload prompts to clean display format
723
- const displayText = role === 'user' ? formatUserPrompt(text) : text;
724
- const rendered = renderMarkdown(displayText);
725
- const label = escapeHtml(role === 'user' ? t('msg.you') : getAppName());
726
-
727
- const div = document.createElement('div');
728
- if (role === 'agent') {
729
- div.className = 'msg msg-agent';
730
- div.innerHTML = `<div class="agent-icon" aria-hidden="true">${getAgentIcon(cli)}</div><div class="agent-body"><div class="msg-content">${rendered}</div><button class="msg-copy" title="Copy" aria-label="Copy message"></button></div>`;
731
- } else {
732
- div.className = `msg msg-${role}`;
733
- div.innerHTML = `<div class="user-body"><div class="msg-label">${label}</div><div class="msg-content">${rendered}</div><button class="msg-copy" title="Copy" aria-label="Copy message"></button></div><div class="user-icon" aria-hidden="true">${getUserAvatarMarkup()}</div>`;
734
- }
735
- const contentEl = div.querySelector('.msg-content');
736
- if (contentEl) contentEl.setAttribute('data-raw', stripOrchestration(text));
737
-
738
- // Streaming placeholder (agent + empty text) must stay in real DOM
739
- // so state.currentAgentDiv reference remains valid during streaming.
740
- const isStreamingPlaceholder = role === 'agent' && !text;
741
-
742
- if (vs.active && !isStreamingPlaceholder) {
743
- if (div.classList.contains('msg-agent')) normalizeAgentToolBlocks(div);
744
- vs.appendLiveItem(div);
745
- } else {
746
- container?.appendChild(div);
747
- activateWidgets(div);
748
-
749
- // Check if live growth crossed threshold — activate VS
750
- if (!vs.active && !isStreamingPlaceholder && container) {
751
- const msgCount = container.querySelectorAll('.msg').length;
752
- if (msgCount >= VS_THRESHOLD) {
753
- // Feed all existing DOM messages into VS items array
754
- container.querySelectorAll('.msg').forEach(el => {
755
- if (el.classList.contains('msg-agent')) normalizeAgentToolBlocks(el as HTMLElement);
756
- vs.addItem(generateId(), el.outerHTML);
757
- });
758
- // Wire widget activation + file path linkification for VS-rendered items
759
- vs.onPostRender = (viewport: HTMLElement) => {
760
- activateWidgets(viewport);
761
- linkifyFilePaths(viewport);
762
- // Phase 127-F7b: render mermaid in newly mounted VS viewport
763
- void renderMermaidBlocks(viewport, { immediate: true });
764
- };
765
- }
766
- }
767
- }
768
- // Force scroll for user messages so they're always visible after sending;
769
- // agent/system messages respect the user's current scroll position.
770
- scrollToBottom(role === 'user');
771
- return div;
772
- }
773
-
774
- let scrollRAF: number | null = null;
775
- let userNearBottom = true;
776
- type ScrollIntent = 'unknown' | 'following' | 'pinnedAway';
777
- let scrollIntent: ScrollIntent = 'unknown';
778
- let scrollTrackingBound = false;
779
- const SCROLL_BOTTOM_THRESHOLD = 80; // px
780
- const RESTORE_INDICATOR_SETTLE_MS = 1100;
781
- let chatRestoreIndicatorHideTimer: number | null = null;
782
- const chatRestorePassTimers = new Set<number>();
783
- const chatRestorePassRafs = new Set<number>();
784
-
785
- function canFollowAfterRestore(): boolean {
786
- return scrollIntent !== 'pinnedAway';
787
- }
788
-
789
- function markFollowingBottom(): void {
790
- userNearBottom = true;
791
- scrollIntent = 'following';
792
- }
793
-
794
- function updateScrollIntentFromDistance(dist: number): void {
795
- userNearBottom = dist < SCROLL_BOTTOM_THRESHOLD;
796
- scrollIntent = userNearBottom ? 'following' : 'pinnedAway';
797
- if (scrollIntent === 'pinnedAway') cancelPendingChatRestorePasses();
798
- }
799
-
800
- function cancelPendingChatRestorePasses(): void {
801
- for (const timer of chatRestorePassTimers) window.clearTimeout(timer);
802
- chatRestorePassTimers.clear();
803
- for (const raf of chatRestorePassRafs) cancelAnimationFrame(raf);
804
- chatRestorePassRafs.clear();
805
- }
806
-
807
- function requestChatRestoreFrame(callback: () => void): void {
808
- const raf = requestAnimationFrame(() => {
809
- chatRestorePassRafs.delete(raf);
810
- callback();
811
- });
812
- chatRestorePassRafs.add(raf);
813
- }
814
-
815
- function trackChatRestoreTimer(timer: number): void {
816
- chatRestorePassTimers.add(timer);
817
- }
818
-
819
- function scheduleChatRestoreTimer(callback: () => void, delayMs: number): void {
820
- const timer = window.setTimeout(() => {
821
- chatRestorePassTimers.delete(timer);
822
- callback();
823
- }, delayMs);
824
- trackChatRestoreTimer(timer);
825
- }
826
-
827
- function ensureScrollTracking(): void {
828
- getVirtualScroll().setRestoreFollowPredicate(canFollowAfterRestore);
829
- window.__jawProcessBlockLayoutMutation = (anchor, mutate) => {
830
- const vs = getVirtualScroll();
831
- if (vs.active) {
832
- vs.preserveScrollDuringMutation(anchor, mutate);
833
- return;
834
- }
835
- mutate();
836
- };
837
- if (scrollTrackingBound) return;
838
- const c = document.getElementById('chatMessages');
839
- if (!c) return;
840
- scrollTrackingBound = true;
841
- c.addEventListener('scroll', () => {
842
- const dist = c.scrollHeight - c.scrollTop - c.clientHeight;
843
- updateScrollIntentFromDistance(dist);
844
- }, { passive: true });
845
- }
846
-
847
- export function isChatNearBottom(): boolean {
848
- ensureScrollTracking();
849
- const c = document.getElementById('chatMessages');
850
- if (!c) return userNearBottom;
851
- const vs = getVirtualScroll();
852
- if (vs.active) return vs.isNearBottom(SCROLL_BOTTOM_THRESHOLD);
853
- const dist = c.scrollHeight - c.scrollTop - c.clientHeight;
854
- return dist < SCROLL_BOTTOM_THRESHOLD;
855
- }
856
-
857
- export function reconcileChatBottomAfterLayout(shouldFollow = isChatNearBottom()): void {
858
- ensureScrollTracking();
859
- if (!shouldFollow) return;
860
- markFollowingBottom();
861
- const vs = getVirtualScroll();
862
- if (vs.active) {
863
- vs.reconcileBottomAfterLayout('reconnect', true);
864
- return;
865
- }
866
- requestAnimationFrame(() => {
867
- requestAnimationFrame(() => {
868
- const c = document.getElementById('chatMessages');
869
- if (c) c.scrollTop = c.scrollHeight;
870
- });
871
- });
872
- }
873
-
874
- export function showChatRestoreIndicator(reason: string): void {
875
- if (chatRestoreIndicatorHideTimer !== null) {
876
- window.clearTimeout(chatRestoreIndicatorHideTimer);
877
- chatRestoreIndicatorHideTimer = null;
878
- }
879
- const host = document.querySelector('.chat-area') as HTMLElement | null;
880
- if (!host) return;
881
- let indicator = host.querySelector('[data-restore-indicator="true"]') as HTMLElement | null;
882
- if (!indicator) {
883
- indicator = document.createElement('div');
884
- indicator.className = 'chat-restore-indicator';
885
- indicator.setAttribute('data-restore-indicator', 'true');
886
- indicator.setAttribute('role', 'status');
887
- indicator.setAttribute('aria-live', 'polite');
888
- indicator.innerHTML = '<span class="chat-restore-dot"></span><span class="chat-restore-text">Restoring</span>';
889
- host.appendChild(indicator);
890
- }
891
- indicator.dataset['restoreReason'] = reason;
892
- }
893
-
894
- export function hideChatRestoreIndicator(): void {
895
- if (chatRestoreIndicatorHideTimer !== null) {
896
- window.clearTimeout(chatRestoreIndicatorHideTimer);
897
- chatRestoreIndicatorHideTimer = null;
898
- }
899
- document.querySelectorAll('[data-restore-indicator="true"]').forEach(el => el.remove());
900
- }
901
-
902
- export function hideChatRestoreIndicatorAfterSettle(delayMs = RESTORE_INDICATOR_SETTLE_MS): void {
903
- if (chatRestoreIndicatorHideTimer !== null) {
904
- window.clearTimeout(chatRestoreIndicatorHideTimer);
905
- }
906
- chatRestoreIndicatorHideTimer = window.setTimeout(() => {
907
- chatRestoreIndicatorHideTimer = null;
908
- hideChatRestoreIndicator();
909
- }, delayMs);
910
- }
911
-
912
- export function reconcileChatBottomAfterRestore(reason: string): void {
913
- showChatRestoreIndicator(reason);
914
- hideChatRestoreIndicatorAfterSettle();
915
- ensureScrollTracking();
916
- const vs = getVirtualScroll();
917
- if (vs.active) {
918
- vs.reconcileAfterRestore(reason as RestoreReason, canFollowAfterRestore);
919
- return;
920
- }
921
- if (!canFollowAfterRestore()) return;
922
- const scrollIfFollowing = () => {
923
- if (!canFollowAfterRestore()) {
924
- cancelPendingChatRestorePasses();
925
- return;
926
- }
927
- const c = document.getElementById('chatMessages');
928
- if (c) {
929
- c.scrollTop = c.scrollHeight;
930
- markFollowingBottom();
931
- }
932
- };
933
- const runRestorePass = () => {
934
- if (!canFollowAfterRestore()) {
935
- cancelPendingChatRestorePasses();
936
- return;
937
- }
938
- requestChatRestoreFrame(scrollIfFollowing);
939
- };
940
- runRestorePass();
941
- requestChatRestoreFrame(runRestorePass);
942
- requestChatRestoreFrame(() => requestChatRestoreFrame(runRestorePass));
943
- scheduleChatRestoreTimer(runRestorePass, 250);
944
- scheduleChatRestoreTimer(runRestorePass, 1000);
945
- void document.fonts?.ready.then(runRestorePass);
946
- }
947
-
948
- /** Scroll chat to bottom.
949
- * @param force - bypass user-scroll detection (use for explicit user actions) */
950
- export function scrollToBottom(force = false): void {
951
- ensureScrollTracking();
952
- if (!force && !userNearBottom) return;
953
- // After force scroll, mark as near-bottom so subsequent
954
- // streaming chunks keep auto-scrolling until user scrolls up
955
- if (force) markFollowingBottom();
956
-
957
- const vs = getVirtualScroll();
958
- if (vs.active) {
959
- vs.scrollToBottom();
960
- return;
961
- }
962
- if (scrollRAF) return;
963
- scrollRAF = requestAnimationFrame(() => {
964
- scrollRAF = null;
965
- const c = document.getElementById('chatMessages');
966
- if (c) c.scrollTop = c.scrollHeight;
967
- });
968
- }
969
-
970
357
  export function switchTab(name: string, targetBtn: Element): void {
971
358
  document.querySelectorAll('.tab-btn').forEach(b => {
972
359
  b.classList.remove('active');
@@ -994,215 +381,10 @@ export function handleSave(): void {
994
381
  }
995
382
  }
996
383
 
997
- function updateStatMsgs(count: number): void {
998
- const el = document.getElementById('statMsgs');
999
- if (el) el.textContent = t('stat.messages', { count });
1000
- }
1001
-
1002
- export async function loadStats(): Promise<void> {
1003
- const msgs = await api<MessageItem[]>('/api/messages');
1004
- if (!msgs) return;
1005
- updateStatMsgs(msgs.length);
1006
- }
1007
-
1008
- // ── Virtual scroll bootstrap helpers ──
1009
-
1010
- function buildLazyVirtualMessageItem(m: MessageItem): VirtualItem {
1011
- const role = m.role === 'assistant' ? 'agent' : m.role;
1012
- const rawContent = stripOrchestration(
1013
- role === 'user' ? formatUserPrompt(m.content) : m.content,
1014
- );
1015
- const label = escapeHtml(role === 'user' ? t('msg.you') : getAppName());
1016
- const sanitizedToolLog = m.role === 'assistant' && m.tool_log
1017
- ? sanitizedToolLogJson(m.tool_log)
1018
- : null;
1019
- const rawToolLog = sanitizedToolLog ? escapeHtml(sanitizedToolLog) : '';
1020
- const toolAttr = rawToolLog ? ` data-tool-log="${rawToolLog}"` : '';
1021
- const contentHtml = `<div class="msg-content lazy-pending" data-raw="${escapeHtml(rawContent)}"></div>`;
1022
- const html = role === 'agent'
1023
- ? `<div class="msg msg-agent"><div class="agent-icon" aria-hidden="true">${getAgentIcon(m.cli)}</div><div class="agent-body"${toolAttr}>${contentHtml}<button class="msg-copy" title="Copy" aria-label="Copy message"></button></div></div>`
1024
- : `<div class="msg msg-${role}"><div class="user-body"><div class="msg-label">${label}</div>${contentHtml}<button class="msg-copy" title="Copy" aria-label="Copy message"></button></div><div class="user-icon" aria-hidden="true">${getUserAvatarMarkup()}</div></div>`;
1025
- return { id: generateId(), html, height: 80, rehydratesProcessDetails: Boolean(rawToolLog) };
1026
- }
1027
-
1028
- function buildVirtualHistoryItems(msgs: MessageItem[]): VirtualItem[] {
1029
- return msgs.map((m) => buildLazyVirtualMessageItem(normalizeMessageToolLog(m)));
1030
- }
1031
-
1032
- function registerVirtualScrollCallbacks(vs: ReturnType<typeof getVirtualScroll>): void {
1033
- vs.onLazyRender = (targets: HTMLElement[]) => {
1034
- for (const el of targets) {
1035
- if (!el.classList.contains('lazy-pending')) continue;
1036
- const raw = el.getAttribute('data-raw') || '';
1037
- const msgEl = el.closest('.msg-agent') as HTMLElement | null;
1038
- const body = msgEl?.querySelector('.agent-body') as HTMLElement | null;
1039
- const rawToolLog = body?.dataset['toolLog'] || '';
1040
- if (msgEl && body && rawToolLog && !hasAgentToolBlock(msgEl)) {
1041
- const tools = parseToolLog(rawToolLog);
1042
- if (tools.length > 0) {
1043
- el.insertAdjacentHTML(
1044
- 'beforebegin',
1045
- buildProcessBlockHtml(toProcessSteps(tools), true),
1046
- );
1047
- }
1048
- delete body.dataset['toolLog'];
1049
- normalizeAgentToolBlocks(msgEl);
1050
- }
1051
- el.innerHTML = raw ? renderMarkdown(raw) : '';
1052
- el.classList.remove('lazy-pending');
1053
- activateWidgets(el);
1054
- // Phase 127-F7a: lazy-rendered blocks (fresh markdown just converted)
1055
- void renderMermaidBlocks(el, { immediate: true });
1056
- }
1057
- };
1058
- vs.onPostRender = (viewport: HTMLElement) => {
1059
- activateWidgets(viewport);
1060
- linkifyFilePaths(viewport);
1061
- // Phase 127-F7b: mounted viewport scope — handles VS items that arrive
1062
- // pre-rendered with .mermaid-pending (buildVirtualHistoryItems path,
1063
- // addMessage append path). These are NOT .lazy-pending so F7a misses them.
1064
- void renderMermaidBlocks(viewport, { immediate: true });
1065
- };
1066
- }
1067
-
1068
- function makeBootstrapDeps(
1069
- vs: ReturnType<typeof getVirtualScroll>,
1070
- ): VirtualHistoryBootstrapDeps {
1071
- return {
1072
- registerCallbacks: () => registerVirtualScrollCallbacks(vs),
1073
- setItems: (items, opts) => vs.setItems(items, opts),
1074
- activateIfNeeded: (toBottom) => vs.activateIfNeeded(toBottom),
1075
- scrollToBottom: () => vs.scrollToBottom(),
1076
- shouldFollowBottom: canFollowAfterRestore,
1077
- onBeforeVirtualHistoryBootstrap: () => {
1078
- ensureScrollTracking();
1079
- },
1080
- onAfterVirtualHistoryBottomed: () => {
1081
- markFollowingBottom();
1082
- },
1083
- };
1084
- }
1085
-
1086
- export async function loadMessages(): Promise<void> {
1087
- const vs = getVirtualScroll();
1088
- const chatEl = document.getElementById('chatMessages');
1089
-
1090
- // Set scope from server workingDir (localStorage fallback if server is down)
1091
- try {
1092
- const settings = await api<{ workingDir?: string }>('/api/settings');
1093
- if (settings?.workingDir) setMessageScope(settings.workingDir);
1094
- } catch { /* localStorage fallback already initialized currentScope */ }
1095
-
1096
- const msgs = await api<MessageItem[]>('/api/messages');
1097
-
1098
- if (msgs !== null) {
1099
- const safeMsgs = msgs.map(normalizeMessageToolLog);
1100
- // Successful fetch — clear DOM and render (even if empty array after /clear)
1101
- vs.clear();
1102
- if (chatEl) chatEl.innerHTML = '';
1103
-
1104
- if (safeMsgs.length >= VS_THRESHOLD) {
1105
- const vsItems = buildVirtualHistoryItems(safeMsgs);
1106
- bootstrapVirtualHistory(vsItems, makeBootstrapDeps(vs));
1107
- } else {
1108
- safeMsgs.forEach(m => {
1109
- const div = addMessage(m.role === 'assistant' ? 'agent' : m.role, m.content, m.cli);
1110
- if (m.role === 'assistant') {
1111
- const tools = parseToolLog(m.tool_log);
1112
- if (tools.length > 0) {
1113
- const body = div.querySelector('.agent-body') as HTMLElement;
1114
- if (body) {
1115
- const pb = createProcessBlock(body);
1116
- for (const tool of toProcessSteps(tools)) addStep(pb, tool);
1117
- collapseBlock(pb);
1118
- }
1119
- }
1120
- }
1121
- });
1122
- }
1123
- // Sync to IndexedDB (full replace — server is source of truth)
1124
- cacheMessages(safeMsgs.map(m => ({
1125
- role: m.role, content: m.content, cli: m.cli ?? null, tool_log: m.tool_log ?? null, timestamp: Date.now(),
1126
- }))).catch(() => {});
1127
- updateStatMsgs(safeMsgs.length);
1128
- showEmptyState();
1129
- return;
1130
- }
1131
-
1132
- // Server unreachable (api() returned null) — preserve existing DOM messages
1133
- if (chatEl && chatEl.children.length > 0) {
1134
- showEmptyState();
1135
- return;
1136
- }
1137
- // DOM empty + server down — try IndexedDB cache
1138
- const cached = await getScopedMessages();
1139
- if (cached.length > 0) {
1140
- const safeCached = (cached as MessageItem[]).map(normalizeMessageToolLog);
1141
- if (safeCached.length >= VS_THRESHOLD) {
1142
- const vsItems = buildVirtualHistoryItems(safeCached);
1143
- bootstrapVirtualHistory(vsItems, makeBootstrapDeps(vs));
1144
- } else {
1145
- safeCached.forEach(m => {
1146
- const div = addMessage(m.role === 'assistant' ? 'agent' : m.role, m.content, m.cli);
1147
- if (m.role === 'assistant' && m.tool_log) {
1148
- const tools = parseToolLog(m.tool_log);
1149
- if (tools.length > 0) {
1150
- const body = div.querySelector('.agent-body') as HTMLElement;
1151
- if (body) {
1152
- const pb = createProcessBlock(body);
1153
- for (const tool of toProcessSteps(tools)) addStep(pb, tool);
1154
- collapseBlock(pb);
1155
- }
1156
- }
1157
- }
1158
- });
1159
- }
1160
- addSystemMsg(`${ICONS.warning} ${t('ui.offline.banner')}`);
1161
- updateStatMsgs(safeCached.length);
1162
- }
1163
- showEmptyState();
1164
- }
1165
-
1166
384
  // loadMemory removed — #memoryList element does not exist in HTML.
1167
385
  // Memory is now handled by features/memory.ts via the modal UI.
1168
386
 
1169
- // ── Message copy delegation ──
387
+ // ── Message action delegation ──
1170
388
  export function initMsgCopy(): void {
1171
- const chatMessages = document.getElementById('chatMessages');
1172
- if (!chatMessages) return;
1173
- bindProcessBlockInteractions(chatMessages);
1174
- bindToolItemInteractions(chatMessages);
1175
- chatMessages.addEventListener('click', (e) => {
1176
- const target = e.target as HTMLElement;
1177
-
1178
- // Tool group toggle (event delegation instead of inline onclick)
1179
- const summary = target.closest('.tool-group-summary') as HTMLElement | null;
1180
- if (summary) {
1181
- const group = summary.closest('.tool-group');
1182
- const details = summary.nextElementSibling as HTMLElement;
1183
- if (group && details) {
1184
- const isExpanding = !group.classList.contains('expanded');
1185
- group.classList.toggle('expanded');
1186
- details.classList.toggle('collapsed');
1187
- summary.setAttribute('aria-expanded', isExpanding ? 'true' : 'false');
1188
- }
1189
- return;
1190
- }
1191
-
1192
- // Message copy
1193
- const btn = target.closest('.msg-copy') as HTMLElement | null;
1194
- if (!btn) return;
1195
- const msg = btn.closest('.msg');
1196
- const content = msg?.querySelector('.msg-content') as HTMLElement | null;
1197
- if (!content) return;
1198
- const text = content.getAttribute('data-raw') || content.innerText || content.textContent || '';
1199
- navigator.clipboard.writeText(text).then(() => {
1200
- btn.classList.add('copied');
1201
- btn.innerHTML = ICONS.checkSimple;
1202
- setTimeout(() => {
1203
- btn.classList.remove('copied');
1204
- btn.textContent = '';
1205
- }, 600);
1206
- }).catch(() => { });
1207
- });
389
+ initMessageActions({ onStatus: addSystemMsg });
1208
390
  }