chainlesschain 0.162.48 → 0.162.49

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 (161) hide show
  1. package/package.json +2 -2
  2. package/src/assets/web-panel/assets/{AIOps-BeMOUkMq.js → AIOps-DmdtNmWd.js} +1 -1
  3. package/src/assets/web-panel/assets/{ActionButton-DrRegmIt.js → ActionButton-DkuYVafg.js} +1 -1
  4. package/src/assets/web-panel/assets/{Analytics-BM7hvUn8.js → Analytics-Ba_h8Tub.js} +3 -3
  5. package/src/assets/web-panel/assets/{AppLayout-BfLLYSz_.js → AppLayout-yb8Zm9MX.js} +5 -5
  6. package/src/assets/web-panel/assets/{Audit-BpiPR-rs.js → Audit-BGjex2fm.js} +1 -1
  7. package/src/assets/web-panel/assets/{Backup-BLq4IRI9.js → Backup-IFQ2hOF2.js} +1 -1
  8. package/src/assets/web-panel/assets/{BaseInput-Bv89IFrM.js → BaseInput-W8AkPkrV.js} +1 -1
  9. package/src/assets/web-panel/assets/{Chat-BUM9MdnH.js → Chat-BgI7t-iW.js} +6 -6
  10. package/src/assets/web-panel/assets/ChatBubbleRenderer-CfpKEQUF.js +1 -0
  11. package/src/assets/web-panel/assets/{Checkbox-82jih_zU.js → Checkbox-DuWsZP4g.js} +1 -1
  12. package/src/assets/web-panel/assets/{Codegen-ygmM1mNL.js → Codegen-DyoTNmYg.js} +1 -1
  13. package/src/assets/web-panel/assets/{Col-CqXc8_ft.js → Col-DttmlDRk.js} +1 -1
  14. package/src/assets/web-panel/assets/{Community-CqPPfyR3.js → Community-D9nnIdKn.js} +1 -1
  15. package/src/assets/web-panel/assets/{Compact-LR8Z206-.js → Compact-C8KVQaHb.js} +1 -1
  16. package/src/assets/web-panel/assets/{Compliance-D4alm_Gx.js → Compliance-R2owqgjj.js} +1 -1
  17. package/src/assets/web-panel/assets/{Cowork-D-DLVanT.js → Cowork-DwGMMjRn.js} +3 -3
  18. package/src/assets/web-panel/assets/{Cron-UDtmjvd4.js → Cron-BSTcN_lK.js} +2 -2
  19. package/src/assets/web-panel/assets/{Crosschain-iqbVDPNE.js → Crosschain-CTNuIbFD.js} +1 -1
  20. package/src/assets/web-panel/assets/{DID-DzdUfzc9.js → DID-CgApGsFP.js} +2 -2
  21. package/src/assets/web-panel/assets/{Dashboard-DvKp1skY.js → Dashboard-D_OJ3UN5.js} +2 -2
  22. package/src/assets/web-panel/assets/{Dropdown-S6ORlfef.js → Dropdown-B84Jwra_.js} +1 -1
  23. package/src/assets/web-panel/assets/{EmailListRenderer-mJViHjiq.js → EmailListRenderer-Bv-YO-6y.js} +1 -1
  24. package/src/assets/web-panel/assets/{FamilyGuardDashboard-A4Lu5m_e.js → FamilyGuardDashboard-drgZ408Y.js} +1 -1
  25. package/src/assets/web-panel/assets/{Federation-Bv-6737r.js → Federation-CtzFkdW2.js} +1 -1
  26. package/src/assets/web-panel/assets/{FormItemContext-DNcZoiWu.js → FormItemContext-BFAvNhl9.js} +1 -1
  27. package/src/assets/web-panel/assets/{GenericCardRenderer-B_a0l9U4.js → GenericCardRenderer-DnuEyz_l.js} +1 -1
  28. package/src/assets/web-panel/assets/{Git-BbVAsh_N.js → Git-jlHajmRJ.js} +2 -2
  29. package/src/assets/web-panel/assets/{Governance-ChxUMZEU.js → Governance-DmJC7PGL.js} +1 -1
  30. package/src/assets/web-panel/assets/{Inference-Buu_bAQ-.js → Inference-B-u7xD2n.js} +1 -1
  31. package/src/assets/web-panel/assets/{KnowledgeGraph-BMmMjxEu.js → KnowledgeGraph-BaYCA2Cd.js} +1 -1
  32. package/src/assets/web-panel/assets/{Logs-BeGHcPEC.js → Logs-DTNYQWfp.js} +2 -2
  33. package/src/assets/web-panel/assets/{Marketplace-CP2Qg_xJ.js → Marketplace-CUu1xYvo.js} +1 -1
  34. package/src/assets/web-panel/assets/{McpTools-C42B4JYb.js → McpTools-BmoeTyrC.js} +5 -5
  35. package/src/assets/web-panel/assets/{Memory-D5VeitFY.js → Memory-DxTU3QU7.js} +2 -2
  36. package/src/assets/web-panel/assets/{MobileBridge-BouoJ2FQ.js → MobileBridge-CpcOlKAD.js} +2 -2
  37. package/src/assets/web-panel/assets/{MobileProjects-mAJsyk7U.js → MobileProjects-Bjh_z16l.js} +1 -1
  38. package/src/assets/web-panel/assets/{Mtc-Chn6ES5y.js → Mtc-LfxwOm0x.js} +5 -5
  39. package/src/assets/web-panel/assets/{MtcAudit-D0_Q6GEn.js → MtcAudit-D6A9Gjkh.js} +2 -2
  40. package/src/assets/web-panel/assets/{Multisig-BWaCi_wo.js → Multisig-Ch_jofPV.js} +3 -3
  41. package/src/assets/web-panel/assets/{NLProgramming-CoSEiroa.js → NLProgramming-Bkvogg0I.js} +1 -1
  42. package/src/assets/web-panel/assets/{Notes-CiqCbDw3.js → Notes-C5t5Xihm.js} +3 -3
  43. package/src/assets/web-panel/assets/{NotificationSettings-BXmzKL9F.js → NotificationSettings-CTpDUNCb.js} +1 -1
  44. package/src/assets/web-panel/assets/OrderTableRenderer-ST2lr-Bi.js +1 -0
  45. package/src/assets/web-panel/assets/{Organization-CAVgSxyI.js → Organization-Dr37BaXa.js} +4 -4
  46. package/src/assets/web-panel/assets/{Overflow-BIVUo8YB.js → Overflow-ZGjsdP7N.js} +1 -1
  47. package/src/assets/web-panel/assets/{P2P-DX-Ov-eJ.js → P2P-bWJU5Vxd.js} +2 -2
  48. package/src/assets/web-panel/assets/{PdhVaultBrowser-6clRu-J6.js → PdhVaultBrowser-BRVoW-ye.js} +3 -3
  49. package/src/assets/web-panel/assets/{Permissions-CANl-V55.js → Permissions-BOSnFZaC.js} +4 -4
  50. package/src/assets/web-panel/assets/{PersonalDataHub-BE90gjUO.js → PersonalDataHub-X4SgjP6P.js} +2 -2
  51. package/src/assets/web-panel/assets/{Pipeline-Ck8lV8Pn.js → Pipeline-DoJhB9bj.js} +1 -1
  52. package/src/assets/web-panel/assets/{Privacy-BBYUDK4T.js → Privacy-OM9lDj-R.js} +1 -1
  53. package/src/assets/web-panel/assets/{ProjectInit-DcbOrnbt.js → ProjectInit-BXQEOmLn.js} +2 -2
  54. package/src/assets/web-panel/assets/{ProjectSettings-DZ6-whxP.js → ProjectSettings-DBXo3K5u.js} +2 -2
  55. package/src/assets/web-panel/assets/Projects-CJ4DBJlS.js +1 -0
  56. package/src/assets/web-panel/assets/{Providers-VWoO_Y9u.js → Providers-Tk9SawmO.js} +1 -1
  57. package/src/assets/web-panel/assets/{QuickAsk-RJfSXWYg.js → QuickAsk-DRI1-nTC.js} +1 -1
  58. package/src/assets/web-panel/assets/{Recommend-Y7MWWkXa.js → Recommend-DtrIVTu9.js} +1 -1
  59. package/src/assets/web-panel/assets/{Reputation-nG8nut3l.js → Reputation-DkH8ImwZ.js} +1 -1
  60. package/src/assets/web-panel/assets/{Row-DAio6Dx2.js → Row-DpA9dlvi.js} +1 -1
  61. package/src/assets/web-panel/assets/{RssFeed-DwY72Lc7.js → RssFeed-DV3OhxWd.js} +2 -2
  62. package/src/assets/web-panel/assets/{Search-qGG6AUWY.js → Search-QxdntiQx.js} +1 -1
  63. package/src/assets/web-panel/assets/{Security-Djsmom8n.js → Security-CGuEnrD2.js} +3 -3
  64. package/src/assets/web-panel/assets/{Services-DTEgHkUO.js → Services-BvwSSD8b.js} +2 -2
  65. package/src/assets/web-panel/assets/{Skeleton-R5xY0J9Y.js → Skeleton-sx_8L3-5.js} +1 -1
  66. package/src/assets/web-panel/assets/{Skills-CFaRLO3o.js → Skills-dWOwxRsu.js} +1 -1
  67. package/src/assets/web-panel/assets/{Sla-B5bzoY1I.js → Sla-zxXnfKrT.js} +1 -1
  68. package/src/assets/web-panel/assets/{SpeechSettings-B_al6SiQ.js → SpeechSettings-CmFlcNjr.js} +1 -1
  69. package/src/assets/web-panel/assets/{SyncSettings-DkUU63oJ.js → SyncSettings-BeXeqURL.js} +2 -2
  70. package/src/assets/web-panel/assets/Tasks-iImd8xSO.js +1 -0
  71. package/src/assets/web-panel/assets/{Templates-CcYJxTNB.js → Templates-DlgR3XFH.js} +1 -1
  72. package/src/assets/web-panel/assets/{Tenant-B0_sIpl2.js → Tenant-0P8HgQaM.js} +1 -1
  73. package/src/assets/web-panel/assets/Terminal-B5VDEEHD.js +3 -0
  74. package/src/assets/web-panel/assets/{TimelineRenderer-DuMkErBx.js → TimelineRenderer-hbO7agZs.js} +1 -1
  75. package/src/assets/web-panel/assets/{Tokens-BXjPA6rV.js → Tokens-CsmhgTBO.js} +1 -1
  76. package/src/assets/web-panel/assets/{Trigger-BMAAx3Uu.js → Trigger-DnaF_2PP.js} +1 -1
  77. package/src/assets/web-panel/assets/{Trust-BPeNXpsi.js → Trust-C1oafGj1.js} +1 -1
  78. package/src/assets/web-panel/assets/{UkeySign-A0qabV14.js → UkeySign-eLL4DOmC.js} +1 -1
  79. package/src/assets/web-panel/assets/{VideoEditing-BMLfYykd.js → VideoEditing-CX45sVq7.js} +1 -1
  80. package/src/assets/web-panel/assets/{Wallet-BfDI3zjs.js → Wallet-aWPqpHdQ.js} +4 -4
  81. package/src/assets/web-panel/assets/{WebAuthn-BCnZEScW.js → WebAuthn-DMYV1MAo.js} +5 -5
  82. package/src/assets/web-panel/assets/{WorkflowEditor-DvtS5Asf.js → WorkflowEditor-D9uRIJvH.js} +1 -1
  83. package/src/assets/web-panel/assets/{chat-CDJZtBM7.js → chat-BmWYfCxG.js} +1 -1
  84. package/src/assets/web-panel/assets/{colors-fbH1Saco.js → colors-DqvTCkBe.js} +1 -1
  85. package/src/assets/web-panel/assets/{compact-item-CuZFa9l8.js → compact-item-Bh0L0ejI.js} +1 -1
  86. package/src/assets/web-panel/assets/{createContext-CQuPOqqD.js → createContext-r2qgp1mn.js} +1 -1
  87. package/src/assets/web-panel/assets/devWarning-CusWDjWW.js +1 -0
  88. package/src/assets/web-panel/assets/{hasIn-CDZlDrhJ.js → hasIn-BcffXa-S.js} +1 -1
  89. package/src/assets/web-panel/assets/{index-CAzMdnkI.js → index-585fuGAN.js} +1 -1
  90. package/src/assets/web-panel/assets/{index-BhZkMMey.js → index-B3mmDuOv.js} +1 -1
  91. package/src/assets/web-panel/assets/{index-BSy0noke.js → index-B6pAm1iJ.js} +1 -1
  92. package/src/assets/web-panel/assets/{index-DjgZRX0_.js → index-B7z0qK1W.js} +1 -1
  93. package/src/assets/web-panel/assets/{index-Bt1j0mjJ.js → index-BCBqTRWH.js} +1 -1
  94. package/src/assets/web-panel/assets/{index-CnfR7qmj.js → index-BLN-neIf.js} +1 -1
  95. package/src/assets/web-panel/assets/{index-BKaue5Pv.js → index-BMn_luHQ.js} +1 -1
  96. package/src/assets/web-panel/assets/{index-14gri6Vh.js → index-BQlAPNSU.js} +1 -1
  97. package/src/assets/web-panel/assets/{index--SCX46Az.js → index-BXXxkeij.js} +1 -1
  98. package/src/assets/web-panel/assets/{index-CmeO_DfK.js → index-BXae4ZyX.js} +1 -1
  99. package/src/assets/web-panel/assets/{index-BNEG9-EK.js → index-BbMox24t.js} +1 -1
  100. package/src/assets/web-panel/assets/{index-DNN-xBWV.js → index-BeV-KoQl.js} +1 -1
  101. package/src/assets/web-panel/assets/index-BhYltBvN.js +1 -0
  102. package/src/assets/web-panel/assets/{index-ycBlRXAf.js → index-Bma_yHcC.js} +1 -1
  103. package/src/assets/web-panel/assets/{index-CkhudJH0.js → index-BtyXyl3t.js} +1 -1
  104. package/src/assets/web-panel/assets/{index-BJCXJCUA.js → index-C2-02rrp.js} +1 -1
  105. package/src/assets/web-panel/assets/{index-BzIDfObk.js → index-C9nh3ANl.js} +1 -1
  106. package/src/assets/web-panel/assets/{index-DwAiu9LT.js → index-CAlxkpnv.js} +1 -1
  107. package/src/assets/web-panel/assets/{index-CzeRvhia.js → index-CCyB-RK5.js} +3 -3
  108. package/src/assets/web-panel/assets/{index-CD9ml9ZJ.js → index-CDR3GmaO.js} +1 -1
  109. package/src/assets/web-panel/assets/{index-Dkuecn17.js → index-CJt0iuep.js} +1 -1
  110. package/src/assets/web-panel/assets/{index-D365qmj8.js → index-CKnEtlZD.js} +1 -1
  111. package/src/assets/web-panel/assets/{index-Bz5_6E63.js → index-COrfHebA.js} +1 -1
  112. package/src/assets/web-panel/assets/{index-CNmJrCxV.js → index-CY8RXaZR.js} +1 -1
  113. package/src/assets/web-panel/assets/index-CZiIHw4e.js +1 -0
  114. package/src/assets/web-panel/assets/{index-kGzPbvry.js → index-Cbj6C3pA.js} +1 -1
  115. package/src/assets/web-panel/assets/{index-WDQkyh-E.js → index-Ct8qhPZe.js} +1 -1
  116. package/src/assets/web-panel/assets/{index-COSyGm80.js → index-DPFT7J7I.js} +1 -1
  117. package/src/assets/web-panel/assets/{index-DGAK9Dj4.js → index-DRXcGa5y.js} +1 -1
  118. package/src/assets/web-panel/assets/{index-CQ5FVEji.js → index-DW1y18GR.js} +1 -1
  119. package/src/assets/web-panel/assets/{index-BdORz0iZ.js → index-DWlDSE0F.js} +1 -1
  120. package/src/assets/web-panel/assets/{index-oe9ZPRtQ.js → index-Dcjol7ot.js} +1 -1
  121. package/src/assets/web-panel/assets/{index-Deqod8La.js → index-De36_UgR.js} +1 -1
  122. package/src/assets/web-panel/assets/{index-BivMeInw.js → index-DutDlDUF.js} +1 -1
  123. package/src/assets/web-panel/assets/{index-Cr_shi_7.js → index-DxajFkK2.js} +1 -1
  124. package/src/assets/web-panel/assets/{index-DhFyStIG.js → index-S4E77Aer.js} +1 -1
  125. package/src/assets/web-panel/assets/{index-C6epsHef.js → index-U86pxDyR.js} +1 -1
  126. package/src/assets/web-panel/assets/{index-DvBgQoaw.js → index-XwbSqOB2.js} +1 -1
  127. package/src/assets/web-panel/assets/{index-CdiNFWPp.js → index-h4O0AcBt.js} +1 -1
  128. package/src/assets/web-panel/assets/{initDefaultProps-DHRWNV-y.js → initDefaultProps-C1d8I-BX.js} +1 -1
  129. package/src/assets/web-panel/assets/{motion-MJ2jhdVO.js → motion-Dq7fiy4Y.js} +1 -1
  130. package/src/assets/web-panel/assets/{move-Bly0QFE5.js → move-Bqb2dySM.js} +1 -1
  131. package/src/assets/web-panel/assets/{omit-DkoMB0pZ.js → omit-BUYqb4My.js} +1 -1
  132. package/src/assets/web-panel/assets/{pickAttrs-9M-gsXIc.js → pickAttrs-DeytiKlZ.js} +1 -1
  133. package/src/assets/web-panel/assets/{placementArrow-Bnht1xci.js → placementArrow-xrXZWCqG.js} +1 -1
  134. package/src/assets/web-panel/assets/{responsiveObserve-B0Cn1i64.js → responsiveObserve-CcL2K-YY.js} +1 -1
  135. package/src/assets/web-panel/assets/{slide-BdI4DDyM.js → slide-DmCWaic7.js} +1 -1
  136. package/src/assets/web-panel/assets/{statusUtils-DdBktcMD.js → statusUtils-CqNrFif7.js} +1 -1
  137. package/src/assets/web-panel/assets/{styleChecker-aYEwS4Pw.js → styleChecker-C436m5Xy.js} +1 -1
  138. package/src/assets/web-panel/assets/{useFlexGapSupport-y8cUyKiP.js → useFlexGapSupport-CVhutCN8.js} +1 -1
  139. package/src/assets/web-panel/assets/{useFs-Bk1SrPFp.js → useFs-DUd49Bui.js} +1 -1
  140. package/src/assets/web-panel/assets/{usePersonalDataHub-Dp04zAB3.js → usePersonalDataHub-fuS9raic.js} +1 -1
  141. package/src/assets/web-panel/assets/{vnode-B52a38TC.js → vnode-C3kmDmk-.js} +1 -1
  142. package/src/assets/web-panel/assets/{zoom-DFwyL43U.js → zoom-hX-F1dT-.js} +1 -1
  143. package/src/assets/web-panel/index.html +1 -1
  144. package/src/lib/agent-core.js +2 -0
  145. package/src/lib/mcp-client.js +1 -0
  146. package/src/repl/agent-repl.js +179 -1
  147. package/src/repl/config-summary.js +59 -0
  148. package/src/repl/conversation-export.js +133 -0
  149. package/src/repl/doctor-status.js +114 -0
  150. package/src/repl/memory-status.js +45 -0
  151. package/src/repl/permissions-status.js +51 -0
  152. package/src/repl/recent-sessions.js +46 -0
  153. package/src/repl/session-cost.js +119 -0
  154. package/src/assets/web-panel/assets/ChatBubbleRenderer-rH_t53ZW.js +0 -1
  155. package/src/assets/web-panel/assets/OrderTableRenderer-BhlKyGrE.js +0 -1
  156. package/src/assets/web-panel/assets/Projects-DO25SEFT.js +0 -1
  157. package/src/assets/web-panel/assets/Tasks-C2y4XdrQ.js +0 -1
  158. package/src/assets/web-panel/assets/Terminal-B_waZb0O.js +0 -3
  159. package/src/assets/web-panel/assets/devWarning-mhGhHpNs.js +0 -1
  160. package/src/assets/web-panel/assets/index-DGBjZXvW.js +0 -1
  161. package/src/assets/web-panel/assets/index-DujMkFuc.js +0 -1
@@ -0,0 +1,45 @@
1
+ /**
2
+ * `/memory` REPL command (Claude-Code parity) — show the project-memory files
3
+ * the agent auto-loads into its system prompt (the cc.md > CLAUDE.md > AGENTS.md
4
+ * hierarchy + @imports + path-scoped rules), so "why does it know/behave like
5
+ * this?" is answerable in-session. Distinct from `#` (append a note to cc.md)
6
+ * and `cc memory recall` (the scoped MemoryStore).
7
+ *
8
+ * Pure: renders a loadProjectInstructions() result. The REPL does the I/O.
9
+ */
10
+
11
+ /**
12
+ * @param {{files?:Array<{path:string,scope:string,bytes?:number,truncated?:boolean}>,warnings?:string[]}} loaded
13
+ * @param {{enabled?:boolean}} [opts] enabled=false when CC_PROJECT_MEMORY=0
14
+ * @returns {string}
15
+ */
16
+ export function renderMemoryFiles(loaded, { enabled = true } = {}) {
17
+ const files = Array.isArray(loaded?.files) ? loaded.files : [];
18
+ const warnings = Array.isArray(loaded?.warnings) ? loaded.warnings : [];
19
+ const lines = ["Project memory (auto-loaded into the system prompt):"];
20
+
21
+ if (!enabled) {
22
+ lines.push(
23
+ " ⚠ disabled via CC_PROJECT_MEMORY=0 — the files below are NOT loaded this session",
24
+ );
25
+ }
26
+
27
+ if (files.length === 0) {
28
+ lines.push(" (none found — run `cc init` to generate a cc.md)");
29
+ return lines.join("\n");
30
+ }
31
+
32
+ let total = 0;
33
+ for (const f of files) {
34
+ const bytes = Number(f.bytes) || 0;
35
+ total += bytes;
36
+ lines.push(
37
+ ` [${String(f.scope || "?").padEnd(7)}] ${f.path} ${bytes}B${f.truncated ? " (truncated)" : ""}`,
38
+ );
39
+ }
40
+ lines.push(
41
+ ` ${files.length} file(s), ${total}B total · precedence cc.md > CLAUDE.md > AGENTS.md`,
42
+ );
43
+ for (const w of warnings) lines.push(` ⚠ ${w}`);
44
+ return lines.join("\n");
45
+ }
@@ -0,0 +1,51 @@
1
+ /**
2
+ * `/permissions` REPL command (Claude-Code parity) — show the allow / ask /
3
+ * deny rules in effect for THIS session (loaded from .claude/settings.json by
4
+ * settings-loader, applied by permission-rules in agent-core's executeTool).
5
+ * These rules are otherwise invisible mid-session; surfacing them tells the
6
+ * user exactly what the agent can do unprompted, what it asks about, and what
7
+ * is blocked.
8
+ *
9
+ * Pure: takes the effective rules ({allow,ask,deny}) + the contributing source
10
+ * files, returns plain text. The REPL does the I/O.
11
+ */
12
+
13
+ function pushGroup(lines, label, marker, rules) {
14
+ const list = Array.isArray(rules) ? rules : [];
15
+ if (list.length === 0) return;
16
+ lines.push(` ${label}:`);
17
+ for (const r of list) lines.push(` ${marker} ${r}`);
18
+ }
19
+
20
+ /**
21
+ * @param {{allow?:string[],ask?:string[],deny?:string[]}|null} rules
22
+ * @param {{files?:string[]}} [opts]
23
+ * @returns {string}
24
+ */
25
+ export function renderPermissions(rules, { files = [] } = {}) {
26
+ const r = rules || { allow: [], ask: [], deny: [] };
27
+ const count =
28
+ (r.allow?.length || 0) + (r.ask?.length || 0) + (r.deny?.length || 0);
29
+
30
+ if (count === 0) {
31
+ return [
32
+ "Permission rules: none configured.",
33
+ " Tools run under the default gate — dangerous shell commands still",
34
+ " always require approval. Add rules in .claude/settings.json or via",
35
+ " `cc permissions add`.",
36
+ ].join("\n");
37
+ }
38
+
39
+ const lines = ["Permission rules (effective this session):"];
40
+ // Render most-restrictive first to mirror the deny > ask > allow precedence.
41
+ pushGroup(lines, "deny (blocked)", "✗", r.deny);
42
+ pushGroup(lines, "ask (prompt first)", "?", r.ask);
43
+ pushGroup(lines, "allow (auto-approved)", "✓", r.allow);
44
+ if (Array.isArray(files) && files.length) {
45
+ lines.push(` sources: ${files.join(", ")}`);
46
+ }
47
+ lines.push(
48
+ " precedence: deny > ask > allow · dangerous shell commands are always denied",
49
+ );
50
+ return lines.join("\n");
51
+ }
@@ -0,0 +1,46 @@
1
+ /**
2
+ * `/sessions` REPL command — list recent RESUMABLE conversations (Claude-Code
3
+ * `/resume` parity, read-only half + mirrors the VS Code panel's /sessions).
4
+ * Where `/session` shows the CURRENT session, this lists past ones (across the
5
+ * DB + JSONL stores, via lib/recent-session.js listRecentSessions) with the
6
+ * ids you can pass to `cc agent --resume <id>`.
7
+ *
8
+ * Pure: renders the listRecentSessions row shape
9
+ * { id, title?, message_count?, updated_at?, _store? }
10
+ * The REPL gathers the rows + supplies the current id.
11
+ */
12
+
13
+ /**
14
+ * @param {Array} sessions listRecentSessions() rows
15
+ * @param {{currentId?:string, limit?:number}} [opts]
16
+ * @returns {string}
17
+ */
18
+ export function renderRecentSessions(sessions, opts = {}) {
19
+ const limit = opts.limit || 15;
20
+ const list = Array.isArray(sessions) ? sessions : [];
21
+ if (list.length === 0) {
22
+ return "No recent sessions found. Start chatting, or resume one later with `cc agent --resume <id>`.";
23
+ }
24
+
25
+ const lines = ["Recent sessions (resume with `cc agent --resume <id>`):"];
26
+ for (const s of list.slice(0, limit)) {
27
+ const id = String(s?.id || "");
28
+ if (!id) continue;
29
+ const current =
30
+ opts.currentId && id === opts.currentId ? " ← current" : "";
31
+ const store = s._store ? `[${s._store}]` : "";
32
+ const msgs = Number.isFinite(s.message_count)
33
+ ? `${s.message_count} msgs`
34
+ : "";
35
+ const when = s.updated_at
36
+ ? String(s.updated_at).slice(0, 19).replace("T", " ")
37
+ : "";
38
+ const title = s.title && s.title !== "Untitled" ? ` — ${s.title}` : "";
39
+ const meta = [store, msgs, when].filter(Boolean).join(" · ");
40
+ lines.push(` ${id.slice(0, 12)}${current} ${meta}${title}`);
41
+ }
42
+ if (list.length > limit) {
43
+ lines.push(` … +${list.length - limit} more`);
44
+ }
45
+ return lines.join("\n");
46
+ }
@@ -0,0 +1,119 @@
1
+ /**
2
+ * `/cost` REPL command — running token spend + estimated $ for the LIVE agent
3
+ * session (Claude-Code parity). Where `cc cost` reads persisted JSONL usage,
4
+ * this accumulates each turn's usage events IN MEMORY, so it reflects exactly
5
+ * this conversation and works even for anonymous (non-persisted) sessions.
6
+ *
7
+ * Pure and dependency-light (only the shared pricing lib); the REPL feeds it
8
+ * usage events and prints the rendered string.
9
+ */
10
+ import { priceRollup, mergePricing } from "../lib/llm-pricing.js";
11
+
12
+ function num(n) {
13
+ const v = Number(n);
14
+ return Number.isFinite(v) ? v : 0;
15
+ }
16
+
17
+ function fmtUsd(n) {
18
+ const v = Number(n) || 0;
19
+ if (v === 0) return "$0.00";
20
+ if (v < 0.01) return `$${v.toFixed(6)}`;
21
+ return `$${v.toFixed(4)}`;
22
+ }
23
+
24
+ /** A fresh accumulator: running totals + per provider/model rows. */
25
+ export function newCostStore() {
26
+ return {
27
+ total: { inputTokens: 0, outputTokens: 0, totalTokens: 0, calls: 0 },
28
+ byKey: new Map(),
29
+ };
30
+ }
31
+
32
+ /**
33
+ * Fold the REPL's per-turn usage events ([{ provider, model, usage }]) into the
34
+ * store. Each `usage` is the raw LLM usage object (input_tokens/output_tokens/…
35
+ * with provider-variant aliases). Zero-token events are ignored. Mutates and
36
+ * returns the store.
37
+ */
38
+ export function addUsage(store, events) {
39
+ const s = store || newCostStore();
40
+ for (const ev of Array.isArray(events) ? events : []) {
41
+ const u = ev && ev.usage ? ev.usage : null;
42
+ if (!u) continue;
43
+ const inT = num(u.input_tokens ?? u.prompt_tokens ?? u.inputTokens);
44
+ const outT = num(u.output_tokens ?? u.completion_tokens ?? u.outputTokens);
45
+ const totT = num(u.total_tokens ?? u.totalTokens ?? inT + outT);
46
+ if (inT === 0 && outT === 0 && totT === 0) continue;
47
+ s.total.inputTokens += inT;
48
+ s.total.outputTokens += outT;
49
+ s.total.totalTokens += totT;
50
+ s.total.calls += 1;
51
+ const key = `${ev.provider || "?"}/${ev.model || "?"}`;
52
+ let row = s.byKey.get(key);
53
+ if (!row) {
54
+ row = {
55
+ provider: ev.provider || null,
56
+ model: ev.model || null,
57
+ inputTokens: 0,
58
+ outputTokens: 0,
59
+ totalTokens: 0,
60
+ calls: 0,
61
+ };
62
+ s.byKey.set(key, row);
63
+ }
64
+ row.inputTokens += inT;
65
+ row.outputTokens += outT;
66
+ row.totalTokens += totT;
67
+ row.calls += 1;
68
+ }
69
+ return s;
70
+ }
71
+
72
+ /** Snapshot the store as the `{ total, byModel[] }` aggregate priceRollup wants. */
73
+ export function costAggregate(store) {
74
+ const s = store || newCostStore();
75
+ return {
76
+ total: { ...s.total },
77
+ byModel: Array.from(s.byKey.values()).sort(
78
+ (a, b) => b.totalTokens - a.totalTokens,
79
+ ),
80
+ };
81
+ }
82
+
83
+ /**
84
+ * Render the live session cost. `pricingOverrides` is typically
85
+ * `config.llm.pricing`. Returns plain text (the REPL does the I/O).
86
+ */
87
+ export function renderSessionCost(store, { pricingOverrides } = {}) {
88
+ const agg = costAggregate(store);
89
+ if (agg.total.calls === 0) {
90
+ return "Session cost: no LLM calls yet this session.";
91
+ }
92
+ const table = mergePricing(pricingOverrides);
93
+ const result = priceRollup(agg, { table });
94
+ const lines = [];
95
+ lines.push("Session cost (estimated):");
96
+ lines.push(
97
+ ` total: ${fmtUsd(result.cost.totalCost)} USD ` +
98
+ `(${result.total.totalTokens.toLocaleString()} tokens, ${result.total.calls} calls)`,
99
+ );
100
+ for (const row of result.byModel) {
101
+ const provider = (row.provider || "?").padEnd(10);
102
+ const model = (row.model || "?").padEnd(24);
103
+ const tokens = `in=${row.inputTokens} out=${row.outputTokens}`;
104
+ const price = row.free
105
+ ? "free (local)"
106
+ : row.matched
107
+ ? fmtUsd(row.cost)
108
+ : "unpriced";
109
+ lines.push(` ${provider} ${model} ${price} ${tokens}`);
110
+ }
111
+ if (result.unpriced.length > 0) {
112
+ lines.push(
113
+ ` note: ${result.unpriced.length} model(s) have no rate — ` +
114
+ "tokens excluded from total. Add rates via config: llm.pricing.",
115
+ );
116
+ }
117
+ lines.push(" prices are estimates of public list rates (USD/1M tokens).");
118
+ return lines.join("\n");
119
+ }
@@ -1 +0,0 @@
1
- import{I as v,P as u,J as S,U as n,R as a,S as y,K as C,Q as x,V as w,_ as B,b as s}from"./vendor-BvqAck49.js";import{_ as k}from"./index-CzeRvhia.js";import"./icons-DP3uiYxy.js";const N={__name:"ChatBubbleRenderer",props:{event:{type:Object,required:!0}},setup(c,{expose:d}){d();const t=c,r=s(()=>{const e=t.event.content||{};return e.text||e.body||e.message||e.title||JSON.stringify(e).slice(0,200)}),i=s(()=>{const e=t.event.content||{};return e.from||e.sender||e.senderName||t.event.actor||"(unknown)"}),l=s(()=>{const e=(t.event.actor||"").toLowerCase();return e.includes("self")||e==="me"||e.endsWith("_self")}),o=s(()=>{const e=t.event.source.adapter||"";return e.startsWith("messaging-qq")?"magenta":e==="wechat"?"green":"blue"}),m=s(()=>{if(!t.event.occurredAt)return"";try{const e=new Date(t.event.occurredAt),p=e.getFullYear(),f=String(e.getMonth()+1).padStart(2,"0"),g=String(e.getDate()).padStart(2,"0"),b=String(e.getHours()).padStart(2,"0"),h=String(e.getMinutes()).padStart(2,"0");return`${p}-${f}-${g} ${b}:${h}`}catch{return""}}),_={props:t,messageText:r,actorLabel:i,isMine:l,adapterColor:o,formattedTime:m,computed:s};return Object.defineProperty(_,"__isScriptSetup",{enumerable:!1,value:!0}),_}},T={class:"bubble"},M={class:"meta"},R={class:"actor"},V={class:"time"},q={class:"body"};function D(c,d,t,r,i,l){const o=v("a-tag");return u(),S("div",{class:B(["chat-row",{mine:r.isMine}])},[n("div",T,[n("div",M,[n("span",R,a(r.actorLabel),1),n("span",V,a(r.formattedTime),1)]),n("div",q,a(r.messageText),1),t.event.source.adapter?(u(),y(o,{key:0,class:"src",color:r.adapterColor},{default:C(()=>[x(a(t.event.source.adapter),1)]),_:1},8,["color"])):w("v-if",!0)])],2)}const A=k(N,[["render",D],["__scopeId","data-v-49238629"],["__file","/tmp/cc-web-panel-sT7rhN/repo/packages/web-panel/src/components/pdh/renderers/ChatBubbleRenderer.vue"]]);export{A as default};
@@ -1 +0,0 @@
1
- import{I as w,P as l,J as u,U as s,R as n,c as i,K as v,Q as _,V as g,b as a}from"./vendor-BvqAck49.js";import{_ as N}from"./index-CzeRvhia.js";import"./icons-DP3uiYxy.js";const S={__name:"OrderTableRenderer",props:{event:{type:Object,required:!0}},setup(p,{expose:r}){r();const o=p,e=a(()=>o.event.content||{}),d=a(()=>o.event.extra||{}),m=a(()=>d.value.merchant||e.value.merchant||e.value.counterparty||"—"),c=a(()=>e.value.title||e.value.name||e.value.itemName||e.value.text||"—"),y=a(()=>{const t=e.value.amount??e.value.price??e.value.total;return t==null?"—":`${e.value.currency||"¥"} ${typeof t=="number"?t.toFixed(2):t}`}),T=a(()=>d.value.orderNo||e.value.orderNo||e.value.orderId),f=a(()=>e.value.status||e.value.state),h=a(()=>{const t=(f.value||"").toLowerCase();return t.includes("成功")||t.includes("succe")||t.includes("paid")?"green":t.includes("退")||t.includes("refund")?"orange":t.includes("失败")||t.includes("fail")?"red":"default"}),b=a(()=>{if(!o.event.occurredAt)return"";const t=new Date(o.event.occurredAt);return`${t.getFullYear()}-${String(t.getMonth()+1).padStart(2,"0")}-${String(t.getDate()).padStart(2,"0")} ${String(t.getHours()).padStart(2,"0")}:${String(t.getMinutes()).padStart(2,"0")}`}),x={props:o,c:e,e:d,merchantText:m,itemText:c,amountText:y,orderNo:T,statusText:f,statusColor:h,formattedTime:b,computed:a};return Object.defineProperty(x,"__isScriptSetup",{enumerable:!1,value:!0}),x}},k={class:"order-card"},C={class:"head"},O={class:"time"},V={class:"row"},R={class:"val"},B={class:"row"},D={class:"val"},I={class:"row amount-row"},j={class:"val amount"},A={key:0,class:"row"},F={class:"val mono"},M={key:1,class:"row"};function P(p,r,o,e,d,m){const c=w("a-tag");return l(),u("div",k,[s("div",C,[s("span",O,n(e.formattedTime),1),i(c,{color:"gold"},{default:v(()=>[_(n(o.event.source.adapter),1)]),_:1}),i(c,null,{default:v(()=>[_(n(o.event.subtype),1)]),_:1})]),s("div",V,[r[0]||(r[0]=s("span",{class:"key"},"商户",-1)),s("span",R,n(e.merchantText),1)]),s("div",B,[r[1]||(r[1]=s("span",{class:"key"},"商品/项目",-1)),s("span",D,n(e.itemText),1)]),s("div",I,[r[2]||(r[2]=s("span",{class:"key"},"金额",-1)),s("span",j,n(e.amountText),1)]),e.orderNo?(l(),u("div",A,[r[3]||(r[3]=s("span",{class:"key"},"单号",-1)),s("span",F,n(e.orderNo),1)])):g("v-if",!0),e.statusText?(l(),u("div",M,[r[4]||(r[4]=s("span",{class:"key"},"状态",-1)),i(c,{color:e.statusColor},{default:v(()=>[_(n(e.statusText),1)]),_:1},8,["color"])])):g("v-if",!0)])}const K=N(S,[["render",P],["__scopeId","data-v-5ed5524d"],["__file","/tmp/cc-web-panel-sT7rhN/repo/packages/web-panel/src/components/pdh/renderers/OrderTableRenderer.vue"]]);export{K as default};
@@ -1 +0,0 @@
1
- import{I as i,J as A,U as I,R as s,Q as r,c as t,K as a,V as _,S as x,o as oe,b as ee,r as u,P as p,F as ne,a2 as ge}from"./vendor-BvqAck49.js";import{_ as we,b as re,u as ie,d}from"./index-CzeRvhia.js";import{u as de}from"./useShellMode-CgR0wCYM.js";import{aM as ke,aN as Fe,aO as be,f as je,a4 as Ce,a7 as xe,as as Oe,R as Se}from"./icons-DP3uiYxy.js";const Pe={__name:"Projects",setup(se,{expose:l}){l();const{t:te}=re(),e=ie(),{isEmbedded:W}=de();async function w(o,f,ye,ae=8e3){if(W){const Z=await e.sendRaw({...f,type:o},ae);if(!Z?.ok){const $=Z?.error;throw new Error(typeof $=="string"?$:$?.message||`${o} failed`)}return Z.result}return e.executeJson(ye,ae)}const B=u(!1),y=u([]),O=u(""),S=u(""),P=u(!1),v=u(null),U=u([]),k=u(!1),E=u(!1),L=u(!1),F=u({name:"",description:"",type:"document",rootPath:""}),z=[{title:"ID",key:"id",width:110},{title:"名称",key:"name"},{title:"类型",key:"project_type",width:120},{title:"状态",key:"status",width:100},{title:"同步",key:"sync_status",width:100},{title:"更新",key:"updated_at",width:160},{title:"操作",key:"actions",width:180,fixed:"right"}],M=ee(()=>{const o={active:0,draft:0,completed:0,archived:0};for(const f of y.value)o[f.status]!==void 0&&(o[f.status]+=1);return o}),J=ee(()=>y.value.filter(o=>!(O.value&&o.status!==O.value||S.value&&!o.name.toLowerCase().includes(S.value.toLowerCase()))));function g(o){switch(o){case"active":return"green";case"draft":return"default";case"completed":return"cyan";case"archived":return"gold";default:return"default"}}function K(o){switch(o){case"synced":return"green";case"pending":return"orange";case"conflict":return"red";case"error":return"red";default:return"default"}}function Q(o){return!o&&o!==0?"-":new Date(typeof o=="number"?o:parseInt(o,10)).toLocaleString("zh-CN",{hour12:!1})}async function C(){B.value=!0;try{const o=await w("project.list",{limit:500},"project list --limit 500 --json",1e4);y.value=Array.isArray(o?.projects)?o.projects:Array.isArray(o)?o:[]}catch(o){d.error("加载失败: "+(o.message||o)),y.value=[]}finally{B.value=!1}}function G(o){return{onClick:()=>V(o)}}async function V(o){v.value=o,P.value=!0,k.value=!0,U.value=[];try{const f=await w("project.listFiles",{projectId:o.id,limit:200},`project list-files ${o.id} --json`,8e3);U.value=Array.isArray(f?.files)?f.files:[]}catch{U.value=[]}finally{k.value=!1}}async function H(o){try{(await w("project.delete",{id:o.id},`project delete ${o.id} --json`,8e3))?.ok?(d.success(`已删除项目 '${o.name}'`),P.value=!1,await C()):d.error("删除失败")}catch(f){d.error(f.message||String(f))}}function X(){F.value={name:"",description:"",type:"document",rootPath:""},E.value=!0}async function T(){if(!F.value.name.trim()){d.warning("请输入项目名称");return}L.value=!0;try{const o=await w("project.init",{name:F.value.name.trim(),description:F.value.description||null,projectType:F.value.type,rootPath:F.value.rootPath||null},`project init "${F.value.name.trim()}" --type ${F.value.type} --json`,8e3);o?.id?(d.success(`已创建项目 '${o.name}'`),E.value=!1,await C()):d.error("创建失败")}catch(o){d.error(o.message||String(o))}finally{L.value=!1}}const b=u(!1),D=u(!1),j=u({path:"",content:""}),h=u(!1),m=u(!1),R=u({path:""}),n=u(!1),c=u(null),N=u(""),Y=u(!1);function ue(){j.value={path:"",content:""},b.value=!0}function ce(){R.value={path:""},h.value=!0}async function q(){if(v.value){k.value=!0;try{const o=await w("project.listFiles",{projectId:v.value.id,limit:200},`project list-files ${v.value.id} --json`,8e3);U.value=Array.isArray(o?.files)?o.files:[]}catch(o){d.error("刷新文件失败: "+(o.message||o))}finally{k.value=!1}}}async function fe(){if(!j.value.path.trim()){d.warning("请输入文件路径");return}if(v.value){D.value=!0;try{const o=await w("project.createFile",{projectId:v.value.id,filePath:j.value.path.trim(),content:j.value.content||""},`project create-file ${v.value.id} "${j.value.path.trim()}" --json`,8e3);o?.id?(d.success(`已创建文件 '${o.file_name}'`),b.value=!1,await q(),await C()):d.error("创建文件失败")}catch(o){d.error(o.message||String(o))}finally{D.value=!1}}}async function pe(){if(!R.value.path.trim()){d.warning("请输入文件夹路径");return}if(v.value){m.value=!0;try{const o=await w("project.createFolder",{projectId:v.value.id,folderPath:R.value.path.trim()},`project create-folder ${v.value.id} "${R.value.path.trim()}" --json`,8e3);o?.id?(d.success(`已创建文件夹 '${o.file_name}'`),h.value=!1,await q(),await C()):d.error("创建文件夹失败")}catch(o){d.error(o.message||String(o))}finally{m.value=!1}}}async function ve(o){try{(await w("project.deleteFile",{fileId:o.id},`project delete-file ${o.id} --json`,8e3))?.deleted?(d.success(`已删除 '${o.file_name}'`),await q(),await C()):d.error("删除失败")}catch(f){d.error(f.message||String(f))}}async function me(o){if(o.is_folder){d.info("文件夹无内容可编辑");return}try{const f=await w("project.getFile",{fileId:o.id},`project get-file ${o.id} --json`,8e3);c.value=o,N.value=f?.content||"",n.value=!0}catch(f){d.error("读取文件失败: "+(f.message||f))}}async function _e(){if(c.value){Y.value=!0;try{(await w("project.writeFile",{fileId:c.value.id,content:N.value},`project write-file ${c.value.id} --json`,8e3))?.id?(d.success("已保存"),n.value=!1,await q()):d.error("保存失败")}catch(o){d.error(o.message||String(o))}finally{Y.value=!1}}}oe(()=>{C()});const le={t:te,ws:e,isEmbedded:W,callProjectTopic:w,loading:B,projects:y,statusFilter:O,nameFilter:S,detailOpen:P,detail:v,files:U,filesLoading:k,createOpen:E,creating:L,newProject:F,columns:z,countByStatus:M,filteredProjects:J,statusColor:g,syncColor:K,formatTime:Q,loadAll:C,onRowClick:G,openDetail:V,onDelete:H,onShowCreate:X,onCreate:T,createFileOpen:b,creatingFile:D,newFile:j,createFolderOpen:h,creatingFolder:m,newFolder:R,editFileOpen:n,editingFile:c,editFileContent:N,savingFile:Y,openCreateFile:ue,openCreateFolder:ce,refreshFiles:q,onCreateFile:fe,onCreateFolder:pe,onDeleteFile:ve,openEditFile:me,onSaveFile:_e,ref:u,computed:ee,onMounted:oe,get ReloadOutlined(){return Se},get PlusOutlined(){return Oe},get FolderOutlined(){return xe},get FileTextOutlined(){return Ce},get CheckCircleOutlined(){return je},get CheckSquareOutlined(){return be},get FileAddOutlined(){return Fe},get FolderAddOutlined(){return ke},get message(){return d},get useI18n(){return re},get useWsStore(){return ie},get useShellMode(){return de}};return Object.defineProperty(le,"__isScriptSetup",{enumerable:!1,value:!0}),le}},ze={style:{display:"flex","align-items":"center","justify-content":"space-between","margin-bottom":"24px"}},he={class:"page-title"},Ae={class:"page-sub"},Ie={class:"filter-bar",style:{"margin-bottom":"12px",display:"flex",gap:"12px","align-items":"center"}},Ue={key:0,style:{"font-size":"12px",color:"#888"}},De={key:0,style:{color:"#888","font-size":"12px"}},Re={key:0},Te={key:1,style:{color:"#888"}},Be=["onClick"],Ee={key:0,style:{color:"#888","font-size":"12px"}};function Le(se,l,te,e,W,w){const B=i("router-link"),y=i("a-button"),O=i("a-space"),S=i("a-statistic"),P=i("a-card"),v=i("a-col"),U=i("a-row"),k=i("a-radio-button"),E=i("a-radio-group"),L=i("a-input-search"),F=i("a-alert"),z=i("a-tag"),M=i("a-popconfirm"),J=i("a-table"),g=i("a-descriptions-item"),K=i("a-descriptions"),Q=i("a-divider"),C=i("a-empty"),G=i("a-list-item"),V=i("a-list"),H=i("a-spin"),X=i("a-drawer"),T=i("a-input"),b=i("a-form-item"),D=i("a-textarea"),j=i("a-form"),h=i("a-modal"),m=i("a-select-option"),R=i("a-select");return p(),A("div",null,[I("div",ze,[I("div",null,[I("h2",he,s(e.t("projects.title")),1),I("p",Ae,[l[17]||(l[17]=r("桌面 / CLI / 手机 三端共享同一份项目数据,Phase 3d sync 自动同步 (",-1)),t(B,{to:"/project-init"},{default:a(()=>[...l[16]||(l[16]=[r("项目初始化 / 环境设置",-1)])]),_:1}),l[18]||(l[18]=r(")",-1))])]),t(O,null,{default:a(()=>[t(y,{loading:e.loading,onClick:e.loadAll},{icon:a(()=>[t(e.ReloadOutlined)]),default:a(()=>[l[19]||(l[19]=r(" 刷新 ",-1))]),_:1},8,["loading"]),t(y,{type:"primary",onClick:e.onShowCreate},{icon:a(()=>[t(e.PlusOutlined)]),default:a(()=>[l[20]||(l[20]=r(" 新建项目 ",-1))]),_:1})]),_:1})]),_(" Stats "),t(U,{gutter:[16,16],style:{"margin-bottom":"20px"}},{default:a(()=>[t(v,{xs:12,sm:8,lg:6},{default:a(()=>[t(P,{style:{background:"var(--bg-card)","border-color":"var(--border-color)"}},{default:a(()=>[t(S,{title:"总项目",value:e.projects.length,"value-style":{color:"#1677ff",fontSize:"20px"}},{prefix:a(()=>[t(e.FolderOutlined)]),_:1},8,["value"])]),_:1})]),_:1}),t(v,{xs:12,sm:8,lg:6},{default:a(()=>[t(P,{style:{background:"var(--bg-card)","border-color":"var(--border-color)"}},{default:a(()=>[t(S,{title:"活跃",value:e.countByStatus.active,"value-style":{color:"#52c41a",fontSize:"20px"}},{prefix:a(()=>[t(e.CheckCircleOutlined)]),_:1},8,["value"])]),_:1})]),_:1}),t(v,{xs:12,sm:8,lg:6},{default:a(()=>[t(P,{style:{background:"var(--bg-card)","border-color":"var(--border-color)"}},{default:a(()=>[t(S,{title:"草稿",value:e.countByStatus.draft,"value-style":{color:"#8c8c8c",fontSize:"20px"}},{prefix:a(()=>[t(e.FileTextOutlined)]),_:1},8,["value"])]),_:1})]),_:1}),t(v,{xs:12,sm:8,lg:6},{default:a(()=>[t(P,{style:{background:"var(--bg-card)","border-color":"var(--border-color)"}},{default:a(()=>[t(S,{title:"已完成",value:e.countByStatus.completed,"value-style":{color:"#13c2c2",fontSize:"20px"}},{prefix:a(()=>[t(e.CheckSquareOutlined)]),_:1},8,["value"])]),_:1})]),_:1})]),_:1}),_(" Filter "),I("div",Ie,[t(E,{value:e.statusFilter,"onUpdate:value":l[0]||(l[0]=n=>e.statusFilter=n),size:"small","button-style":"solid"},{default:a(()=>[t(k,{value:""},{default:a(()=>[...l[21]||(l[21]=[r("全部",-1)])]),_:1}),t(k,{value:"active"},{default:a(()=>[...l[22]||(l[22]=[r("活跃",-1)])]),_:1}),t(k,{value:"draft"},{default:a(()=>[...l[23]||(l[23]=[r("草稿",-1)])]),_:1}),t(k,{value:"completed"},{default:a(()=>[...l[24]||(l[24]=[r("已完成",-1)])]),_:1}),t(k,{value:"archived"},{default:a(()=>[...l[25]||(l[25]=[r("已归档",-1)])]),_:1})]),_:1},8,["value"]),t(L,{value:e.nameFilter,"onUpdate:value":l[1]||(l[1]=n=>e.nameFilter=n),placeholder:"按名称过滤",style:{"max-width":"260px"},"allow-clear":""},null,8,["value"])]),!e.projects.length&&!e.loading?(p(),x(F,{key:0,type:"info",message:"还没有项目",description:'运行 `cc project init <name>` 或点击右上角"新建项目"按钮创建。桌面 / CLI / 手机三端共享同一份数据。',"show-icon":"",style:{"margin-bottom":"16px"}})):_("v-if",!0),e.projects.length||e.loading?(p(),x(J,{key:1,columns:e.columns,"data-source":e.filteredProjects,loading:e.loading,pagination:{pageSize:20,showSizeChanger:!0},"row-key":"id","custom-row":e.onRowClick},{bodyCell:a(({column:n,record:c})=>[n.key==="id"?(p(),A("code",Ue,s(c.id.slice(0,8))+"…",1)):n.key==="name"?(p(),A(ne,{key:1},[I("strong",null,s(c.name),1),c.description?(p(),A("div",De,s(c.description),1)):_("v-if",!0)],64)):n.key==="project_type"?(p(),x(z,{key:2},{default:a(()=>[r(s(c.project_type),1)]),_:2},1024)):n.key==="status"?(p(),x(z,{key:3,color:e.statusColor(c.status)},{default:a(()=>[r(s(c.status),1)]),_:2},1032,["color"])):n.key==="sync_status"?(p(),x(z,{key:4,color:e.syncColor(c.sync_status)},{default:a(()=>[r(s(c.sync_status||"unknown"),1)]),_:2},1032,["color"])):n.key==="updated_at"?(p(),A(ne,{key:5},[r(s(e.formatTime(c.updated_at)),1)],64)):n.key==="actions"?(p(),x(O,{key:6,onClick:l[2]||(l[2]=ge(()=>{},["stop"]))},{default:a(()=>[t(y,{size:"small",onClick:N=>e.openDetail(c)},{default:a(()=>[...l[26]||(l[26]=[r("详情",-1)])]),_:1},8,["onClick"]),t(M,{title:`删除项目 '${c.name}' ?`,"ok-text":"删除","ok-type":"danger","cancel-text":"取消",onConfirm:N=>e.onDelete(c)},{default:a(()=>[t(y,{size:"small",danger:""},{default:a(()=>[...l[27]||(l[27]=[r("删除",-1)])]),_:1})]),_:1},8,["title","onConfirm"])]),_:2},1024)):_("v-if",!0)]),_:1},8,["data-source","loading"])):_("v-if",!0),_(" Detail drawer "),t(X,{open:e.detailOpen,"onUpdate:open":l[3]||(l[3]=n=>e.detailOpen=n),title:e.detail?.name||"项目详情",width:"640",placement:"right"},{default:a(()=>[e.detail?(p(),x(K,{key:0,column:1,bordered:"",size:"small"},{default:a(()=>[t(g,{label:"ID"},{default:a(()=>[I("code",null,s(e.detail.id),1)]),_:1}),t(g,{label:"名称"},{default:a(()=>[r(s(e.detail.name),1)]),_:1}),t(g,{label:"描述"},{default:a(()=>[r(s(e.detail.description||"-"),1)]),_:1}),t(g,{label:"类型"},{default:a(()=>[t(z,null,{default:a(()=>[r(s(e.detail.project_type),1)]),_:1})]),_:1}),t(g,{label:"状态"},{default:a(()=>[t(z,{color:e.statusColor(e.detail.status)},{default:a(()=>[r(s(e.detail.status),1)]),_:1},8,["color"])]),_:1}),t(g,{label:"同步状态"},{default:a(()=>[t(z,{color:e.syncColor(e.detail.sync_status)},{default:a(()=>[r(s(e.detail.sync_status||"unknown"),1)]),_:1},8,["color"])]),_:1}),t(g,{label:"用户"},{default:a(()=>[r(s(e.detail.user_id),1)]),_:1}),t(g,{label:"根路径"},{default:a(()=>[e.detail.root_path?(p(),A("code",Re,s(e.detail.root_path),1)):(p(),A("span",Te,"(无)"))]),_:1}),t(g,{label:"文件数"},{default:a(()=>[r(s(e.detail.file_count||0),1)]),_:1}),t(g,{label:"创建"},{default:a(()=>[r(s(e.formatTime(e.detail.created_at)),1)]),_:1}),t(g,{label:"更新"},{default:a(()=>[r(s(e.formatTime(e.detail.updated_at)),1)]),_:1})]),_:1})):_("v-if",!0),t(Q,null,{default:a(()=>[...l[28]||(l[28]=[r("文件",-1)])]),_:1}),_(" Sub-phase 7.3 (2026-05-17): 文件 CRUD toolbar "),t(O,{style:{"margin-bottom":"12px"}},{default:a(()=>[t(y,{size:"small",type:"primary",disabled:!e.detail,onClick:e.openCreateFile},{default:a(()=>[t(e.FileAddOutlined),l[29]||(l[29]=r(" 新建文件 ",-1))]),_:1},8,["disabled"]),t(y,{size:"small",disabled:!e.detail,onClick:e.openCreateFolder},{default:a(()=>[t(e.FolderAddOutlined),l[30]||(l[30]=r(" 新建文件夹 ",-1))]),_:1},8,["disabled"])]),_:1}),t(H,{spinning:e.filesLoading},{default:a(()=>[!e.files.length&&!e.filesLoading?(p(),x(C,{key:0,description:"无文件 (点上方按钮新建)"})):(p(),x(V,{key:1,size:"small","data-source":e.files},{renderItem:a(({item:n})=>[t(G,null,{extra:a(()=>[t(O,null,{default:a(()=>[n.file_size?(p(),A("span",Ee,s(n.file_size)+" B",1)):_("v-if",!0),t(M,{title:`删除 ${n.is_folder?"文件夹":"文件"} '${n.file_name}' ?`,"ok-text":"删除","ok-type":"danger","cancel-text":"取消",onConfirm:c=>e.onDeleteFile(n)},{default:a(()=>[t(y,{size:"small",danger:""},{default:a(()=>[...l[31]||(l[31]=[r("×",-1)])]),_:1})]),_:1},8,["title","onConfirm"])]),_:2},1024)]),default:a(()=>[I("span",{style:{cursor:"pointer"},onClick:c=>e.openEditFile(n)},s(n.is_folder?"📁":"📄")+" "+s(n.file_path),9,Be)]),_:2},1024)]),_:1},8,["data-source"]))]),_:1},8,["spinning"])]),_:1},8,["open","title"]),_(" Sub-phase 7.3: Create file modal "),t(h,{open:e.createFileOpen,"onUpdate:open":l[6]||(l[6]=n=>e.createFileOpen=n),title:"新建文件","confirm-loading":e.creatingFile,"ok-text":"创建","cancel-text":"取消",onOk:e.onCreateFile},{default:a(()=>[t(j,{layout:"vertical"},{default:a(()=>[t(b,{label:"文件路径(项目内)",required:""},{default:a(()=>[t(T,{value:e.newFile.path,"onUpdate:value":l[4]||(l[4]=n=>e.newFile.path=n),placeholder:"例如 README.md 或 src/main.kt"},null,8,["value"])]),_:1}),t(b,{label:"初始内容(可选)"},{default:a(()=>[t(D,{value:e.newFile.content,"onUpdate:value":l[5]||(l[5]=n=>e.newFile.content=n),rows:6,placeholder:"留空创建空文件"},null,8,["value"])]),_:1})]),_:1})]),_:1},8,["open","confirm-loading"]),_(" Sub-phase 7.3: Create folder modal "),t(h,{open:e.createFolderOpen,"onUpdate:open":l[8]||(l[8]=n=>e.createFolderOpen=n),title:"新建文件夹","confirm-loading":e.creatingFolder,"ok-text":"创建","cancel-text":"取消",onOk:e.onCreateFolder},{default:a(()=>[t(j,{layout:"vertical"},{default:a(()=>[t(b,{label:"文件夹路径(项目内)",required:""},{default:a(()=>[t(T,{value:e.newFolder.path,"onUpdate:value":l[7]||(l[7]=n=>e.newFolder.path=n),placeholder:"例如 src/utils"},null,8,["value"])]),_:1})]),_:1})]),_:1},8,["open","confirm-loading"]),_(" Sub-phase 7.3: Edit file content modal "),t(h,{open:e.editFileOpen,"onUpdate:open":l[10]||(l[10]=n=>e.editFileOpen=n),title:`编辑 ${e.editingFile?.file_name||""}`,"confirm-loading":e.savingFile,"ok-text":"保存","cancel-text":"取消",onOk:e.onSaveFile,width:"720"},{default:a(()=>[t(D,{value:e.editFileContent,"onUpdate:value":l[9]||(l[9]=n=>e.editFileContent=n),rows:18,"auto-size":{minRows:12,maxRows:24},style:{"font-family":"monospace"}},null,8,["value"])]),_:1},8,["open","title","confirm-loading"]),_(" Create modal "),t(h,{open:e.createOpen,"onUpdate:open":l[15]||(l[15]=n=>e.createOpen=n),title:"新建项目","confirm-loading":e.creating,"ok-text":"创建","cancel-text":"取消",onOk:e.onCreate},{default:a(()=>[t(j,{layout:"vertical"},{default:a(()=>[t(b,{label:"项目名称",required:""},{default:a(()=>[t(T,{value:e.newProject.name,"onUpdate:value":l[11]||(l[11]=n=>e.newProject.name=n),placeholder:"例如:旅行计划-上海"},null,8,["value"])]),_:1}),t(b,{label:"描述"},{default:a(()=>[t(D,{value:e.newProject.description,"onUpdate:value":l[12]||(l[12]=n=>e.newProject.description=n),rows:2},null,8,["value"])]),_:1}),t(b,{label:"类型"},{default:a(()=>[t(R,{value:e.newProject.type,"onUpdate:value":l[13]||(l[13]=n=>e.newProject.type=n)},{default:a(()=>[t(m,{value:"document"},{default:a(()=>[...l[32]||(l[32]=[r("文档 (document)",-1)])]),_:1}),t(m,{value:"data"},{default:a(()=>[...l[33]||(l[33]=[r("数据 (data)",-1)])]),_:1}),t(m,{value:"web"},{default:a(()=>[...l[34]||(l[34]=[r("Web",-1)])]),_:1}),t(m,{value:"app"},{default:a(()=>[...l[35]||(l[35]=[r("应用 (app)",-1)])]),_:1}),t(m,{value:"presentation"},{default:a(()=>[...l[36]||(l[36]=[r("演示文稿",-1)])]),_:1}),t(m,{value:"spreadsheet"},{default:a(()=>[...l[37]||(l[37]=[r("表格",-1)])]),_:1}),t(m,{value:"design"},{default:a(()=>[...l[38]||(l[38]=[r("设计",-1)])]),_:1}),t(m,{value:"code"},{default:a(()=>[...l[39]||(l[39]=[r("代码",-1)])]),_:1}),t(m,{value:"workflow"},{default:a(()=>[...l[40]||(l[40]=[r("工作流",-1)])]),_:1}),t(m,{value:"knowledge"},{default:a(()=>[...l[41]||(l[41]=[r("知识库",-1)])]),_:1})]),_:1},8,["value"])]),_:1}),t(b,{label:"根路径 (可选)"},{default:a(()=>[t(T,{value:e.newProject.rootPath,"onUpdate:value":l[14]||(l[14]=n=>e.newProject.rootPath=n),placeholder:"留空则仅元数据"},null,8,["value"])]),_:1})]),_:1})]),_:1},8,["open","confirm-loading"])])}const We=we(Pe,[["render",Le],["__scopeId","data-v-019ac5aa"],["__file","/tmp/cc-web-panel-sT7rhN/repo/packages/web-panel/src/views/Projects.vue"]]);export{We as default};
@@ -1 +0,0 @@
1
- import{E as B,b as z,r as D,I as p,J as c,U as _,c as n,K as a,S,V as w,o as P,x as I,P as i,Q as g,R as r,F as h,Z as F}from"./vendor-BvqAck49.js";import{u as R,_ as O}from"./index-CzeRvhia.js";import{R as E}from"./icons-DP3uiYxy.js";const V=B("tasks",()=>{const k=D([]),l=D(!1);let u=null,e=null;const x=z(()=>k.value.filter(t=>t.status==="running")),N=z(()=>k.value.filter(t=>t.status==="pending")),y=z(()=>k.value.filter(t=>t.status==="completed"||t.status==="failed"||t.status==="timeout"));async function f(){const t=R();l.value=!0;try{const s=await t.sendRaw({type:"tasks-list"});s&&Array.isArray(s.tasks)&&(k.value=s.tasks)}catch{}finally{l.value=!1}}async function o(t){const s=R();try{await s.sendRaw({type:"tasks-stop",taskId:t})}catch{}finally{await f()}}const d=D(null);function m(t=5e3){v(),f(),u=setInterval(f,t),C()}function v(){u&&(clearInterval(u),u=null),e&&(e(),e=null)}function C(){const t=R();if(e)return;const s=T=>{T.type==="task:notification"&&T.payload?.task&&(d.value=T.payload.task,f(),setTimeout(()=>{d.value=null},8e3))};e=t.onRuntimeEvent(s)}function b(t){return!t||t<0?"-":t<1e3?`${t}ms`:t<6e4?`${(t/1e3).toFixed(1)}s`:`${(t/6e4).toFixed(1)}m`}function A(t){switch(t){case"running":return"processing";case"pending":return"default";case"completed":return"success";case"failed":return"error";case"timeout":return"warning";default:return"default"}}return{tasks:k,loading:l,running:x,pending:N,completed:y,lastNotification:d,fetchTasks:f,stopTask:o,startPolling:m,stopPolling:v,formatDuration:b,getStatusColor:A}}),U={__name:"Tasks",setup(k,{expose:l}){l();const u=V(),e=[{title:"状态",key:"status",width:90},{title:"描述",key:"description",ellipsis:!0},{title:"类型",dataIndex:"type",width:100},{title:"耗时",key:"duration",width:90},{title:"创建时间",key:"createdAt",width:180},{title:"结果",key:"result",ellipsis:!0},{title:"操作",key:"action",width:80}];function x(o){return o.status==="running"&&o.startedAt?u.formatDuration(Date.now()-o.startedAt):o.completedAt&&o.startedAt?u.formatDuration(o.completedAt-o.startedAt):"-"}function N(o){return o?new Date(o).toLocaleString():"-"}function y(o,d){return o?o.length>d?`${o.slice(0,d)}...`:o:""}P(()=>u.startPolling(5e3)),I(()=>u.stopPolling());const f={store:u,columns:e,getDuration:x,formatTime:N,truncate:y,onMounted:P,onUnmounted:I,get ReloadOutlined(){return E},get useTasksStore(){return V}};return Object.defineProperty(f,"__isScriptSetup",{enumerable:!1,value:!0}),f}},L={class:"page-header"},j={key:0,class:"error-text"},J={key:1},K={class:"task-header"},M={class:"task-desc"},Q={class:"task-id"},W={class:"task-meta"},Z=["title"],q={key:0,class:"error-text"},G={key:1,class:"success-text"},H={key:2,class:"muted-text"};function X(k,l,u,e,x,N){const y=p("a-button"),f=p("a-space"),o=p("a-alert"),d=p("a-statistic"),m=p("a-card"),v=p("a-col"),C=p("a-row"),b=p("a-tag"),A=p("a-table");return i(),c("div",null,[_("div",L,[l[3]||(l[3]=_("div",null,[_("h2",{class:"page-title"},"后台任务"),_("p",{class:"page-sub"},"查看后台任务队列、实时通知和任务执行结果。")],-1)),n(f,null,{default:a(()=>[n(y,{ghost:"",loading:e.store.loading,onClick:l[0]||(l[0]=t=>e.store.fetchTasks())},{icon:a(()=>[n(e.ReloadOutlined)]),default:a(()=>[l[2]||(l[2]=g(" 刷新 ",-1))]),_:1},8,["loading"])]),_:1})]),e.store.lastNotification?(i(),S(o,{key:0,type:e.store.lastNotification.status==="completed"?"success":"error","show-icon":"",closable:"",class:"banner",onClose:l[1]||(l[1]=t=>e.store.lastNotification=null)},{message:a(()=>[g(" 任务"+r(e.store.lastNotification.status==="completed"?"完成":"结束")+": "+r(e.store.lastNotification.description||e.store.lastNotification.id.slice(0,16)),1)]),description:a(()=>[e.store.lastNotification.error?(i(),c("span",j,r(e.store.lastNotification.error),1)):e.store.lastNotification.result?(i(),c("span",J,r(e.truncate(String(e.store.lastNotification.result),120)),1)):w("v-if",!0)]),_:1},8,["type"])):w("v-if",!0),n(C,{gutter:[16,16],class:"stats-row"},{default:a(()=>[n(v,{xs:12,sm:6},{default:a(()=>[n(m,{class:"stat-card",size:"small"},{default:a(()=>[n(d,{title:"全部任务",value:e.store.tasks.length},null,8,["value"])]),_:1})]),_:1}),n(v,{xs:12,sm:6},{default:a(()=>[n(m,{class:"stat-card",size:"small"},{default:a(()=>[n(d,{title:"运行中",value:e.store.running.length},null,8,["value"])]),_:1})]),_:1}),n(v,{xs:12,sm:6},{default:a(()=>[n(m,{class:"stat-card",size:"small"},{default:a(()=>[n(d,{title:"等待中",value:e.store.pending.length},null,8,["value"])]),_:1})]),_:1}),n(v,{xs:12,sm:6},{default:a(()=>[n(m,{class:"stat-card",size:"small"},{default:a(()=>[n(d,{title:"已完成",value:e.store.completed.length},null,8,["value"])]),_:1})]),_:1})]),_:1}),e.store.running.length>0?(i(),S(m,{key:1,title:"运行中的任务",class:"panel-card",size:"small"},{default:a(()=>[(i(!0),c(h,null,F(e.store.running,t=>(i(),c("div",{key:t.id,class:"task-item running"},[_("div",K,[n(b,{color:e.store.getStatusColor(t.status)},{default:a(()=>[g(r(t.status.toUpperCase()),1)]),_:2},1032,["color"]),_("span",M,r(t.description),1),_("span",Q,r(t.id.slice(0,16)),1)]),_("div",W,[_("span",null,"类型: "+r(t.type||"-"),1),_("span",null,"已运行: "+r(e.store.formatDuration(Date.now()-t.startedAt)),1),n(y,{size:"small",danger:"",onClick:s=>e.store.stopTask(t.id)},{default:a(()=>[...l[4]||(l[4]=[g("停止",-1)])]),_:1},8,["onClick"])])]))),128))]),_:1})):w("v-if",!0),n(m,{class:"panel-card"},{default:a(()=>[n(A,{columns:e.columns,"data-source":e.store.tasks,pagination:{pageSize:15,size:"small"},loading:e.store.loading,"row-key":"id",size:"small"},{bodyCell:a(({column:t,record:s})=>[t.key==="status"?(i(),S(b,{key:0,color:e.store.getStatusColor(s.status)},{default:a(()=>[g(r(s.status),1)]),_:2},1032,["color"])):t.key==="description"?(i(),c("span",{key:1,title:s.command},r(s.description),9,Z)):t.key==="duration"?(i(),c(h,{key:2},[g(r(e.getDuration(s)),1)],64)):t.key==="createdAt"?(i(),c(h,{key:3},[g(r(e.formatTime(s.createdAt)),1)],64)):t.key==="result"?(i(),c(h,{key:4},[s.error?(i(),c("span",q,r(e.truncate(s.error,60)),1)):s.result?(i(),c("span",G,r(e.truncate(String(s.result),60)),1)):(i(),c("span",H,"-"))],64)):t.key==="action"?(i(),c(h,{key:5},[s.status==="running"?(i(),S(y,{key:0,size:"small",danger:"",onClick:T=>e.store.stopTask(s.id)},{default:a(()=>[...l[5]||(l[5]=[g(" 停止 ",-1)])]),_:1},8,["onClick"])):w("v-if",!0)],64)):w("v-if",!0)]),_:1},8,["data-source","loading"])]),_:1})])}const et=O(U,[["render",X],["__scopeId","data-v-da5ff6a2"],["__file","/tmp/cc-web-panel-sT7rhN/repo/packages/web-panel/src/views/Tasks.vue"]]);export{et as default};
@@ -1,3 +0,0 @@
1
- const __vite__mapDeps=(i,m=__vite__mapDeps,d=(m.f||(m.f=["./xterm-BZcWGsqw.js","./markdown-CsiA8-E5.js","./markdown-Dfs9RUU9.css","./addon-fit-CK6X9sAG.js","./xterm-DFuMZ0ql.css"])))=>i.map(i=>d[i]);
2
- import{u as fe,_ as me,d as V,e as U}from"./index-CzeRvhia.js";import{I as z,J as x,U as _,Q as D,S as Q,K as T,V as R,c as A,F as Y,Z,R as E,o as G,f as H,w as $,n as q,b as ee,r as S,P as y,_ as ve,a2 as we,a3 as he,a4 as _e}from"./vendor-BvqAck49.js";import{R as ye,b as pe,as as ge}from"./icons-DP3uiYxy.js";const k=new Map,C=new Map,N=new Set,F=new Set;let te=!1;function xe(c){te||(te=!0,c.onMessage(o=>{if(!(!o||typeof o.type!="string")){if(o.type==="terminal.stdout"){const{sessionId:s,data:e,seq:i}=o.payload||{};if(!s)return;let w;try{const d=atob(e||""),f=new Uint8Array(d.length);for(let n=0;n<d.length;n++)f[n]=d.charCodeAt(n);w=new TextDecoder("utf-8").decode(f)}catch{w=""}const u={sessionId:s,data:w,seq:i};k.get(s)?.forEach(d=>d(u)),N.forEach(d=>d(u))}else if(o.type==="terminal.exit"){const{sessionId:s,exitCode:e,signal:i}=o.payload||{};if(!s)return;const w={sessionId:s,exitCode:e,signal:i};C.get(s)?.forEach(u=>u(w)),F.forEach(u=>u(w))}}}))}function ne(c){const o=new TextEncoder().encode(c);let s="";for(let e=0;e<o.length;e++)s+=String.fromCharCode(o[e]);return btoa(s)}function ae(c){const o=atob(c||""),s=new Uint8Array(o.length);for(let e=0;e<o.length;e++)s[e]=o.charCodeAt(e);return new TextDecoder("utf-8").decode(s)}function oe(){const c=fe();xe(c);async function o(n={}){const a=await c.sendRaw({type:"terminal.create",payload:{shell:n.shell,cwd:n.cwd,env:n.env,cols:n.cols,rows:n.rows}});if(a.ok===!1)throw new Error(a.error||"terminal_create_failed");return a.result??a}async function s(){const n=await c.sendRaw({type:"terminal.list",payload:{}});if(n.ok===!1)throw new Error(n.error||"terminal_list_failed");const a=n.result??n;return Array.isArray(a.sessions)?a.sessions:[]}async function e(n,a){const r=await c.sendRaw({type:"terminal.stdin",payload:{sessionId:n,data:ne(String(a))}});if(r.ok===!1)throw new Error(r.error||"terminal_stdin_failed");return r.result??r}async function i(n,a,r){const h=await c.sendRaw({type:"terminal.resize",payload:{sessionId:n,cols:a,rows:r}});if(h.ok===!1)throw new Error(h.error||"terminal_resize_failed");return h.result??h}async function w(n){const a=await c.sendRaw({type:"terminal.close",payload:{sessionId:n}});if(a.ok===!1)throw new Error(a.error||"terminal_close_failed");return a.result??a}async function u(n,a=0){const r=await c.sendRaw({type:"terminal.history",payload:{sessionId:n,fromSeq:a}});if(r.ok===!1)throw new Error(r.error||"terminal_history_failed");const h=r.result??r;return{truncated:!!h.truncated,chunks:(h.chunks||[]).map(L=>({seq:L.seq,data:ae(L.data)}))}}function d(n,a){return n?(k.has(n)||k.set(n,new Set),k.get(n).add(a),()=>{k.get(n)?.delete(a),k.get(n)?.size===0&&k.delete(n)}):(N.add(a),()=>N.delete(a))}function f(n,a){return n?(C.has(n)||C.set(n,new Set),C.get(n).add(a),()=>{C.get(n)?.delete(a),C.get(n)?.size===0&&C.delete(n)}):(F.add(a),()=>F.delete(a))}return{create:o,list:s,stdin:e,resize:i,close:w,history:u,onStdout:d,onExit:f,_internal:{stdoutSubs:k,exitSubs:C,toBase64Utf8:ne,fromBase64Utf8:ae}}}const Se={__name:"Terminal",setup(c,{expose:o}){o();const s=oe(),e=S([]),i=S(null),w=S("pwsh"),u=S(!1),d=S(!1),f=S(""),n=S(""),a=S([]),r=[{value:"pwsh",label:"PowerShell"},{value:"cmd",label:"CMD"},{value:"bash",label:"Bash"},{value:"wsl",label:"WSL"}],h=ee(()=>e.value.find(t=>t.id===i.value));function L(t){return t?t.slice(0,8):""}let O=null,M=null;async function I(){if(O)return{xtermMod:O,fitAddonMod:M};try{O=await U(()=>import("./xterm-BZcWGsqw.js").then(t=>t.x),__vite__mapDeps([0,1,2]),import.meta.url),M=await U(()=>import("./addon-fit-CK6X9sAG.js").then(t=>t.a),__vite__mapDeps([3,1,2]),import.meta.url),await U(()=>Promise.resolve({}),__vite__mapDeps([4]),import.meta.url)}catch(t){throw n.value="xterm 资源加载失败:"+(t?.message||"未知错误"),t}return{xtermMod:O,fitAddonMod:M}}async function B(t){await q();const{xtermMod:l,fitAddonMod:p}=await I(),b=a.value.find(v=>v?.dataset?.sessionId===t.id);if(!b)return;const m=new l.Terminal({cursorBlink:!0,fontFamily:'Consolas, "Courier New", monospace',fontSize:13,theme:{background:"#1e1e1e",foreground:"#d4d4d4"},convertEol:!1}),P=new p.FitAddon;m.loadAddon(P),m.open(b);try{P.fit()}catch{}t.xterm=m,t.fitAddon=P;const ie=m.onData(v=>{s.stdin(t.id,v).catch(g=>{String(g?.message||"").includes("dangerous_keyword_blocked")?V.warning("该命令被桌面端拦截(高危关键字)"):V.error("stdin 失败: "+(g?.message||g))})}),ce=s.onStdout(t.id,({data:v,seq:g})=>{t.lastSeq=g,m.write(v)}),de=s.onExit(t.id,({exitCode:v,signal:g})=>{t.alive=!1,t.exitCode=v,t.signal=g,m.writeln(`\r
3
- \x1B[33m[session exited, code=${v}, signal=${g??"-"}]\x1B[0m`)});t.offs=()=>{try{ie.dispose?.()}catch{}ce(),de()};try{const{chunks:v,truncated:g}=await s.history(t.id,0);g&&m.writeln("\x1B[2m[history truncated — earlier output was evicted]\x1B[0m");for(const K of v)m.write(K.data),t.lastSeq=K.seq}catch(v){m.writeln(`\x1B[31m[history fetch failed: ${v?.message||v}]\x1B[0m`)}const J=new ResizeObserver(()=>{try{P.fit(),s.resize(t.id,m.cols,m.rows).catch(()=>{})}catch{}});J.observe(b);const ue=t.offs;t.offs=()=>{try{J.disconnect()}catch{}ue()}}async function re(){u.value=!0,f.value="";try{const t=await s.create({shell:w.value,cols:80,rows:24}),l={id:t.sessionId,shell:t.shell,cwd:"",alive:!0,lastSeq:0,exitCode:null,xterm:null,fitAddon:null,offs:()=>{}};e.value.push(l),i.value=l.id,await B(l)}catch(t){f.value=t?.message||String(t)}finally{u.value=!1}}async function se(t){try{await s.close(t)}catch(l){f.value=l?.message||String(l)}setTimeout(()=>W(t),500)}function W(t){const l=e.value.findIndex(b=>b.id===t);if(l===-1)return;const p=e.value[l];try{p.offs?.()}catch{}try{p.xterm?.dispose?.()}catch{}e.value.splice(l,1),i.value===t&&(i.value=e.value[0]?.id||null)}function le(t){i.value=t,q(()=>{const l=e.value.find(p=>p.id===t);try{l?.fitAddon?.fit()}catch{}})}async function X(){d.value=!0,f.value="";try{const t=await s.list();for(const l of t){const p=e.value.find(m=>m.id===l.id);if(p){p.alive=l.alive,p.lastSeq=l.lastSeq;continue}const b={id:l.id,shell:l.shell,cwd:l.cwd,alive:l.alive,lastSeq:l.lastSeq,exitCode:null,xterm:null,fitAddon:null,offs:()=>{}};e.value.push(b),await B(b)}!i.value&&e.value.length>0&&(i.value=e.value[0].id)}catch(t){f.value=t?.message||String(t)}finally{d.value=!1}}G(async()=>{await X()}),H(()=>{for(const t of e.value){try{t.offs?.()}catch{}try{t.xterm?.dispose?.()}catch{}}}),$(i,()=>{q(()=>{const t=h.value;try{t?.fitAddon?.fit()}catch{}})});const j={term:s,sessions:e,activeId:i,newShell:w,creating:u,loadingList:d,error:f,warning:n,xtermContainers:a,shellOptions:r,active:h,shortId:L,get xtermMod(){return O},set xtermMod(t){O=t},get fitAddonMod(){return M},set fitAddonMod(t){M=t},loadXterm:I,mountXterm:B,onCreate:re,onClose:se,removeSession:W,activate:le,refreshList:X,ref:S,computed:ee,onMounted:G,onBeforeUnmount:H,nextTick:q,watch:$,get message(){return V},get PlusOutlined(){return ge},get CloseOutlined(){return pe},get ReloadOutlined(){return ye},get useTerminal(){return oe}};return Object.defineProperty(j,"__isScriptSetup",{enumerable:!1,value:!0}),j}},be={class:"terminal-page"},ke={class:"terminal-header"},Ce={class:"page-sub"},Ae={class:"terminal-body"},Ee={class:"session-tabs"},Oe=["onClick"],Te={class:"session-shell"},Re={class:"session-id"},Me={key:0,class:"session-empty"},ze={class:"xterm-host"},Le=["data-session-id"],Pe={key:0,class:"xterm-placeholder"},De={key:1,class:"terminal-footer"},qe={key:0,class:"footer-exit"};function Be(c,o,s,e,i,w){const u=z("a-tag"),d=z("a-select"),f=z("a-button"),n=z("a-space"),a=z("a-alert");return y(),x("div",be,[_("div",ke,[_("div",null,[o[3]||(o[3]=_("h2",{class:"page-title"},"远程终端",-1)),_("p",Ce,[o[2]||(o[2]=D(" 桌面端托管的 PTY 会话;Android 端可远程操控同一通道 ",-1)),e.warning?(y(),Q(u,{key:0,color:"orange",style:{"margin-left":"8px"}},{default:T(()=>[D(E(e.warning),1)]),_:1})):R("v-if",!0)])]),A(n,null,{default:T(()=>[A(d,{value:e.newShell,"onUpdate:value":o[0]||(o[0]=r=>e.newShell=r),style:{width:"130px"},size:"small",options:e.shellOptions},null,8,["value"]),A(f,{type:"primary",size:"small",loading:e.creating,onClick:e.onCreate},{icon:T(()=>[A(e.PlusOutlined)]),default:T(()=>[o[4]||(o[4]=D(" 新会话 ",-1))]),_:1},8,["loading"]),A(f,{size:"small",loading:e.loadingList,onClick:e.refreshList},{icon:T(()=>[A(e.ReloadOutlined)]),default:T(()=>[o[5]||(o[5]=D(" 刷新 ",-1))]),_:1},8,["loading"])]),_:1})]),e.error?(y(),Q(a,{key:0,message:e.error,type:"error","show-icon":"",closable:"",style:{"margin-bottom":"12px"},onClose:o[1]||(o[1]=r=>e.error="")},null,8,["message"])):R("v-if",!0),_("div",Ae,[_("div",Ee,[(y(!0),x(Y,null,Z(e.sessions,r=>(y(),x("div",{key:r.id,class:ve(["session-tab",{active:r.id===e.activeId,dead:!r.alive}]),onClick:h=>e.activate(r.id)},[_("span",Te,E(r.shell),1),_("span",Re,E(e.shortId(r.id)),1),A(e.CloseOutlined,{class:"session-close",onClick:we(h=>e.onClose(r.id),["stop"])},null,8,["onClick"])],10,Oe))),128)),e.sessions.length===0?(y(),x("div",Me,' 点击 "新会话" 创建第一个终端 ')):R("v-if",!0)]),_("div",ze,[(y(!0),x(Y,null,Z(e.sessions,r=>he((y(),x("div",{key:r.id,ref_for:!0,ref:"xtermContainers","data-session-id":r.id,class:"xterm-container"},null,8,Le)),[[_e,r.id===e.activeId]])),128)),e.sessions.length===0?(y(),x("div",Pe,[...o[6]||(o[6]=[_("span",null,"无活跃会话",-1)])])):R("v-if",!0)])]),e.active?(y(),x("div",De,[_("span",null,E(e.active.shell)+" · "+E(e.active.cwd||"(默认 cwd)")+" · seq "+E(e.active.lastSeq),1),e.active.alive?R("v-if",!0):(y(),x("span",qe,"已退出 (code="+E(e.active.exitCode??"-")+")",1))])):R("v-if",!0)])}const Fe=me(Se,[["render",Be],["__scopeId","data-v-65366a29"],["__file","/tmp/cc-web-panel-sT7rhN/repo/packages/web-panel/src/views/Terminal.vue"]]);export{Fe as default};
@@ -1 +0,0 @@
1
- import{O as r}from"./index-CzeRvhia.js";const o=((n,a,e)=>{r(n,`[ant-design-vue: ${a}] ${e}`)});export{o as d};
@@ -1 +0,0 @@
1
- import{C as o}from"./Col-CqXc8_ft.js";import{U as t}from"./index-CzeRvhia.js";import"./vendor-BvqAck49.js";import"./index-WDQkyh-E.js";import"./icons-DP3uiYxy.js";const s=t(o);export{s as default};
@@ -1 +0,0 @@
1
- import{A as o}from"./Row-DAio6Dx2.js";import{U as t}from"./index-CzeRvhia.js";import"./vendor-BvqAck49.js";import"./responsiveObserve-B0Cn1i64.js";import"./useFlexGapSupport-y8cUyKiP.js";import"./styleChecker-aYEwS4Pw.js";import"./index-WDQkyh-E.js";import"./icons-DP3uiYxy.js";const l=t(o);export{l as default};