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.
- package/README.ja.md +17 -0
- package/README.ko.md +17 -0
- package/README.md +17 -0
- package/README.zh-CN.md +16 -1
- package/dist/bin/cli-jaw.js +7 -1
- package/dist/bin/cli-jaw.js.map +1 -1
- package/dist/bin/commands/browser.js +5 -0
- package/dist/bin/commands/browser.js.map +1 -1
- package/dist/bin/commands/connector.js +227 -0
- package/dist/bin/commands/connector.js.map +1 -0
- package/dist/bin/commands/dashboard-memory.js +243 -0
- package/dist/bin/commands/dashboard-memory.js.map +1 -0
- package/dist/bin/commands/dashboard.js +5 -0
- package/dist/bin/commands/dashboard.js.map +1 -1
- package/dist/bin/commands/dispatch-helpers.js +17 -0
- package/dist/bin/commands/dispatch-helpers.js.map +1 -0
- package/dist/bin/commands/dispatch.js +8 -3
- package/dist/bin/commands/dispatch.js.map +1 -1
- package/dist/bin/commands/doctor.js +34 -8
- package/dist/bin/commands/doctor.js.map +1 -1
- package/dist/bin/commands/reminders.js +105 -0
- package/dist/bin/commands/reminders.js.map +1 -0
- package/dist/bin/postinstall.js +26 -0
- package/dist/bin/postinstall.js.map +1 -1
- package/dist/server.js +71 -9
- package/dist/server.js.map +1 -1
- package/dist/src/agent/events.js +58 -30
- package/dist/src/agent/events.js.map +1 -1
- package/dist/src/agent/spawn.js +20 -3
- package/dist/src/agent/spawn.js.map +1 -1
- package/dist/src/agent/tool-timeout.js +27 -0
- package/dist/src/agent/tool-timeout.js.map +1 -0
- package/dist/src/agent/watchdog.js +20 -3
- package/dist/src/agent/watchdog.js.map +1 -1
- package/dist/src/browser/launch-policy.js +1 -1
- package/dist/src/browser/launch-policy.js.map +1 -1
- package/dist/src/browser/web-ai/chatgpt-model.js +102 -19
- package/dist/src/browser/web-ai/chatgpt-model.js.map +1 -1
- package/dist/src/browser/web-ai/chatgpt-response.js +30 -1
- package/dist/src/browser/web-ai/chatgpt-response.js.map +1 -1
- package/dist/src/browser/web-ai/chatgpt.js +55 -8
- package/dist/src/browser/web-ai/chatgpt.js.map +1 -1
- package/dist/src/browser/web-ai/diagnostics.js +1 -0
- package/dist/src/browser/web-ai/diagnostics.js.map +1 -1
- package/dist/src/browser/web-ai/gemini-live.js +24 -3
- package/dist/src/browser/web-ai/gemini-live.js.map +1 -1
- package/dist/src/browser/web-ai/interstitial.js +64 -0
- package/dist/src/browser/web-ai/interstitial.js.map +1 -0
- package/dist/src/browser/web-ai/session.js +12 -0
- package/dist/src/browser/web-ai/session.js.map +1 -1
- package/dist/src/browser/web-ai/watcher.js +25 -7
- package/dist/src/browser/web-ai/watcher.js.map +1 -1
- package/dist/src/cli/connector.js +60 -0
- package/dist/src/cli/connector.js.map +1 -0
- package/dist/src/cli/handlers-runtime.js +39 -0
- package/dist/src/cli/handlers-runtime.js.map +1 -1
- package/dist/src/cli/reminders.js +29 -0
- package/dist/src/cli/reminders.js.map +1 -0
- package/dist/src/core/cli-detect.js +161 -0
- package/dist/src/core/cli-detect.js.map +1 -0
- package/dist/src/core/config.js +9 -22
- package/dist/src/core/config.js.map +1 -1
- package/dist/src/core/db.js +14 -1
- package/dist/src/core/db.js.map +1 -1
- package/dist/src/core/settings-merge.js +2 -2
- package/dist/src/core/settings-merge.js.map +1 -1
- package/dist/src/jaw-ceo/completion.js +101 -0
- package/dist/src/jaw-ceo/completion.js.map +1 -0
- package/dist/src/jaw-ceo/confirmations.js +39 -0
- package/dist/src/jaw-ceo/confirmations.js.map +1 -0
- package/dist/src/jaw-ceo/coordinator-admin.js +295 -0
- package/dist/src/jaw-ceo/coordinator-admin.js.map +1 -0
- package/dist/src/jaw-ceo/coordinator-completions.js +201 -0
- package/dist/src/jaw-ceo/coordinator-completions.js.map +1 -0
- package/dist/src/jaw-ceo/coordinator-realtime-tools.js +108 -0
- package/dist/src/jaw-ceo/coordinator-realtime-tools.js.map +1 -0
- package/dist/src/jaw-ceo/coordinator-types.js +2 -0
- package/dist/src/jaw-ceo/coordinator-types.js.map +1 -0
- package/dist/src/jaw-ceo/coordinator-utils.js +86 -0
- package/dist/src/jaw-ceo/coordinator-utils.js.map +1 -0
- package/dist/src/jaw-ceo/coordinator-workers.js +234 -0
- package/dist/src/jaw-ceo/coordinator-workers.js.map +1 -0
- package/dist/src/jaw-ceo/coordinator.js +116 -0
- package/dist/src/jaw-ceo/coordinator.js.map +1 -0
- package/dist/src/jaw-ceo/docs-edit.js +142 -0
- package/dist/src/jaw-ceo/docs-edit.js.map +1 -0
- package/dist/src/jaw-ceo/openai-key.js +22 -0
- package/dist/src/jaw-ceo/openai-key.js.map +1 -0
- package/dist/src/jaw-ceo/policy.js +92 -0
- package/dist/src/jaw-ceo/policy.js.map +1 -0
- package/dist/src/jaw-ceo/realtime-sideband.js +372 -0
- package/dist/src/jaw-ceo/realtime-sideband.js.map +1 -0
- package/dist/src/jaw-ceo/store.js +206 -0
- package/dist/src/jaw-ceo/store.js.map +1 -0
- package/dist/src/jaw-ceo/transcript-persistence.js +28 -0
- package/dist/src/jaw-ceo/transcript-persistence.js.map +1 -0
- package/dist/src/jaw-ceo/types.js +2 -0
- package/dist/src/jaw-ceo/types.js.map +1 -0
- package/dist/src/manager/connector/audit-log.js +95 -0
- package/dist/src/manager/connector/audit-log.js.map +1 -0
- package/dist/src/manager/connector/routes.js +251 -0
- package/dist/src/manager/connector/routes.js.map +1 -0
- package/dist/src/manager/connector/types.js +2 -0
- package/dist/src/manager/connector/types.js.map +1 -0
- package/dist/src/manager/memory/embedding/hybrid-search.js +61 -0
- package/dist/src/manager/memory/embedding/hybrid-search.js.map +1 -0
- package/dist/src/manager/memory/embedding/index.js +6 -0
- package/dist/src/manager/memory/embedding/index.js.map +1 -0
- package/dist/src/manager/memory/embedding/provider.js +187 -0
- package/dist/src/manager/memory/embedding/provider.js.map +1 -0
- package/dist/src/manager/memory/embedding/state-machine.js +68 -0
- package/dist/src/manager/memory/embedding/state-machine.js.map +1 -0
- package/dist/src/manager/memory/embedding/sync.js +156 -0
- package/dist/src/manager/memory/embedding/sync.js.map +1 -0
- package/dist/src/manager/memory/embedding/vec-store.js +166 -0
- package/dist/src/manager/memory/embedding/vec-store.js.map +1 -0
- package/dist/src/manager/memory/federation.js +56 -0
- package/dist/src/manager/memory/federation.js.map +1 -0
- package/dist/src/manager/memory/index.js +5 -0
- package/dist/src/manager/memory/index.js.map +1 -0
- package/dist/src/manager/memory/instance-discovery.js +83 -0
- package/dist/src/manager/memory/instance-discovery.js.map +1 -0
- package/dist/src/manager/memory/result-rerank.js +28 -0
- package/dist/src/manager/memory/result-rerank.js.map +1 -0
- package/dist/src/manager/memory/types.js +2 -0
- package/dist/src/manager/memory/types.js.map +1 -0
- package/dist/src/manager/observability.js.map +1 -1
- package/dist/src/manager/process-verify.js +4 -1
- package/dist/src/manager/process-verify.js.map +1 -1
- package/dist/src/manager/registry.js +59 -1
- package/dist/src/manager/registry.js.map +1 -1
- package/dist/src/manager/reminders/api.js +16 -0
- package/dist/src/manager/reminders/api.js.map +1 -0
- package/dist/src/manager/reminders/dispatcher.js +40 -0
- package/dist/src/manager/reminders/dispatcher.js.map +1 -0
- package/dist/src/manager/reminders/due-time.js +16 -0
- package/dist/src/manager/reminders/due-time.js.map +1 -0
- package/dist/src/manager/reminders/instance-link.js +22 -0
- package/dist/src/manager/reminders/instance-link.js.map +1 -0
- package/dist/src/manager/reminders/routes.js +170 -0
- package/dist/src/manager/reminders/routes.js.map +1 -0
- package/dist/src/manager/reminders/scheduler.js +67 -0
- package/dist/src/manager/reminders/scheduler.js.map +1 -0
- package/dist/src/manager/reminders/store.js +311 -0
- package/dist/src/manager/reminders/store.js.map +1 -0
- package/dist/src/manager/routes/dashboard-memory.js +380 -0
- package/dist/src/manager/routes/dashboard-memory.js.map +1 -0
- package/dist/src/manager/server.js +209 -4
- package/dist/src/manager/server.js.map +1 -1
- package/dist/src/manager/worker-messages.js +60 -0
- package/dist/src/manager/worker-messages.js.map +1 -0
- package/dist/src/manager/workspace/routes.js +240 -0
- package/dist/src/manager/workspace/routes.js.map +1 -0
- package/dist/src/manager/workspace/store.js +369 -0
- package/dist/src/manager/workspace/store.js.map +1 -0
- package/dist/src/manager/workspace/types.js +5 -0
- package/dist/src/manager/workspace/types.js.map +1 -0
- package/dist/src/memory/indexing.js +38 -11
- package/dist/src/memory/indexing.js.map +1 -1
- package/dist/src/memory/shared.js +6 -0
- package/dist/src/memory/shared.js.map +1 -1
- package/dist/src/orchestrator/parser.js +3 -8
- package/dist/src/orchestrator/parser.js.map +1 -1
- package/dist/src/prompt/builder.js +35 -16
- package/dist/src/prompt/builder.js.map +1 -1
- package/dist/src/prompt/templates/a1-system.md +23 -1
- package/dist/src/reminders/jaw-reminders-bridge.js +313 -0
- package/dist/src/reminders/jaw-reminders-bridge.js.map +1 -0
- package/dist/src/reminders/types.js +7 -0
- package/dist/src/reminders/types.js.map +1 -0
- package/dist/src/routes/jaw-ceo.js +290 -0
- package/dist/src/routes/jaw-ceo.js.map +1 -0
- package/dist/src/routes/jaw-memory.js +37 -1
- package/dist/src/routes/jaw-memory.js.map +1 -1
- package/dist/src/routes/settings.js +47 -12
- package/dist/src/routes/settings.js.map +1 -1
- package/package.json +4 -1
- package/public/css/chat.css +90 -12
- package/public/css/diagram.css +9 -9
- package/public/css/modals.css +60 -1
- package/public/css/sidebar.css +18 -0
- package/public/dist/assets/{AdvancedExport-DJZ2VmBR.js → AdvancedExport-BAdZUC-6.js} +1 -1
- package/public/dist/assets/{Agent-CgpLT8IY.js → Agent-DNpehKB2.js} +1 -1
- package/public/dist/assets/{Browser-CrkiQoB8.js → Browser-BsdxDVgM.js} +1 -1
- package/public/dist/assets/{ChannelsDiscord-CVUC22D4.js → ChannelsDiscord-Dg_jto6l.js} +1 -1
- package/public/dist/assets/{ChannelsTelegram-DEatIQNM.js → ChannelsTelegram-DGyZfJGS.js} +1 -1
- package/public/dist/assets/DashboardEmbeddingSection-BYdGgqg7.js +2 -0
- package/public/dist/assets/{DashboardMeta-BKoxRc7i.js → DashboardMeta-CAH9ONTb.js} +1 -1
- package/public/dist/assets/{Display-DnNGV9Km.js → Display-DM_yvyKL.js} +1 -1
- package/public/dist/assets/{Employees-DZ2iJYKy.js → Employees-CuYuTy0R.js} +1 -1
- package/public/dist/assets/{HealthBadge-Cq2c7G9s.js → HealthBadge-Dtr-dDnw.js} +1 -1
- package/public/dist/assets/{Heartbeat-BML6eTXZ.js → Heartbeat-C-vq02MW.js} +1 -1
- package/public/dist/assets/{InlineWarn-BooBRm7o.js → InlineWarn-E64UaKFh.js} +1 -1
- package/public/dist/assets/{Mcp-BlEviQ3h.js → Mcp-Dlp2X7X7.js} +1 -1
- package/public/dist/assets/{Memory-BRyH80He.js → Memory-BPKWJDXK.js} +1 -1
- package/public/dist/assets/MilkdownWysiwygEditor-Ctww8i0L.js +160 -0
- package/public/dist/assets/{ModelProvider-DxyR7EL9.js → ModelProvider-Bd6vGkXT.js} +1 -1
- package/public/dist/assets/{Network-DDOOESh1.js → Network-Df1R2YcQ.js} +1 -1
- package/public/dist/assets/{Permissions-Br0eSbKb.js → Permissions-BKJ5K6EL.js} +1 -1
- package/public/dist/assets/{Permissions-QHkzStqQ.js → Permissions-CcWZoOVP.js} +1 -1
- package/public/dist/assets/{Profile-C79NKumk.js → Profile-DZ7xf1WZ.js} +1 -1
- package/public/dist/assets/{Prompts-BmiIDiXW.js → Prompts-Bh5DYt8e.js} +1 -1
- package/public/dist/assets/{SpeechKeys-B8304XJK.js → SpeechKeys-CQwtVxOP.js} +1 -1
- package/public/dist/assets/app-BxsIleo0.js +32 -0
- package/public/dist/assets/app-CB9n5A77.css +1 -0
- package/public/dist/assets/architecture-YZFGNWBL-BRI-0IaU.js +1 -0
- package/public/dist/assets/{architectureDiagram-Q4EWVU46-D0VjH4Hz.js → architectureDiagram-Q4EWVU46-z0JCgZrJ.js} +1 -1
- package/public/dist/assets/{blockDiagram-DXYQGD6D-CY4ROZPD.js → blockDiagram-DXYQGD6D-KaOz3aFS.js} +1 -1
- package/public/dist/assets/{c4Diagram-AHTNJAMY-CXSKBhA5.js → c4Diagram-AHTNJAMY-Bp2QLC-s.js} +1 -1
- package/public/dist/assets/channel-jLopKIgm.js +1 -0
- package/public/dist/assets/{chunk-2KRD3SAO-DKaML5Aw.js → chunk-2KRD3SAO-C8hJZPJu.js} +1 -1
- package/public/dist/assets/{chunk-336JU56O-CU3sZss9.js → chunk-336JU56O-CAt5aBOe.js} +1 -1
- package/public/dist/assets/chunk-426QAEUC-atJRhvKN.js +1 -0
- package/public/dist/assets/{chunk-4BX2VUAB-X0ZSlark.js → chunk-4BX2VUAB-Cm3KqBkS.js} +1 -1
- package/public/dist/assets/{chunk-4TB4RGXK-DxI0GIVy.js → chunk-4TB4RGXK-BD7UhIN8.js} +1 -1
- package/public/dist/assets/chunk-55IACEB6-CAWcPDdr.js +1 -0
- package/public/dist/assets/{chunk-5FUZZQ4R-BqlA0aLQ.js → chunk-5FUZZQ4R-Cg1pzs6p.js} +1 -1
- package/public/dist/assets/{chunk-5PVQY5BW-BVW1DG4s.js → chunk-5PVQY5BW-otv3HwE_.js} +1 -1
- package/public/dist/assets/{chunk-67CJDMHE-Blm2TZ-J.js → chunk-67CJDMHE-AU2iOKLT.js} +1 -1
- package/public/dist/assets/{chunk-7N4EOEYR-BoIOnnji.js → chunk-7N4EOEYR-BTIS4jBw.js} +1 -1
- package/public/dist/assets/{chunk-AA7GKIK3-Dsfi7oEW.js → chunk-AA7GKIK3-c9UUJO_T.js} +1 -1
- package/public/dist/assets/{chunk-BSJP7CBP-DMGz9IOn.js → chunk-BSJP7CBP-DnSYKjii.js} +1 -1
- package/public/dist/assets/{chunk-CIAEETIT-T5_zEp7h.js → chunk-CIAEETIT-Dwbln6rM.js} +1 -1
- package/public/dist/assets/{chunk-EDXVE4YY-DjfvQwok.js → chunk-EDXVE4YY-COudQKkJ.js} +1 -1
- package/public/dist/assets/{chunk-ENJZ2VHE-LawplD1x.js → chunk-ENJZ2VHE-PT_jAckw.js} +1 -1
- package/public/dist/assets/{chunk-FMBD7UC4-C7tckXtF.js → chunk-FMBD7UC4-6yXYicjw.js} +1 -1
- package/public/dist/assets/{chunk-FOC6F5B3-xPGbUzqY.js → chunk-FOC6F5B3-BqDmFFeg.js} +1 -1
- package/public/dist/assets/{chunk-ICPOFSXX-Cdp_Bt1o.js → chunk-ICPOFSXX-P6V6jqiT.js} +2 -2
- package/public/dist/assets/{chunk-K5T4RW27-BwIFil23.js → chunk-K5T4RW27-C42sOmZR.js} +1 -1
- package/public/dist/assets/{chunk-KGLVRYIC-BsNQ1OkZ.js → chunk-KGLVRYIC-K-BVa8b2.js} +1 -1
- package/public/dist/assets/{chunk-LIHQZDEY-DbaX7k3n.js → chunk-LIHQZDEY-Xw9ZX8of.js} +1 -1
- package/public/dist/assets/{chunk-ORNJ4GCN-C6D0ZHHr.js → chunk-ORNJ4GCN-CEXU6WaD.js} +1 -1
- package/public/dist/assets/{chunk-OYMX7WX6-C1WV9uUT.js → chunk-OYMX7WX6-DmJG2z1S.js} +1 -1
- package/public/dist/assets/chunk-QZHKN3VN-BkVLii_3.js +1 -0
- package/public/dist/assets/{chunk-U2HBQHQK-B-5yzlzA.js → chunk-U2HBQHQK-Ny6UwUgK.js} +1 -1
- package/public/dist/assets/{chunk-X2U36JSP-yL0Tpsl1.js → chunk-X2U36JSP-CoMFMVNF.js} +1 -1
- package/public/dist/assets/{chunk-XPW4576I-CeaUaSCU.js → chunk-XPW4576I-CmAz5C-j.js} +1 -1
- package/public/dist/assets/{chunk-YZCP3GAM-CASMOrQq.js → chunk-YZCP3GAM-c7FYtZME.js} +1 -1
- package/public/dist/assets/{chunk-ZZ45TVLE-B4c7TOr9.js → chunk-ZZ45TVLE-D5oxCC1O.js} +1 -1
- package/public/dist/assets/classDiagram-6PBFFD2Q-BgOyAynm.js +1 -0
- package/public/dist/assets/classDiagram-v2-HSJHXN6E-5gBdoVH9.js +1 -0
- package/public/dist/assets/{cose-bilkent-S5V4N54A-B21AcnwY.js → cose-bilkent-S5V4N54A-CvH6qY_r.js} +1 -1
- package/public/dist/assets/{dagre-DzTKkA3a.js → dagre--20B2-ZQ.js} +1 -1
- package/public/dist/assets/{dagre-KV5264BT-fAyzMxIJ.js → dagre-KV5264BT-CQ0wo0ma.js} +1 -1
- package/public/dist/assets/{diagram-5BDNPKRD-O30SB9_x.js → diagram-5BDNPKRD-Cl4PzCD0.js} +1 -1
- package/public/dist/assets/{diagram-G4DWMVQ6-DZ1S54sq.js → diagram-G4DWMVQ6-_aP6kMTh.js} +1 -1
- package/public/dist/assets/{diagram-MMDJMWI5-DRO_GSWW.js → diagram-MMDJMWI5-J1l6Q0JG.js} +1 -1
- package/public/dist/assets/{diagram-TYMM5635-BxWgQj5r.js → diagram-TYMM5635-B_Xoa1Gp.js} +1 -1
- package/public/dist/assets/{dist-W51oDoeA.js → dist-2oDfqE98.js} +1 -1
- package/public/dist/assets/{dist-BZosRD2u.js → dist-55MYVjjj.js} +1 -1
- package/public/dist/assets/dist-B1p80u1b.js +1 -0
- package/public/dist/assets/{dist-Bd9PlnQm.js → dist-B1rKu9eP.js} +1 -1
- package/public/dist/assets/{dist-Ch5VAlV9.js → dist-B6G8pbap.js} +1 -1
- package/public/dist/assets/{dist-CpUK8ypo.js → dist-BA7sRne4.js} +1 -1
- package/public/dist/assets/{dist-eU7TyC86.js → dist-BDMNMdPF.js} +1 -1
- package/public/dist/assets/{dist-DFYRUAjN.js → dist-BGzHP3f8.js} +1 -1
- package/public/dist/assets/dist-BKyzWv22.js +1 -0
- package/public/dist/assets/{dist-BNfXO3Yr.js → dist-BTp_Oy_x.js} +1 -1
- package/public/dist/assets/{dist-BD0UXfgF2.js → dist-BhzKO6nt2.js} +1 -1
- package/public/dist/assets/{dist-BUnPYbK3.js → dist-BjqyutOM.js} +1 -1
- package/public/dist/assets/dist-BoG5I6U5.js +1 -0
- package/public/dist/assets/{dist-yHP6L0Ty.js → dist-BuaQLcgQ.js} +1 -1
- package/public/dist/assets/{dist-DZsFVYF4.js → dist-C0sOT_UM.js} +5 -5
- package/public/dist/assets/{dist-CIuXW-sc.js → dist-C5S-Rbvc.js} +1 -1
- package/public/dist/assets/{dist-Db16ogVk.js → dist-C9LWf2uC.js} +1 -1
- package/public/dist/assets/{dist-urPTQzXL.js → dist-CCKktDoF.js} +1 -1
- package/public/dist/assets/{dist-D7bCuS3f.js → dist-CEDX2HGI.js} +1 -1
- package/public/dist/assets/dist-Ch6JG5jE.js +23 -0
- package/public/dist/assets/{dist-SU-YTAIg.js → dist-CkMC2PPt.js} +1 -1
- package/public/dist/assets/dist-CxJpXP6s.js +1 -0
- package/public/dist/assets/dist-DNLFuTrS.js +1 -0
- package/public/dist/assets/{dist-CL4PTYWf.js → dist-DdhWu7OM.js} +1 -1
- package/public/dist/assets/{dist-D6cUXP7K.js → dist-G7QUHtDS.js} +1 -1
- package/public/dist/assets/{dist-UYn7T-GH.js → dist-W7IGn2ug.js} +1 -1
- package/public/dist/assets/{dist-Cp42cMcI.js → dist-wdLr2dSH.js} +1 -1
- package/public/dist/assets/{dockerfile-zC7EsydA.js → dockerfile-CrC2HXHP.js} +1 -1
- package/public/dist/assets/employees-CIkIyvtL.js +33 -0
- package/public/dist/assets/{erDiagram-SMLLAGMA-oYlCN-2d.js → erDiagram-SMLLAGMA-BvtqhMsS.js} +1 -1
- package/public/dist/assets/{error-normalize-5n-zlEQ3.js → error-normalize-CkhPeQYx.js} +1 -1
- package/public/dist/assets/esm-BmJkIat2.js +4 -0
- package/public/dist/assets/{factor-DSufIRGh.js → factor-1CttFx2G.js} +1 -1
- package/public/dist/assets/{fields-BtncIZYA.js → fields-BISouxp2.js} +1 -1
- package/public/dist/assets/{flowDiagram-DWJPFMVM-er6Wfb34.js → flowDiagram-DWJPFMVM-C_F0rqqT.js} +1 -1
- package/public/dist/assets/{ganttDiagram-T4ZO3ILL-Co2j1qn3.js → ganttDiagram-T4ZO3ILL-B5S8EZRg.js} +1 -1
- package/public/dist/assets/gitGraph-7Q5UKJZL-DTTW6pxr.js +1 -0
- package/public/dist/assets/{gitGraphDiagram-UUTBAWPF-clYndUgV.js → gitGraphDiagram-UUTBAWPF-DMGzYJeb.js} +1 -1
- package/public/dist/assets/{graphlib-BjOif-SH.js → graphlib-CZk_Ii16.js} +1 -1
- package/public/dist/assets/idb-cache-BnZfG5FD.js +1 -0
- package/public/dist/assets/info-OMHHGYJF-BU-iuaSm.js +1 -0
- package/public/dist/assets/{infoDiagram-42DDH7IO-uYs7OoSA.js → infoDiagram-42DDH7IO-dzSaYFFK.js} +1 -1
- package/public/dist/assets/{ishikawaDiagram-UXIWVN3A-32bIEtCb.js → ishikawaDiagram-UXIWVN3A-DZW-Nqsf.js} +1 -1
- package/public/dist/assets/javascript-BhB45e0W.js +1 -0
- package/public/dist/assets/{journeyDiagram-VCZTEJTY-hBNsU7mK.js → journeyDiagram-VCZTEJTY-ByVRJxVF.js} +1 -1
- package/public/dist/assets/{kanban-definition-6JOO6SKY-DoepMHBI.js → kanban-definition-6JOO6SKY-xwN0YCW2.js} +1 -1
- package/public/dist/assets/katex-DamPUmTE.js +1 -0
- package/public/dist/assets/manager-B84u-pcn.js +25 -0
- package/public/dist/assets/manager-DMg_sTEP.css +1 -0
- package/public/dist/assets/memory-BV62wlsG.js +1 -0
- package/public/dist/assets/{memory-dJGp6QBv.js → memory-CnBc2_Va.js} +1 -1
- package/public/dist/assets/mermaid-loader-DxFGz4EE.js +1 -0
- package/public/dist/assets/{mermaid-parser.core-CrXy-45O.js → mermaid-parser.core-CMIZ0my_.js} +1 -1
- package/public/dist/assets/mermaid-preprocess-p7OIpOon.js +5 -0
- package/public/dist/assets/mermaid.core-CicVBD9l.js +1 -0
- package/public/dist/assets/{mermaid.core-DYApH8sc.js → mermaid.core-Dj6viEzN.js} +2 -2
- package/public/dist/assets/{mindmap-definition-QFDTVHPH-C-QWWjaf.js → mindmap-definition-QFDTVHPH-Chqgsh1N.js} +1 -1
- package/public/dist/assets/{nsis-DLskXEqR.js → nsis-_CjIiUyF.js} +1 -1
- package/public/dist/assets/packet-4T2RLAQJ-DzmBsAf0.js +1 -0
- package/public/dist/assets/{page-shell-D5tbivHH.js → page-shell-CTxVDJ8Y.js} +1 -1
- package/public/dist/assets/pie-ZZUOXDRM-DyG1KzTn.js +1 -0
- package/public/dist/assets/{pieDiagram-DEJITSTG-DvMWxH5R.js → pieDiagram-DEJITSTG-DJLiSymi.js} +1 -1
- package/public/dist/assets/{pug-B2rz0Rz1.js → pug-CbR8lCtK.js} +1 -1
- package/public/dist/assets/{quadrantDiagram-34T5L4WZ-DjoQDyzO.js → quadrantDiagram-34T5L4WZ-qcnWmcRr.js} +1 -1
- package/public/dist/assets/radar-PYXPWWZC-CEBZARUu.js +1 -0
- package/public/dist/assets/render-lpTpN1El.js +28 -0
- package/public/dist/assets/{requirementDiagram-MS252O5E-sQav3xIA.js → requirementDiagram-MS252O5E-wnPtv0LG.js} +1 -1
- package/public/dist/assets/{sankeyDiagram-XADWPNL6-DkdEFMeo.js → sankeyDiagram-XADWPNL6-Bn3pUd_K.js} +1 -1
- package/public/dist/assets/{sequenceDiagram-FGHM5R23-kaplo282.js → sequenceDiagram-FGHM5R23-DfxcHSdj.js} +1 -1
- package/public/dist/assets/settings-CfAXLa8a.js +1 -0
- package/public/dist/assets/{settings-C7QWaUHB.js → settings-FWgmcubl.js} +8 -8
- package/public/dist/assets/sidebar-Cwt0FxQl.js +14 -0
- package/public/dist/assets/skills-D-qUVJ-e.js +1 -0
- package/public/dist/assets/{skills-CHkTgM7L.js → skills-DDmTyywh.js} +2 -2
- package/public/dist/assets/slash-commands-BfI19vIz.js +1 -0
- package/public/dist/assets/{slash-commands-2ThyUGvX.js → slash-commands-D5y5AVvS.js} +1 -1
- package/public/dist/assets/{stateDiagram-FHFEXIEX-B2L3h2aE.js → stateDiagram-FHFEXIEX-CiR7P2OG.js} +1 -1
- package/public/dist/assets/stateDiagram-v2-QKLJ7IA2-BaWWpysK.js +1 -0
- package/public/dist/assets/{timeline-definition-GMOUNBTQ-_Nnp2LU-.js → timeline-definition-GMOUNBTQ-DUz1-iOx.js} +1 -1
- package/public/dist/assets/{trace-drawer-Dis80M6X.js → trace-drawer-DVF5F1df.js} +1 -1
- package/public/dist/assets/treeView-SZITEDCU-D5-TP7Qp.js +1 -0
- package/public/dist/assets/treemap-W4RFUUIX-BQhm_ZSp.js +1 -0
- package/public/dist/assets/ui-BJXY16Dk.js +140 -0
- package/public/dist/assets/ui-BXm3OPPh.js +1 -0
- package/public/dist/assets/{vendor-render-984tPmpy.js → vendor-render-DEStnpJ8.js} +22 -22
- package/public/dist/assets/{vennDiagram-DHZGUBPP-vNU8EDoW.js → vennDiagram-DHZGUBPP-CYu7SDdG.js} +1 -1
- package/public/dist/assets/wardley-RL74JXVD-CpnBMRe0.js +1 -0
- package/public/dist/assets/{wardleyDiagram-NUSXRM2D-DB2wK0zr.js → wardleyDiagram-NUSXRM2D-CLAhOzap.js} +1 -1
- package/public/dist/assets/wiki-link-suggestions-LQuYT22G.js +23 -0
- package/public/dist/assets/{xychartDiagram-5P7HB3ND-CisgDiJM.js → xychartDiagram-5P7HB3ND-DLnY9i8z.js} +1 -1
- package/public/dist/index.html +36 -12
- package/public/dist/manager/index.html +2 -2
- package/public/index.html +34 -10
- package/public/js/features/chat-messages.ts +132 -0
- package/public/js/features/chat-scroll.ts +226 -0
- package/public/js/features/employees.ts +4 -1
- package/public/js/features/help-content.ts +11 -1
- package/public/js/features/help-dialog.ts +78 -17
- package/public/js/features/message-actions.ts +157 -0
- package/public/js/features/message-history.ts +138 -0
- package/public/js/features/message-item-html.ts +34 -0
- package/public/js/features/preview-shortcut-bridge.ts +29 -0
- package/public/js/features/process-block-dom.ts +175 -0
- package/public/js/features/process-log-adapter.ts +104 -0
- package/public/js/features/settings-cli-status.ts +52 -1
- package/public/js/features/settings.ts +1 -0
- package/public/js/features/sidebar.ts +7 -1
- package/public/js/features/ui-status.ts +47 -0
- package/public/js/main.ts +10 -3
- package/public/js/render/code-copy.ts +47 -0
- package/public/js/render/delegations.ts +10 -0
- package/public/js/render/file-links.ts +150 -0
- package/public/js/render/highlight.ts +83 -0
- package/public/js/render/html.ts +26 -0
- package/public/js/render/markdown.ts +84 -0
- package/public/js/render/math.ts +76 -0
- package/public/js/render/mermaid-preprocess.ts +77 -0
- package/public/js/render/mermaid.ts +267 -0
- package/public/js/render/post-render.ts +35 -0
- package/public/js/render/sanitize.ts +51 -0
- package/public/js/render/svg-actions.ts +291 -0
- package/public/js/render.ts +17 -1081
- package/public/js/ui.ts +23 -841
- package/public/js/virtual-scroll-bootstrap.ts +18 -5
- package/public/js/virtual-scroll.ts +17 -0
- package/public/js/ws.ts +5 -0
- package/public/locales/en.json +76 -1
- package/public/locales/ja.json +74 -1
- package/public/locales/ko.json +76 -1
- package/public/locales/zh.json +74 -1
- package/public/manager/src/App.tsx +143 -258
- package/public/manager/src/AppChrome.tsx +144 -0
- package/public/manager/src/InstancePreview.tsx +18 -0
- package/public/manager/src/SidebarRailRouter.tsx +229 -0
- package/public/manager/src/components/ActivityTimeline.tsx +9 -1
- package/public/manager/src/components/ProcessControlPanel.tsx +3 -0
- package/public/manager/src/components/SidebarRail.tsx +22 -0
- package/public/manager/src/components/Workbench.tsx +17 -13
- package/public/manager/src/components/WorkbenchHeader.tsx +44 -33
- package/public/manager/src/components/WorkspaceLayout.tsx +3 -0
- package/public/manager/src/dashboard-board/DashboardBoardSidebar.tsx +1 -1
- package/public/manager/src/dashboard-board/DashboardBoardWorkspace.tsx +6 -0
- package/public/manager/src/dashboard-features.ts +4 -1
- package/public/manager/src/dashboard-reminders/DashboardRemindersSidebar.tsx +157 -0
- package/public/manager/src/dashboard-reminders/DashboardRemindersWorkspace.tsx +502 -0
- package/public/manager/src/dashboard-reminders/InlineReminderTitle.tsx +78 -0
- package/public/manager/src/dashboard-reminders/ReminderDetailPopover.tsx +118 -0
- package/public/manager/src/dashboard-reminders/reminder-order.ts +45 -0
- package/public/manager/src/dashboard-reminders/reminders-api.ts +126 -0
- package/public/manager/src/dashboard-reminders/reminders-view-model.ts +63 -0
- package/public/manager/src/dashboard-reminders/useDashboardReminderDrag.ts +55 -0
- package/public/manager/src/dashboard-reminders/useRemindersFeed.ts +72 -0
- package/public/manager/src/dashboard-schedule/DashboardScheduleWorkspace.tsx +10 -4
- package/public/manager/src/dashboard-settings/DashboardEmbeddingSection.tsx +350 -0
- package/public/manager/src/dashboard-settings/DashboardSettingsSidebar.tsx +6 -2
- package/public/manager/src/dashboard-settings/DashboardSettingsWorkspace.tsx +217 -6
- package/public/manager/src/dashboard-settings/dashboard-settings-ui.ts +2 -0
- package/public/manager/src/dashboard-url-state.ts +15 -0
- package/public/manager/src/help/HelpDrawer.tsx +5 -6
- package/public/manager/src/help/HelpTopicButton.tsx +23 -0
- package/public/manager/src/help/help-shortcuts.ts +7 -0
- package/public/manager/src/help/helpContent.tsx +155 -5
- package/public/manager/src/hooks/useDashboardView.ts +12 -1
- package/public/manager/src/hooks/useInstanceMessageEvents.ts +3 -1
- package/public/manager/src/jaw-ceo/JawCeoConsole.tsx +108 -0
- package/public/manager/src/jaw-ceo/JawCeoConsolePanels.tsx +331 -0
- package/public/manager/src/jaw-ceo/JawCeoSettingsPanel.tsx +94 -0
- package/public/manager/src/jaw-ceo/JawCeoTabs.tsx +28 -0
- package/public/manager/src/jaw-ceo/JawCeoVoiceOverlay.tsx +34 -0
- package/public/manager/src/jaw-ceo/JawCeoWorkbenchButton.tsx +61 -0
- package/public/manager/src/jaw-ceo/api.ts +169 -0
- package/public/manager/src/jaw-ceo/jaw-ceo-console.css +481 -0
- package/public/manager/src/jaw-ceo/jaw-ceo-virtual.css +24 -0
- package/public/manager/src/jaw-ceo/jaw-ceo.css +429 -0
- package/public/manager/src/jaw-ceo/types.ts +164 -0
- package/public/manager/src/jaw-ceo/useJawCeo.ts +235 -0
- package/public/manager/src/jaw-ceo/useJawCeoConsoleModel.ts +67 -0
- package/public/manager/src/jaw-ceo/useJawCeoDashboardBridge.tsx +76 -0
- package/public/manager/src/jaw-ceo/useJawCeoVirtualTimeline.ts +109 -0
- package/public/manager/src/jaw-ceo/useJawCeoVoice.ts +219 -0
- package/public/manager/src/jaw-ceo/voice-cues.ts +34 -0
- package/public/manager/src/jaw-ceo/voice-session.ts +87 -0
- package/public/manager/src/main.tsx +10 -0
- package/public/manager/src/manager-components.css +7 -0
- package/public/manager/src/manager-dashboard-board-sidebar-scroll.css +64 -0
- package/public/manager/src/manager-dashboard-reminders-parity.css +156 -0
- package/public/manager/src/manager-dashboard-reminders-priority.css +164 -0
- package/public/manager/src/manager-dashboard-reminders.css +482 -0
- package/public/manager/src/manager-dashboard-schedule.css +12 -0
- package/public/manager/src/manager-dashboard-settings.css +16 -2
- package/public/manager/src/manager-help.css +27 -0
- package/public/manager/src/manager-layout.css +19 -0
- package/public/manager/src/manager-notes.css +13 -4
- package/public/manager/src/manager-p0-1-1.css +3 -3
- package/public/manager/src/manager-polish.css +7 -3
- package/public/manager/src/manager-shortcuts.ts +116 -0
- package/public/manager/src/notes/MarkdownEditor.tsx +20 -3
- package/public/manager/src/notes/MarkdownPreview.tsx +10 -1
- package/public/manager/src/notes/NotesFrontmatterStrip.tsx +61 -0
- package/public/manager/src/notes/NotesQuickSwitcher.tsx +187 -0
- package/public/manager/src/notes/NotesSidebar.tsx +36 -14
- package/public/manager/src/notes/NotesWorkspace.tsx +63 -4
- package/public/manager/src/notes/frontmatter-preview.ts +16 -0
- package/public/manager/src/notes/notes-quick-switcher.css +151 -0
- package/public/manager/src/notes/notes-tags.css +292 -0
- package/public/manager/src/notes/notes-types.ts +4 -0
- package/public/manager/src/notes/rendering/MarkdownRenderer.tsx +67 -3
- package/public/manager/src/notes/rendering/MermaidBlock.tsx +11 -1
- package/public/manager/src/notes/useNoteDocument.ts +14 -2
- package/public/manager/src/notes/useNotesModel.ts +39 -2
- package/public/manager/src/notes/wiki-link-codemirror-completion.ts +55 -0
- package/public/manager/src/notes/wiki-link-rendering.ts +129 -0
- package/public/manager/src/notes/wiki-link-resolver.ts +220 -0
- package/public/manager/src/notes/wiki-link-suggestions.ts +153 -0
- package/public/manager/src/notes/wysiwyg/MilkdownWysiwygEditor.tsx +126 -79
- package/public/manager/src/notes/wysiwyg/WysiwygFrontmatterPanel.tsx +93 -0
- package/public/manager/src/notes/wysiwyg/milkdown-code-block-view.ts +11 -0
- package/public/manager/src/notes/wysiwyg/milkdown-editor-utils.ts +32 -0
- package/public/manager/src/notes/wysiwyg/milkdown-math.ts +9 -0
- package/public/manager/src/notes/wysiwyg/milkdown-wikilink-completion.ts +195 -0
- package/public/manager/src/notes/wysiwyg/milkdown-wikilink-plugin.ts +152 -0
- package/public/manager/src/notes/wysiwyg/milkdown-wysiwyg-types.ts +13 -0
- package/public/manager/src/notes/wysiwyg/wysiwyg-fixtures.ts +4 -0
- package/public/manager/src/notes/wysiwyg/wysiwyg-frontmatter.ts +155 -0
- package/public/manager/src/settings/pages/Embedding.tsx +382 -0
- package/public/manager/src/settings-embedding.css +168 -0
- package/public/manager/src/types.ts +12 -2
- package/scripts/install-officecli.sh +3 -1
- package/scripts/install-wsl.sh +65 -11
- package/public/dist/assets/MilkdownWysiwygEditor-DIebNZF7.js +0 -52
- package/public/dist/assets/app-CphocJzo.css +0 -1
- package/public/dist/assets/app-DJ8ys0j5.js +0 -32
- package/public/dist/assets/architecture-YZFGNWBL-VpdnYwVC.js +0 -1
- package/public/dist/assets/channel-xhp7drfS.js +0 -1
- package/public/dist/assets/chunk-426QAEUC-DadS7Aoc.js +0 -1
- package/public/dist/assets/chunk-55IACEB6-Cr281urY.js +0 -1
- package/public/dist/assets/chunk-QZHKN3VN-BYRU3VjK.js +0 -1
- package/public/dist/assets/classDiagram-6PBFFD2Q-Brk35lpl.js +0 -1
- package/public/dist/assets/classDiagram-v2-HSJHXN6E-CVZHK0Hg.js +0 -1
- package/public/dist/assets/dist-BsT5UdNP.js +0 -1
- package/public/dist/assets/dist-ClqO40BE.js +0 -1
- package/public/dist/assets/dist-CxeLAw2Y.js +0 -1
- package/public/dist/assets/dist-D2SH8nxa.js +0 -1
- package/public/dist/assets/dist-DfodGES_.js +0 -23
- package/public/dist/assets/dist-l9HH00ip.js +0 -1
- package/public/dist/assets/employees-RJ_wRL09.js +0 -33
- package/public/dist/assets/gitGraph-7Q5UKJZL-CBKRRUZJ.js +0 -1
- package/public/dist/assets/idb-cache-C5ilDI6r.js +0 -1
- package/public/dist/assets/info-OMHHGYJF-BqzRlF_E.js +0 -1
- package/public/dist/assets/insert-image-markdown-kk053MvN.js +0 -22
- package/public/dist/assets/javascript-DT-6B2IU.js +0 -1
- package/public/dist/assets/katex-CS4w7U2j.js +0 -1
- package/public/dist/assets/manager-DAe38I94.js +0 -25
- package/public/dist/assets/manager-fQR46YFa.css +0 -1
- package/public/dist/assets/memory-w3yQettQ.js +0 -1
- package/public/dist/assets/mermaid-loader-BEFIOoJn.js +0 -1
- package/public/dist/assets/mermaid.core-D0tXHgVC.js +0 -1
- package/public/dist/assets/packet-4T2RLAQJ-D7VqJs5z.js +0 -1
- package/public/dist/assets/pie-ZZUOXDRM-FbISwuBg.js +0 -1
- package/public/dist/assets/radar-PYXPWWZC-CLzM3m4R.js +0 -1
- package/public/dist/assets/render-KVGsbWj1.js +0 -31
- package/public/dist/assets/settings-DmUCo6lz.js +0 -1
- package/public/dist/assets/skills-SxG_nfwn.js +0 -1
- package/public/dist/assets/slash-commands-BxJkKdhB.js +0 -1
- package/public/dist/assets/stateDiagram-v2-QKLJ7IA2-DScAhoyh.js +0 -1
- package/public/dist/assets/treeView-SZITEDCU-HJajkeQK.js +0 -1
- package/public/dist/assets/treemap-W4RFUUIX-CsRQ896G.js +0 -1
- package/public/dist/assets/ui-LhD1VfQs.js +0 -1
- package/public/dist/assets/ui-kS1ZJfez.js +0 -143
- package/public/dist/assets/wardley-RL74JXVD-DMQFnlJp.js +0 -1
- package/public/dist/assets/ws-DVE3eWRj.js +0 -14
- /package/public/dist/assets/{agent-meta-DHddpWHQ.js → agent-meta-C1pQzExf.js} +0 -0
- /package/public/dist/assets/{apl-1H3fJ-n9.js → apl-HYRstREL.js} +0 -0
- /package/public/dist/assets/{asciiarmor-_1f4lqIZ.js → asciiarmor-BDXCrhAK.js} +0 -0
- /package/public/dist/assets/{asn1-B_wp1Pdt.js → asn1-CnSBhb0M.js} +0 -0
- /package/public/dist/assets/{asterisk-9Hxmlnbz.js → asterisk-AqV7rnIi.js} +0 -0
- /package/public/dist/assets/{brainfuck-C-PAVNT8.js → brainfuck-Dv46-iL9.js} +0 -0
- /package/public/dist/assets/{chunk-AGHRB4JF-EroDCcEL.js → chunk-AGHRB4JF-DIMn8T3_.js} +0 -0
- /package/public/dist/assets/{clike-DAVXDhrz.js → clike-GRffz5hY.js} +0 -0
- /package/public/dist/assets/{clojure-BIpQu4v4.js → clojure-DzgT_fqE.js} +0 -0
- /package/public/dist/assets/{cmake-YlOBEvgc.js → cmake-Co1237r5.js} +0 -0
- /package/public/dist/assets/{cobol-CU8lw6mH.js → cobol-rUaLketb.js} +0 -0
- /package/public/dist/assets/{coffeescript-DKG8mABL.js → coffeescript-DQyfHE86.js} +0 -0
- /package/public/dist/assets/{commonlisp-CbWboHR_.js → commonlisp-Buue1PGW.js} +0 -0
- /package/public/dist/assets/{constants-CYjOMbjf.js → constants-4A2GptQT.js} +0 -0
- /package/public/dist/assets/{crystal-WWs3Njra.js → crystal-BMWO0kOJ.js} +0 -0
- /package/public/dist/assets/{css-C9qNtc8P.js → css-DaxibPo5.js} +0 -0
- /package/public/dist/assets/{cypher-DglnWVZO.js → cypher-CmpGfiBR.js} +0 -0
- /package/public/dist/assets/{cytoscape.esm-BH6rhG9A.js → cytoscape.esm-zT8GwLFm.js} +0 -0
- /package/public/dist/assets/{d-DiNW6jcC.js → d-qmdtoIzU.js} +0 -0
- /package/public/dist/assets/{diff-PxsoX0eK.js → diff-B40m6u1h.js} +0 -0
- /package/public/dist/assets/{dist-BRFvH5ne.js → dist-BJyDhGpS.js} +0 -0
- /package/public/dist/assets/{dtd-BSiaE0hE.js → dtd-569ynYVj.js} +0 -0
- /package/public/dist/assets/{dylan-GcUJt0a5.js → dylan-Dxolp30i.js} +0 -0
- /package/public/dist/assets/{ebnf-DSIN9Ycp.js → ebnf-BZiX9Iq3.js} +0 -0
- /package/public/dist/assets/{ecl-LP9l5g0J.js → ecl-B8F1Q0oZ.js} +0 -0
- /package/public/dist/assets/{eiffel-CqpH5qaY.js → eiffel-Bv8Kvgh1.js} +0 -0
- /package/public/dist/assets/{elm-Dw3cJ7hz.js → elm-QWXQaIis.js} +0 -0
- /package/public/dist/assets/{erlang-DL9rpJep.js → erlang-CuDGzTGm.js} +0 -0
- /package/public/dist/assets/{fcl-DUmfTiJ7.js → fcl-BiFeqtHf.js} +0 -0
- /package/public/dist/assets/{forth-BSp-YOKj.js → forth-R7Uc2VcL.js} +0 -0
- /package/public/dist/assets/{fortran-BtIRmo6s.js → fortran-kDRG6BzW.js} +0 -0
- /package/public/dist/assets/{gas-okqIRy8I.js → gas-CyRkuC5T.js} +0 -0
- /package/public/dist/assets/{gherkin-DIlSAX9i.js → gherkin-BE0p00ey.js} +0 -0
- /package/public/dist/assets/{groovy-Dn6VQ7le.js → groovy-B_Sh3D1t.js} +0 -0
- /package/public/dist/assets/{haskell-DxXJRRAf.js → haskell-DWXgCy4o.js} +0 -0
- /package/public/dist/assets/{haxe-Cf0nItU4.js → haxe-xGxZ54Hv.js} +0 -0
- /package/public/dist/assets/{http-C3zQeWFV.js → http-CKv9cSBA.js} +0 -0
- /package/public/dist/assets/{idb-cache-CjnC_K0x.js → idb-cache-CZ3JdK8r.js} +0 -0
- /package/public/dist/assets/{idl-BvqkcZq5.js → idl-DmlI3XzS.js} +0 -0
- /package/public/dist/assets/{javascript-D1nkUbtD.js → javascript-j6r5uf_k.js} +0 -0
- /package/public/dist/assets/{jsx-runtime-DOs-BiPY.js → jsx-runtime-BjL7qHh8.js} +0 -0
- /package/public/dist/assets/{julia-CCUCEuMJ.js → julia-CQSp9Qa0.js} +0 -0
- /package/public/dist/assets/{livescript-DJ-6Zja2.js → livescript-CqvtVTlb.js} +0 -0
- /package/public/dist/assets/{locale-f397XJgr.js → locale-BHMJIzyw.js} +0 -0
- /package/public/dist/assets/{lua-CxE7YJaJ.js → lua-DazQKUZ0.js} +0 -0
- /package/public/dist/assets/{mathematica-oNEp3xtK.js → mathematica-DqhJVCs9.js} +0 -0
- /package/public/dist/assets/{mbox-AvjHxuWl.js → mbox-Dsyo1_UL.js} +0 -0
- /package/public/dist/assets/{mirc-CiJGiXcS.js → mirc-D7ThhoF-.js} +0 -0
- /package/public/dist/assets/{mllike-CV8oQtvy.js → mllike-EPZ6pqQD.js} +0 -0
- /package/public/dist/assets/{modelica-M4BAagfL.js → modelica-PU45hbqg.js} +0 -0
- /package/public/dist/assets/{mscgen-3QXoaqEl.js → mscgen-DEnh2Ojr.js} +0 -0
- /package/public/dist/assets/{mumps-BZoVj2VG.js → mumps-8GhR7rRa.js} +0 -0
- /package/public/dist/assets/{nginx-CBycrar_.js → nginx-CXwagpwp.js} +0 -0
- /package/public/dist/assets/{ntriples-BzBBWYSl.js → ntriples-BDQxxstw.js} +0 -0
- /package/public/dist/assets/{octave-BeIsGSMy.js → octave-2c90WnyU.js} +0 -0
- /package/public/dist/assets/{oz-DdNR63iD.js → oz-CO0rQ8EL.js} +0 -0
- /package/public/dist/assets/{pascal-BvmM4Y7z.js → pascal-CcVwOGeN.js} +0 -0
- /package/public/dist/assets/{path-utils-BuEEtj9w.js → path-utils-DySmCVZK.js} +0 -0
- /package/public/dist/assets/{perl-DFibaVbf.js → perl-B9UvGoAV.js} +0 -0
- /package/public/dist/assets/{pig-DyLcrln-.js → pig-Cd1f86ZJ.js} +0 -0
- /package/public/dist/assets/{powershell-CDOpEwhN.js → powershell-DV4cOnkJ.js} +0 -0
- /package/public/dist/assets/{preload-helper-CMqhPRCH.js → preload-helper-DuH3-WbD.js} +0 -0
- /package/public/dist/assets/{properties-Ju2yMwi3.js → properties-DaSVPhrZ.js} +0 -0
- /package/public/dist/assets/{protobuf-CfKx2SfO.js → protobuf-CEtFnY22.js} +0 -0
- /package/public/dist/assets/{puppet-dmjzqANu.js → puppet-Ca0DHfcW.js} +0 -0
- /package/public/dist/assets/{python-D9BZyYi1.js → python-ydzCwWG-.js} +0 -0
- /package/public/dist/assets/{q-C68SC4ma.js → q-Cl8kzQmk.js} +0 -0
- /package/public/dist/assets/{r-DP6KYC7N.js → r-DKZPJFrD.js} +0 -0
- /package/public/dist/assets/{rough.esm-BRGGwqOp.js → rough.esm-Dnk3WN4D.js} +0 -0
- /package/public/dist/assets/{rpm-DH2Z4z_v.js → rpm-D0lqeHoZ.js} +0 -0
- /package/public/dist/assets/{ruby-CwiqsW2a.js → ruby-RVmtmNDw.js} +0 -0
- /package/public/dist/assets/{sas-Btq5K1IU.js → sas-CEzodAB7.js} +0 -0
- /package/public/dist/assets/{scheme-BvG_RNGf.js → scheme-Pqmt6dh3.js} +0 -0
- /package/public/dist/assets/{settings-client-ajlwI-oK.js → settings-client-BN3XzwEo.js} +0 -0
- /package/public/dist/assets/{shell-waWIPbLL.js → shell-CLG3zy-u.js} +0 -0
- /package/public/dist/assets/{sieve-wIycBENd.js → sieve-C95IWC3O.js} +0 -0
- /package/public/dist/assets/{simple-mode-lHvrAolQ.js → simple-mode-B9ErAHMD.js} +0 -0
- /package/public/dist/assets/{smalltalk-SHoYQOtw.js → smalltalk-4cL-gRJb.js} +0 -0
- /package/public/dist/assets/{solr-D1Apesak.js → solr-CEA7oOx_.js} +0 -0
- /package/public/dist/assets/{sparql-BJozOrmN.js → sparql-w8kHjP6I.js} +0 -0
- /package/public/dist/assets/{spreadsheet-71jLW2fq.js → spreadsheet-C8bDfTgC.js} +0 -0
- /package/public/dist/assets/{sql-BaHU6gLz.js → sql-BrKz8968.js} +0 -0
- /package/public/dist/assets/{stex-BuVGLRtt.js → stex-xFw1nXeT.js} +0 -0
- /package/public/dist/assets/{stylus-Cqweyusm.js → stylus-gS_68vPk.js} +0 -0
- /package/public/dist/assets/{swift-Dx3m4XfV.js → swift-D5lXmY2d.js} +0 -0
- /package/public/dist/assets/{tcl-5yXjCZ8e.js → tcl-D-41REtC.js} +0 -0
- /package/public/dist/assets/{textile-9HVfsYV6.js → textile-D-Znao0j.js} +0 -0
- /package/public/dist/assets/{tiddlywiki-DlSf8L__.js → tiddlywiki-Co8lXJKd.js} +0 -0
- /package/public/dist/assets/{tiki-CDZRH0A4.js → tiki-DsiNRYMP.js} +0 -0
- /package/public/dist/assets/{toml-BgjfE0Y7.js → toml-CaVZGou4.js} +0 -0
- /package/public/dist/assets/{troff-B4ZrJIrZ.js → troff-DJvDVr-d.js} +0 -0
- /package/public/dist/assets/{ttcn-BJIiaYhP.js → ttcn-DAC92l4d.js} +0 -0
- /package/public/dist/assets/{ttcn-cfg-PZp_QKQg.js → ttcn-cfg-C0X-mYlr.js} +0 -0
- /package/public/dist/assets/{turtle--ft5y7IN.js → turtle-D6WHyCrL.js} +0 -0
- /package/public/dist/assets/{vb-DL6cecZs.js → vb-CEo_ccq3.js} +0 -0
- /package/public/dist/assets/{vbscript-Bb1xiv7E.js → vbscript-B_PKDV1v.js} +0 -0
- /package/public/dist/assets/{velocity-DKS9ZfbJ.js → velocity-ClKnKBLH.js} +0 -0
- /package/public/dist/assets/{vendor-utils-CWelFt6C.js → vendor-utils-IVTPs69f.js} +0 -0
- /package/public/dist/assets/{verilog-DSfhYFaT.js → verilog-Pnuspgw6.js} +0 -0
- /package/public/dist/assets/{vhdl-CFTSUmh9.js → vhdl-CZnyY4fs.js} +0 -0
- /package/public/dist/assets/{w3c-keyname-IiiZScED.js → w3c-keyname-DJlC6DTY.js} +0 -0
- /package/public/dist/assets/{webidl-C-IpKmuN.js → webidl-DHr762Dm.js} +0 -0
- /package/public/dist/assets/{xquery-9pXYEith.js → xquery-5-FiyUtN.js} +0 -0
- /package/public/dist/assets/{yacas-CqSYoSTv.js → yacas-GlqsXIeT.js} +0 -0
- /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,
|
|
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,
|
|
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
|
-
|
|
72
|
-
|
|
73
|
-
|
|
74
|
-
|
|
75
|
-
|
|
76
|
-
|
|
77
|
-
|
|
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
|
-
|
|
85
|
-
|
|
86
|
-
|
|
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
|
|
387
|
+
// ── Message action delegation ──
|
|
1170
388
|
export function initMsgCopy(): void {
|
|
1171
|
-
|
|
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
|
}
|