symbiote-ui 0.3.0-alpha.4

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 (322) hide show
  1. package/CHANGELOG.md +46 -0
  2. package/LICENSE +21 -0
  3. package/README.md +76 -0
  4. package/canvas/AutoLayout.js +731 -0
  5. package/canvas/Breadcrumb/Breadcrumb.css.js +75 -0
  6. package/canvas/Breadcrumb/Breadcrumb.js +96 -0
  7. package/canvas/Breadcrumb/Breadcrumb.tpl.js +7 -0
  8. package/canvas/CanvasConnectionRenderer.js +971 -0
  9. package/canvas/CanvasGraph/CanvasGraph.css.js +29 -0
  10. package/canvas/CanvasGraph/CanvasGraph.js +1697 -0
  11. package/canvas/CanvasGraph/CanvasGraphDrawState.js +280 -0
  12. package/canvas/CanvasGraph/CanvasGraphGeometry.js +194 -0
  13. package/canvas/CanvasViewport.js +550 -0
  14. package/canvas/ConnectionRenderer.js +1283 -0
  15. package/canvas/FlowSimulator.js +326 -0
  16. package/canvas/ForceLayout.js +226 -0
  17. package/canvas/ForceWorker.js +1303 -0
  18. package/canvas/FrameManager.js +223 -0
  19. package/canvas/GraphExplorerShell/GraphExplorerShell.css.js +136 -0
  20. package/canvas/GraphExplorerShell/GraphExplorerShell.js +129 -0
  21. package/canvas/GraphExplorerShell/GraphExplorerShell.tpl.js +12 -0
  22. package/canvas/GraphTabs/GraphTabs.css.js +101 -0
  23. package/canvas/GraphTabs/GraphTabs.js +189 -0
  24. package/canvas/GraphTabs/GraphTabs.tpl.js +12 -0
  25. package/canvas/LODManager.js +88 -0
  26. package/canvas/Minimap/Minimap.css.js +73 -0
  27. package/canvas/Minimap/Minimap.js +210 -0
  28. package/canvas/Minimap/Minimap.tpl.js +7 -0
  29. package/canvas/NodeCanvas/NodeCanvas.css.js +398 -0
  30. package/canvas/NodeCanvas/NodeCanvas.js +1499 -0
  31. package/canvas/NodeCanvas/NodeCanvas.tpl.js +22 -0
  32. package/canvas/NodeSearch/NodeSearch.css.js +97 -0
  33. package/canvas/NodeSearch/NodeSearch.js +140 -0
  34. package/canvas/NodeSearch/NodeSearch.tpl.js +25 -0
  35. package/canvas/NodeViewManager.js +748 -0
  36. package/canvas/PcbRouteDiagnostics.js +463 -0
  37. package/canvas/PcbRouter.js +1127 -0
  38. package/canvas/PinExpansion.js +134 -0
  39. package/canvas/PseudoConnection.js +84 -0
  40. package/canvas/SelectionSync.js +163 -0
  41. package/canvas/SubgraphManager.js +203 -0
  42. package/canvas/SubgraphRouter.js +452 -0
  43. package/canvas/ViewportActions.js +473 -0
  44. package/canvas/graph-explorer.js +339 -0
  45. package/canvas/graph-layout.js +148 -0
  46. package/canvas/graph-model.js +68 -0
  47. package/canvas/html-in-canvas.js +202 -0
  48. package/canvas/project-graph-builder.js +440 -0
  49. package/canvas/project-graph-model.js +183 -0
  50. package/chat/ChatComposer/ChatComposer.css.js +652 -0
  51. package/chat/ChatComposer/ChatComposer.js +304 -0
  52. package/chat/ChatList/ChatList.css.js +102 -0
  53. package/chat/ChatList/ChatList.js +99 -0
  54. package/chat/ChatList/ChatList.tpl.js +20 -0
  55. package/chat/ChatListItem/ChatListItem.css.js +117 -0
  56. package/chat/ChatListItem/ChatListItem.js +32 -0
  57. package/chat/ChatListItem/ChatListItem.tpl.js +17 -0
  58. package/chat/ChatMessageItem/ChatMessageItem.css.js +628 -0
  59. package/chat/ChatMessageItem/ChatMessageItem.js +156 -0
  60. package/chat/ChatSidebar/ChatSidebar.css.js +150 -0
  61. package/chat/ChatSidebar/ChatSidebar.js +230 -0
  62. package/chat/ChatSidebar/ChatSidebar.tpl.js +18 -0
  63. package/chat/ChatSidebar/constants.js +11 -0
  64. package/chat/ChatSidebarItem/ChatSidebarItem.css.js +445 -0
  65. package/chat/ChatSidebarItem/ChatSidebarItem.js +304 -0
  66. package/chat/ChatTranscript/ChatTranscript.css.js +90 -0
  67. package/chat/ChatTranscript/ChatTranscript.js +244 -0
  68. package/chat/chat-context.js +123 -0
  69. package/chat/message-model.js +156 -0
  70. package/cli.js +20 -0
  71. package/control/Button/Button.css.js +93 -0
  72. package/control/Button/Button.js +78 -0
  73. package/control/Button/Button.tpl.js +3 -0
  74. package/control/Field/Field.css.js +91 -0
  75. package/control/Field/Field.js +17 -0
  76. package/control/Field/Field.tpl.js +3 -0
  77. package/core/Connection.js +47 -0
  78. package/core/Editor.js +449 -0
  79. package/core/Frame.js +33 -0
  80. package/core/GraphMermaid.js +348 -0
  81. package/core/GraphText.js +228 -0
  82. package/core/Node.js +145 -0
  83. package/core/Portal.js +106 -0
  84. package/core/Socket.js +187 -0
  85. package/core/SubgraphNode.js +121 -0
  86. package/core/base-path.js +55 -0
  87. package/core/dom-utils.js +14 -0
  88. package/core/index.js +18 -0
  89. package/core/local-cache.js +26 -0
  90. package/core/state-sync.js +227 -0
  91. package/custom-elements.json +6380 -0
  92. package/discover.js +240 -0
  93. package/display/Badge/Badge.css.js +44 -0
  94. package/display/Badge/Badge.js +17 -0
  95. package/display/Badge/Badge.tpl.js +3 -0
  96. package/display/Banner/Banner.css.js +61 -0
  97. package/display/Banner/Banner.js +17 -0
  98. package/display/Banner/Banner.tpl.js +3 -0
  99. package/display/CodeBlock/CodeBlock.css.js +194 -0
  100. package/display/CodeBlock/CodeBlock.js +220 -0
  101. package/display/CodeBlock/CodeBlock.tpl.js +11 -0
  102. package/display/DataTable/DataTable.css.js +101 -0
  103. package/display/DataTable/DataTable.js +136 -0
  104. package/display/DataTable/DataTable.tpl.js +13 -0
  105. package/display/EmptyState/EmptyState.css.js +33 -0
  106. package/display/EmptyState/EmptyState.js +17 -0
  107. package/display/EmptyState/EmptyState.tpl.js +3 -0
  108. package/display/EventFeed/EventFeed.css.js +145 -0
  109. package/display/EventFeed/EventFeed.js +64 -0
  110. package/display/EventFeed/EventFeed.tpl.js +14 -0
  111. package/display/EventFeed/EventFeedItem.js +116 -0
  112. package/display/EventFeed/EventFeedItem.tpl.js +22 -0
  113. package/display/LoadingOverlay/LoadingOverlay.css.js +91 -0
  114. package/display/LoadingOverlay/LoadingOverlay.js +48 -0
  115. package/display/LoadingOverlay/LoadingOverlay.tpl.js +12 -0
  116. package/display/Metric/Metric.css.js +60 -0
  117. package/display/Metric/Metric.js +17 -0
  118. package/display/Metric/Metric.tpl.js +6 -0
  119. package/display/OutputGraphPreview/OutputGraphPreview.css.js +122 -0
  120. package/display/OutputGraphPreview/OutputGraphPreview.js +89 -0
  121. package/display/OutputGraphPreview/OutputGraphPreview.tpl.js +13 -0
  122. package/display/OutputListPreview/OutputListPreview.css.js +109 -0
  123. package/display/OutputListPreview/OutputListPreview.js +77 -0
  124. package/display/OutputListPreview/OutputListPreview.tpl.js +13 -0
  125. package/display/SourceEditor/SourceEditor.css.js +39 -0
  126. package/display/SourceEditor/SourceEditor.js +129 -0
  127. package/display/SourceEditor/SourceEditor.tpl.js +10 -0
  128. package/display/SourceViewer/SourceViewer.css.js +80 -0
  129. package/display/SourceViewer/SourceViewer.js +418 -0
  130. package/display/SourceViewer/SourceViewer.tpl.js +17 -0
  131. package/display/StatusRibbon/StatusRibbon.css.js +73 -0
  132. package/display/StatusRibbon/StatusRibbon.js +87 -0
  133. package/display/StatusRibbon/StatusRibbon.tpl.js +7 -0
  134. package/display/event-feed-adapter.js +72 -0
  135. package/display/format-utils.js +29 -0
  136. package/display/highlight.js +659 -0
  137. package/display/icons.js +37 -0
  138. package/display/markdown-formatter.js +60 -0
  139. package/display/network-approval-page.js +487 -0
  140. package/display/output-preview.js +261 -0
  141. package/effects/CellBg/CellBg.css.js +33 -0
  142. package/effects/CellBg/CellBg.js +410 -0
  143. package/effects/CellBg/CellBg.tpl.js +5 -0
  144. package/graph/canvas-adapter.js +223 -0
  145. package/graph/graph-algorithms.js +31 -0
  146. package/graph/index.js +46 -0
  147. package/graph/model.js +176 -0
  148. package/graph/project-graph-build.js +66 -0
  149. package/graph/project-graph-metadata.js +253 -0
  150. package/graph/project-package.js +128 -0
  151. package/graph/project-runtime.js +116 -0
  152. package/graph/project-transaction.js +284 -0
  153. package/graph/skeleton-utils.js +84 -0
  154. package/graph/theme-contract.js +36 -0
  155. package/graph/transaction-parser.js +56 -0
  156. package/icons/MaterialSymbols.js +69 -0
  157. package/icons/material-symbols-outlined-400.ttf +0 -0
  158. package/icons/material-symbols.css +24 -0
  159. package/index.js +95 -0
  160. package/inspector/InspectorPanel/InspectorPanel.css.js +375 -0
  161. package/inspector/InspectorPanel/InspectorPanel.js +368 -0
  162. package/inspector/InspectorPanel/InspectorPanel.tpl.js +96 -0
  163. package/inspector/TemplatePreview/TemplatePreview.css.js +104 -0
  164. package/inspector/TemplatePreview/TemplatePreview.js +145 -0
  165. package/inspector/TemplatePreview/TemplatePreview.tpl.js +33 -0
  166. package/interactions/ConnectFlow.js +304 -0
  167. package/interactions/Drag.js +104 -0
  168. package/interactions/Selector.js +133 -0
  169. package/interactions/SnapGrid.js +66 -0
  170. package/interactions/Zoom.js +139 -0
  171. package/layout/ActionZone/ActionZone.css.js +88 -0
  172. package/layout/ActionZone/ActionZone.js +261 -0
  173. package/layout/ActionZone/ActionZone.tpl.js +11 -0
  174. package/layout/CrossLayoutPortalBridge/CrossLayoutPortalBridge.js +255 -0
  175. package/layout/Layout/Layout.css.js +91 -0
  176. package/layout/Layout/Layout.js +637 -0
  177. package/layout/Layout/Layout.tpl.js +27 -0
  178. package/layout/LayoutNode/LayoutNode.css.js +302 -0
  179. package/layout/LayoutNode/LayoutNode.js +509 -0
  180. package/layout/LayoutNode/LayoutNode.tpl.js +39 -0
  181. package/layout/LayoutPreview/LayoutPreview.css.js +46 -0
  182. package/layout/LayoutPreview/LayoutPreview.js +102 -0
  183. package/layout/LayoutPreview/LayoutPreview.tpl.js +6 -0
  184. package/layout/LayoutRouter/LayoutRouter.js +274 -0
  185. package/layout/LayoutRouter/SectionRegistry.js +135 -0
  186. package/layout/LayoutRouter/routerSync.js +250 -0
  187. package/layout/LayoutSidebar/LayoutSidebar.css.js +411 -0
  188. package/layout/LayoutSidebar/LayoutSidebar.js +368 -0
  189. package/layout/LayoutSidebar/LayoutSidebar.tpl.js +26 -0
  190. package/layout/LayoutSidebar/SidebarSection.css.js +20 -0
  191. package/layout/LayoutSidebar/SidebarSection.js +184 -0
  192. package/layout/LayoutSidebar/SidebarSection.tpl.js +22 -0
  193. package/layout/LayoutTree.js +373 -0
  194. package/layout/PanelMenu/PanelMenu.css.js +43 -0
  195. package/layout/PanelMenu/PanelMenu.js +95 -0
  196. package/layout/PanelMenu/PanelMenu.tpl.js +17 -0
  197. package/layout/ProjectTabs/ProjectTabs.css.js +188 -0
  198. package/layout/ProjectTabs/ProjectTabs.js +77 -0
  199. package/layout/ProjectTabs/ProjectTabs.tpl.js +15 -0
  200. package/layout/index.js +40 -0
  201. package/list/ListDetailShell/ListDetailShell.css.js +128 -0
  202. package/list/ListDetailShell/ListDetailShell.js +72 -0
  203. package/list/ListDetailShell/ListDetailShell.tpl.js +36 -0
  204. package/list/ListItem/ListItem.css.js +111 -0
  205. package/list/ListItem/ListItem.js +66 -0
  206. package/list/ListItem/ListItem.tpl.js +18 -0
  207. package/locale/index.js +503 -0
  208. package/manifest/component-registry.js +2446 -0
  209. package/manifest/graph-schema.js +285 -0
  210. package/manifest/index.js +6 -0
  211. package/manifest/project-schema-catalog.js +246 -0
  212. package/manifest/rule-catalog.js +201 -0
  213. package/manifest/theme-catalog.js +2149 -0
  214. package/manifest/ui-schema-catalog.js +334 -0
  215. package/menu/ContextMenu/ContextMenu.css.js +61 -0
  216. package/menu/ContextMenu/ContextMenu.js +82 -0
  217. package/menu/ContextMenu/ContextMenu.tpl.js +19 -0
  218. package/navigation/QuickOpen/QuickOpen.css.js +92 -0
  219. package/navigation/QuickOpen/QuickOpen.js +185 -0
  220. package/navigation/QuickOpen/QuickOpen.tpl.js +15 -0
  221. package/navigation/quick-open-utils.js +101 -0
  222. package/node/CtrlItem/CtrlItem.css.js +41 -0
  223. package/node/CtrlItem/CtrlItem.js +24 -0
  224. package/node/CtrlItem/CtrlItem.tpl.js +17 -0
  225. package/node/GraphFrame/GraphFrame.css.js +66 -0
  226. package/node/GraphFrame/GraphFrame.js +32 -0
  227. package/node/GraphFrame/GraphFrame.tpl.js +13 -0
  228. package/node/GraphNode/GraphNode.css.js +815 -0
  229. package/node/GraphNode/GraphNode.js +173 -0
  230. package/node/GraphNode/GraphNode.tpl.js +33 -0
  231. package/node/NodeCallout/NodeCallout.css.js +91 -0
  232. package/node/NodeCallout/NodeCallout.js +281 -0
  233. package/node/NodeCallout/NodeCallout.tpl.js +8 -0
  234. package/node/NodeSocket/NodeSocket.css.js +68 -0
  235. package/node/NodeSocket/NodeSocket.js +26 -0
  236. package/node/NodeSocket/NodeSocket.tpl.js +7 -0
  237. package/node/PortItem/PortItem.css.js +93 -0
  238. package/node/PortItem/PortItem.js +87 -0
  239. package/node/PortItem/PortItem.tpl.js +10 -0
  240. package/package.json +165 -0
  241. package/palette/PaletteBrowser/PaletteBrowser.css.js +143 -0
  242. package/palette/PaletteBrowser/PaletteBrowser.js +152 -0
  243. package/palette/PaletteBrowser/PaletteBrowser.tpl.js +23 -0
  244. package/plugins/History.js +408 -0
  245. package/plugins/Readonly.js +60 -0
  246. package/rules/symbiote-3x.json +170 -0
  247. package/schemas/component-descriptor-v1.json +91 -0
  248. package/schemas/component-descriptor-v2.json +145 -0
  249. package/schemas/graph-model-v1.json +179 -0
  250. package/schemas/graph-v1.json +91 -0
  251. package/schemas/project-package-v1.json +102 -0
  252. package/schemas/project-transaction-v1.json +114 -0
  253. package/schemas/runtime-ui-v1.json +80 -0
  254. package/schemas/theme-rule-block-v1.json +73 -0
  255. package/shapes/CircleShape.js +79 -0
  256. package/shapes/CommentShape.js +35 -0
  257. package/shapes/DiamondShape.js +130 -0
  258. package/shapes/NodeShape.js +79 -0
  259. package/shapes/PillShape.js +91 -0
  260. package/shapes/RectShape.js +84 -0
  261. package/shapes/SVGShape.js +525 -0
  262. package/shapes/index.js +63 -0
  263. package/surface/Card/Card.css.js +57 -0
  264. package/surface/Card/Card.js +17 -0
  265. package/surface/Card/Card.tpl.js +3 -0
  266. package/themes/Palette.js +30 -0
  267. package/themes/Skin.js +113 -0
  268. package/themes/Theme.js +82 -0
  269. package/themes/carbon.js +135 -0
  270. package/themes/dark.js +140 -0
  271. package/themes/default-dark.js +714 -0
  272. package/themes/default-provider.css +635 -0
  273. package/themes/default-provider.js +718 -0
  274. package/themes/ebook.js +136 -0
  275. package/themes/grey.js +137 -0
  276. package/themes/light.js +139 -0
  277. package/themes/neon.js +138 -0
  278. package/themes/pcb.js +273 -0
  279. package/themes/synthwave.js +138 -0
  280. package/tokens/base.json +29 -0
  281. package/tokens/themes/carbon.json +11 -0
  282. package/tokens/themes/dark.json +12 -0
  283. package/tokens/themes/default-dark.json +1543 -0
  284. package/tokens/themes/default-provider.json +1543 -0
  285. package/tokens/themes/ebook.json +11 -0
  286. package/tokens/themes/grey.json +11 -0
  287. package/tokens/themes/light.json +12 -0
  288. package/tokens/themes/neon.json +11 -0
  289. package/tokens/themes/pcb.json +11 -0
  290. package/tokens/themes/synthwave.json +11 -0
  291. package/toolbar/QuickToolbar/QuickToolbar.css.js +152 -0
  292. package/toolbar/QuickToolbar/QuickToolbar.js +529 -0
  293. package/toolbar/QuickToolbar/QuickToolbar.tpl.js +34 -0
  294. package/tree/TreePanel/TreePanel.css.js +112 -0
  295. package/tree/TreePanel/TreePanel.js +147 -0
  296. package/tree/TreePanel/TreePanel.tpl.js +18 -0
  297. package/tree/TreeView/TreeView.css.js +122 -0
  298. package/tree/TreeView/TreeView.js +365 -0
  299. package/tree/TreeView/TreeView.tpl.js +10 -0
  300. package/ui/dialogs.js +221 -0
  301. package/ui/host-adapters.js +114 -0
  302. package/ui/index.js +660 -0
  303. package/ui/locale.js +50 -0
  304. package/ui/overlay-stack.js +89 -0
  305. package/ui/shared-styles.js +26 -0
  306. package/webmcp.js +37 -0
  307. package/xr/deep-graph.js +646 -0
  308. package/xr/emulation.js +198 -0
  309. package/xr/gesture.js +228 -0
  310. package/xr/html-canvas-renderer.js +472 -0
  311. package/xr/index.js +15 -0
  312. package/xr/layout-projection.js +1046 -0
  313. package/xr/panel-frame.js +128 -0
  314. package/xr/panel-host.js +267 -0
  315. package/xr/pointer.js +258 -0
  316. package/xr/scene-controller.js +242 -0
  317. package/xr/spatial-scene.js +212 -0
  318. package/xr/theme-bridge.js +105 -0
  319. package/xr/three-webxr-adapter.js +3439 -0
  320. package/xr/webgl-layer-renderer.js +419 -0
  321. package/xr/webxr.js +679 -0
  322. package/xr/workbench.js +516 -0
@@ -0,0 +1,156 @@
1
+ import Symbiote, { html } from '@symbiotejs/symbiote';
2
+ import { escapeHtml, formatMarkdown } from '../../display/markdown-formatter.js';
3
+ import { translate } from '../../locale/index.js';
4
+ import { summarizeToolInput } from '../message-model.js';
5
+ import css from './ChatMessageItem.css.js';
6
+
7
+ export function stringifyBlock(value) {
8
+ if (value == null) return '';
9
+ return typeof value === 'string' ? value : JSON.stringify(value, null, 2);
10
+ }
11
+
12
+ export function truncateResult(value) {
13
+ let text = stringifyBlock(value);
14
+ return text.length > 500 ? `${text.slice(0, 500)}\n...` : text;
15
+ }
16
+
17
+ export class ChatMessageItem extends Symbiote {
18
+ init$ = {
19
+ type: '',
20
+ role: '',
21
+ text: '',
22
+ isStreaming: false,
23
+ isLatestTool: false,
24
+ name: '',
25
+ input: null,
26
+ result: null,
27
+ done: false,
28
+ elapsedText: '',
29
+ status: '',
30
+ metaHtml: '',
31
+ workSummaryHtml: '',
32
+ copyText: '',
33
+ cardItems: [],
34
+ messageClass: 'message',
35
+ bodyHtml: '',
36
+ };
37
+
38
+ renderCallback() {
39
+ this.sub('type', () => this._renderBody());
40
+ this.sub('role', () => this._renderBody());
41
+ this.sub('text', () => this._renderBody());
42
+ this.sub('isStreaming', () => this._renderBody());
43
+ this.sub('isLatestTool', () => this._renderBody());
44
+ this.sub('name', () => this._renderBody());
45
+ this.sub('input', () => this._renderBody());
46
+ this.sub('result', () => this._renderBody());
47
+ this.sub('done', () => this._renderBody());
48
+ this.sub('elapsedText', () => this._renderBody());
49
+ this.sub('status', () => this._renderBody());
50
+ this.sub('metaHtml', () => this._renderBody());
51
+ this.sub('workSummaryHtml', () => this._renderBody());
52
+ this.sub('copyText', () => this._renderBody());
53
+ this.sub('cardItems', () => this._renderBody());
54
+ }
55
+
56
+ _renderBody() {
57
+ let role = this.$.role || this.$.type;
58
+ this.$.messageClass = `message ${role || ''}${this.$.isStreaming ? ' streaming' : ''}`.trim();
59
+
60
+ if (role === 'tool') {
61
+ this.$.bodyHtml = this._renderTool();
62
+ } else if (role === 'board') {
63
+ this.$.bodyHtml = this._renderBoard();
64
+ } else if (role === 'thinking') {
65
+ this.$.bodyHtml = this._renderThinking();
66
+ } else {
67
+ this.$.bodyHtml = this._renderTextMessage();
68
+ }
69
+ }
70
+
71
+ _renderTool() {
72
+ let icon = this.$.isStreaming ? 'build_circle' : 'build';
73
+ let spinClass = this.$.isStreaming ? 'spin-icon' : '';
74
+ let openAttr = this.$.isLatestTool ? ' open' : '';
75
+ let summary = summarizeToolInput(this.$.input);
76
+ let summaryHtml = summary
77
+ ? `<span class="tool-summary" title="${escapeHtml(summary)}">${escapeHtml(summary)}</span>`
78
+ : '';
79
+ let htmlStr = `<details class="tool-card"${openAttr}>
80
+ <summary class="tool-header"><span class="material-symbols-outlined tool-icon ${spinClass}">${icon}</span><span class="tool-name">${escapeHtml(this.$.name || 'tool')}</span>${summaryHtml}</summary>`;
81
+
82
+ if (this.$.input) {
83
+ htmlStr += `<div class="tool-section"><div class="tool-label">${escapeHtml(translate('chat.message.input'))}</div><pre class="tool-code">${escapeHtml(stringifyBlock(this.$.input))}</pre></div>`;
84
+ }
85
+
86
+ if (this.$.result) {
87
+ htmlStr += `<div class="tool-section"><div class="tool-label">${escapeHtml(translate('chat.message.result'))}</div><pre class="tool-code">${escapeHtml(truncateResult(this.$.result))}</pre></div>`;
88
+ } else if (this.$.isStreaming) {
89
+ htmlStr += `<div class="tool-section tool-waiting"><em>${escapeHtml(translate('chat.message.running'))}</em></div>`;
90
+ }
91
+
92
+ return `${htmlStr}</details>`;
93
+ }
94
+
95
+ _renderBoard() {
96
+ let cardsHtml = (this.$.cardItems || []).map((card) => {
97
+ let id = String(card?.id || '');
98
+ let title = card?.title || (id ? `${id.substring(0, 8)}...` : translate('chat.message.item'));
99
+ let status = card?.status || (this.$.isStreaming ? 'running' : 'idle');
100
+ let statusIcon = card?.icon || (status === 'running' ? 'pending' : 'schedule');
101
+ let spinClass = status === 'running' ? 'spin-icon' : '';
102
+ let statusText = card?.statusText || (status === 'running' ? translate('chat.message.running') : translate('chat.message.queued'));
103
+ let linkedAttr = card?.linkId ? ` data-link-id="${escapeHtml(card.linkId)}"` : '';
104
+ return `<div class="status-card" data-card-id="${escapeHtml(id)}"${linkedAttr} data-status="${escapeHtml(status)}">
105
+ <div class="status-card-header">
106
+ <span class="material-symbols-outlined ${spinClass}">${escapeHtml(statusIcon)}</span><span class="card-title">${escapeHtml(title)}</span>
107
+ </div>
108
+ <div class="status-card-status">${escapeHtml(statusText)}</div>
109
+ <div class="status-card-events"></div>
110
+ </div>`;
111
+ }).join('');
112
+
113
+ return `<div class="status-board">${cardsHtml}</div>`;
114
+ }
115
+
116
+ _renderThinking() {
117
+ let className = this.$.done ? 'work-summary' : 'thinking-block';
118
+ let openAttr = this.$.done ? '' : ' open';
119
+ let details = `<details class="${className}"${openAttr}>`;
120
+
121
+ if (this.$.done) {
122
+ details += `<summary><span class="material-symbols-outlined work-summary-icon">check_circle</span>${escapeHtml(translate('chat.message.workedFor', { elapsed: this.$.elapsedText }))}${this._renderCopyButton(this.$.copyText)}</summary>`;
123
+ } else {
124
+ let statusHtml = this.$.status ? `<span class="thinking-status">${escapeHtml(this.$.status)}</span>` : '';
125
+ details += `<summary><span class="material-symbols-outlined thinking-icon spin-icon">pending</span>${escapeHtml(translate('chat.message.thinkingFor', { elapsed: this.$.elapsedText }))}${statusHtml}</summary>`;
126
+ }
127
+
128
+ if (this.$.done && this.$.metaHtml) {
129
+ details += `<div class="work-body">${this.$.metaHtml}</div>`;
130
+ }
131
+
132
+ details += '</details>';
133
+
134
+ if (!this.$.done) return details;
135
+ return `<div class="work-summary-wrap">${details}</div>`;
136
+ }
137
+
138
+ _renderTextMessage() {
139
+ let cursor = this.$.isStreaming ? '<span class="streaming-cursor"></span>' : '';
140
+ let summary = this.$.workSummaryHtml || '';
141
+ return `<div class="msg-content">${formatMarkdown(this.$.text)}${cursor}</div>${summary}`;
142
+ }
143
+
144
+ _renderCopyButton(copyText) {
145
+ return copyText
146
+ ? `<button class="work-copy-btn" type="button" title="${escapeHtml(translate('chat.message.copyResponse'))}" data-copy-text="${escapeHtml(copyText)}"><span class="material-symbols-outlined">content_copy</span></button>`
147
+ : '';
148
+ }
149
+ }
150
+
151
+ ChatMessageItem.template = html`
152
+ <div ${{ className: 'messageClass', innerHTML: 'bodyHtml', onclick: '^onMessageItemClick' }}></div>
153
+ `;
154
+ ChatMessageItem.rootStyles = css;
155
+
156
+ ChatMessageItem.reg('chat-message-item');
@@ -0,0 +1,150 @@
1
+ export default `
2
+ :host,
3
+ chat-sidebar-shell {
4
+ display: flex;
5
+ height: 100%;
6
+ width: var(--chat-nav-width, 200px);
7
+ min-width: var(--chat-nav-width, 200px);
8
+ flex: 0 0 var(--chat-nav-width, 200px);
9
+ position: relative;
10
+ z-index: 10;
11
+ transition: width 0.2s ease, min-width 0.2s ease, flex-basis 0.2s ease;
12
+ }
13
+
14
+ :host([resizing]),
15
+ chat-sidebar-shell[resizing] {
16
+ transition: none;
17
+ }
18
+
19
+ :host([resizing]) .chat-nav,
20
+ chat-sidebar-shell[resizing] .chat-nav {
21
+ transition: none;
22
+ }
23
+
24
+ .chat-nav {
25
+ height: 100%;
26
+ width: var(--chat-nav-width, 200px);
27
+ min-width: var(--chat-nav-width, 200px);
28
+ flex-shrink: 0;
29
+ position: relative;
30
+ display: flex;
31
+ flex-direction: column;
32
+ border-right: none;
33
+ background: var(--sn-node-bg);
34
+ overflow: hidden;
35
+ transition: width 0.2s ease, min-width 0.2s ease;
36
+ user-select: none;
37
+ }
38
+
39
+ .chat-nav[collapsed] {
40
+ width: var(--chat-nav-width, 48px);
41
+ min-width: var(--chat-nav-width, 48px);
42
+ overflow: visible;
43
+ }
44
+
45
+ .chat-nav[resizing],
46
+ .chat-nav[resizing] + * {
47
+ user-select: none;
48
+ }
49
+
50
+ .chat-nav-resize-handle {
51
+ position: absolute;
52
+ top: 0;
53
+ right: -1px;
54
+ bottom: 0;
55
+ width: 4px;
56
+ cursor: col-resize;
57
+ background: transparent;
58
+ z-index: 20;
59
+ transition: background 0.15s ease;
60
+ }
61
+
62
+ .chat-nav-resize-handle:hover,
63
+ .chat-nav-resize-handle.dragging,
64
+ .chat-nav[resizing] .chat-nav-resize-handle {
65
+ background: var(--sn-layout-resizer-hover-bg);
66
+ }
67
+
68
+ .chat-nav-header {
69
+ display: flex;
70
+ align-items: center;
71
+ gap: 2px;
72
+ padding: 2px 4px;
73
+ min-height: 28px;
74
+ background: var(--sn-node-bg);
75
+ border-bottom: none;
76
+ flex-shrink: 0;
77
+ }
78
+
79
+ .chat-nav[collapsed] .chat-nav-header {
80
+ flex-direction: column-reverse;
81
+ justify-content: flex-start;
82
+ padding: 4px 0;
83
+ gap: 8px;
84
+ }
85
+
86
+ .chat-nav-header .nav-spacer {
87
+ flex: 1;
88
+ }
89
+
90
+ .chat-nav[collapsed] .nav-spacer {
91
+ display: none;
92
+ }
93
+
94
+ .chat-nav-header .nav-title {
95
+ font-size: 11px;
96
+ font-weight: 600;
97
+ color: var(--sn-text-dim);
98
+ text-transform: uppercase;
99
+ letter-spacing: 0.5px;
100
+ white-space: nowrap;
101
+ overflow: hidden;
102
+ }
103
+
104
+ .chat-nav[collapsed] .nav-title {
105
+ display: none;
106
+ }
107
+
108
+ .chat-nav-collapse-icon {
109
+ transition: transform 0.2s ease;
110
+ }
111
+
112
+ .chat-nav[collapsed] .chat-nav-collapse-icon {
113
+ transform: rotate(180deg);
114
+ }
115
+
116
+ .nav-btn {
117
+ display: flex;
118
+ align-items: center;
119
+ justify-content: center;
120
+ padding: 4px 6px;
121
+ background: transparent;
122
+ border: none;
123
+ border-radius: 4px;
124
+ cursor: pointer;
125
+ color: var(--sn-text-dim);
126
+ font-size: 0.75rem;
127
+ transition: background 0.1s, color 0.1s;
128
+ flex-shrink: 0;
129
+ }
130
+
131
+ .nav-btn .material-symbols-outlined {
132
+ font-size: 16px;
133
+ }
134
+
135
+ .nav-btn:hover {
136
+ background: var(--sn-node-hover);
137
+ color: var(--sn-text);
138
+ }
139
+
140
+ .chat-items {
141
+ flex: 1;
142
+ overflow-y: auto;
143
+ padding: 4px 0;
144
+ }
145
+
146
+ .chat-nav[collapsed] .chat-items {
147
+ overflow: visible;
148
+ }
149
+
150
+ `;
@@ -0,0 +1,230 @@
1
+ import Symbiote from '@symbiotejs/symbiote';
2
+ import '../ChatSidebarItem/ChatSidebarItem.js';
3
+ import template from './ChatSidebar.tpl.js';
4
+ import css from './ChatSidebar.css.js';
5
+ import { translate } from '../../locale/index.js';
6
+ import {
7
+ AUTO_COLLAPSE_WIDTH,
8
+ AUTO_UNCOLLAPSE_WIDTH,
9
+ COLLAPSE_DRAG_THRESHOLD,
10
+ COLLAPSED_NAV_WIDTH,
11
+ DEFAULT_NAV_WIDTH,
12
+ MAX_NAV_WIDTH,
13
+ MIN_NAV_WIDTH,
14
+ clampChatSidebarWidth,
15
+ } from './constants.js';
16
+
17
+ export {
18
+ AUTO_COLLAPSE_WIDTH,
19
+ AUTO_UNCOLLAPSE_WIDTH,
20
+ COLLAPSE_DRAG_THRESHOLD,
21
+ COLLAPSED_NAV_WIDTH,
22
+ DEFAULT_NAV_WIDTH,
23
+ MAX_NAV_WIDTH,
24
+ MIN_NAV_WIDTH,
25
+ clampChatSidebarWidth,
26
+ } from './constants.js';
27
+
28
+ function emit(el, type, detail = {}) {
29
+ el.dispatchEvent(new CustomEvent(type, { bubbles: true, composed: true, detail }));
30
+ }
31
+
32
+ export class ChatSidebarShell extends Symbiote {
33
+ static isoMode = true;
34
+
35
+ init$ = {
36
+ navCollapsed: true,
37
+ navWidth: DEFAULT_NAV_WIDTH,
38
+ groupDividers: true,
39
+ chats: [],
40
+ title: translate('chat.sidebar.title'),
41
+ newChatTitle: translate('chat.sidebar.new'),
42
+ onToggleNav: () => {
43
+ this._autoCollapsed = false;
44
+ this.setCollapsed(!this.$.navCollapsed);
45
+ },
46
+ onResizeStart: (event) => {
47
+ this._startResize(event);
48
+ },
49
+ onNewChat: () => {
50
+ emit(this, 'chat-sidebar-new');
51
+ },
52
+ onChatClick: (event) => {
53
+ this._handleChatClick(event);
54
+ },
55
+ };
56
+
57
+ initCallback() {
58
+ this.sub('navCollapsed', (value) => {
59
+ this._applyNavWidth();
60
+ emit(this, 'chat-sidebar-collapse-change', {
61
+ collapsed: Boolean(value),
62
+ auto: Boolean(this._autoCollapsed),
63
+ });
64
+ });
65
+
66
+ this.sub('navWidth', (value) => {
67
+ this._applyNavWidth();
68
+ emit(this, 'chat-sidebar-width-change', {
69
+ width: clampChatSidebarWidth(Number(value) || DEFAULT_NAV_WIDTH),
70
+ });
71
+ });
72
+
73
+ this.sub('groupDividers', () => {
74
+ this._applyGroupDividers();
75
+ });
76
+
77
+ if (typeof ResizeObserver === 'function') {
78
+ this._resizeObserver = new ResizeObserver(() => this._syncCollapseForAvailableWidth());
79
+ let shell = this.closest('.chat-shell') || this.parentElement;
80
+ if (shell) this._resizeObserver.observe(shell);
81
+ }
82
+
83
+ queueMicrotask(() => {
84
+ this._applyNavWidth();
85
+ this._applyGroupDividers();
86
+ this._syncCollapseForAvailableWidth();
87
+ });
88
+ }
89
+
90
+ disconnectedCallback() {
91
+ super.disconnectedCallback();
92
+ this._resizeObserver?.disconnect();
93
+ }
94
+
95
+ setChats(chats = []) {
96
+ this.$.chats = Array.isArray(chats) ? chats : [];
97
+ let raf = globalThis.requestAnimationFrame || ((callback) => setTimeout(callback, 0));
98
+ raf(() => this.syncExpandedChatItems(this.$.chats));
99
+ }
100
+
101
+ setCollapsed(collapsed, { auto = false } = {}) {
102
+ this._autoCollapsed = auto;
103
+ this.$.navCollapsed = Boolean(collapsed);
104
+ }
105
+
106
+ setWidth(width) {
107
+ this.$.navWidth = clampChatSidebarWidth(Number(width) || DEFAULT_NAV_WIDTH);
108
+ }
109
+
110
+ setGroupDividers(enabled) {
111
+ this.$.groupDividers = Boolean(enabled);
112
+ }
113
+
114
+ _applyNavWidth() {
115
+ let width = this.$.navCollapsed ? COLLAPSED_NAV_WIDTH : clampChatSidebarWidth(this.$.navWidth);
116
+ this.style.setProperty('--chat-nav-width', `${width}px`);
117
+ let nav = this.querySelector('.chat-nav');
118
+ if (nav) nav.toggleAttribute('collapsed', this.$.navCollapsed);
119
+ this.toggleAttribute('collapsed', this.$.navCollapsed);
120
+ }
121
+
122
+ _applyGroupDividers() {
123
+ let nav = this.querySelector('.chat-nav');
124
+ if (nav) nav.toggleAttribute('data-group-dividers', Boolean(this.$.groupDividers));
125
+ }
126
+
127
+ _syncCollapseForAvailableWidth() {
128
+ if (this._isResizing) return;
129
+ let shell = this.closest('.chat-shell') || this.parentElement;
130
+ if (!shell) return;
131
+ let width = shell.getBoundingClientRect().width;
132
+ if (width <= AUTO_COLLAPSE_WIDTH && !this.$.navCollapsed) {
133
+ this.setCollapsed(true, { auto: true });
134
+ } else if (width >= AUTO_UNCOLLAPSE_WIDTH && this.$.navCollapsed && this._autoCollapsed) {
135
+ this.setCollapsed(false, { auto: false });
136
+ }
137
+ }
138
+
139
+ _startResize(event) {
140
+ event.preventDefault();
141
+ event.stopPropagation();
142
+
143
+ let nav = this.querySelector('.chat-nav');
144
+ let handle = this.querySelector('.chat-nav-resize-handle');
145
+ if (!nav) return;
146
+
147
+ let startX = event.clientX;
148
+ let startWidth = this.$.navCollapsed ? COLLAPSED_NAV_WIDTH : nav.getBoundingClientRect().width;
149
+ let wasCollapsed = this.$.navCollapsed;
150
+ this._isResizing = true;
151
+ this.setAttribute('resizing', '');
152
+ handle?.classList.add('dragging');
153
+ nav.setAttribute('resizing', '');
154
+
155
+ let onMove = (moveEvent) => {
156
+ let rawWidth = startWidth + (moveEvent.clientX - startX);
157
+ if (wasCollapsed && rawWidth > COLLAPSE_DRAG_THRESHOLD) {
158
+ this.setCollapsed(false);
159
+ wasCollapsed = false;
160
+ startX = moveEvent.clientX;
161
+ startWidth = MIN_NAV_WIDTH;
162
+ this.setWidth(startWidth);
163
+ return;
164
+ }
165
+
166
+ if (!wasCollapsed && rawWidth < COLLAPSE_DRAG_THRESHOLD) {
167
+ this.setCollapsed(true);
168
+ return;
169
+ }
170
+
171
+ if (!this.$.navCollapsed) {
172
+ this.setWidth(rawWidth);
173
+ }
174
+ };
175
+
176
+ let onUp = () => {
177
+ handle?.classList.remove('dragging');
178
+ nav.removeAttribute('resizing');
179
+ this.removeAttribute('resizing');
180
+ this._isResizing = false;
181
+ document.removeEventListener('pointermove', onMove);
182
+ document.removeEventListener('pointerup', onUp);
183
+ this._autoCollapsed = false;
184
+ };
185
+
186
+ document.addEventListener('pointermove', onMove);
187
+ document.addEventListener('pointerup', onUp);
188
+ }
189
+
190
+ _handleChatClick(event) {
191
+ let btnDelete = event.target.closest('.chat-item-delete');
192
+ let item = event.target.closest('.chat-item, .chat-item-child');
193
+ if (!item) return;
194
+
195
+ let chatId = item.dataset.id;
196
+ if (btnDelete) {
197
+ event.stopPropagation();
198
+ emit(this, 'chat-sidebar-delete', { id: chatId });
199
+ return;
200
+ }
201
+
202
+ let expandIcon = event.target.closest('.chat-expand-icon');
203
+ if (expandIcon) {
204
+ event.stopPropagation();
205
+ let subContainer = this.querySelector(`.chat-sub-items[data-parent="${chatId}"]`);
206
+ if (!subContainer) return;
207
+ let isExpanded = subContainer.hasAttribute('expanded');
208
+ subContainer.toggleAttribute('expanded', !isExpanded);
209
+ item.classList.toggle('chat-item-expanded', !isExpanded);
210
+ emit(this, 'chat-sidebar-toggle', { id: chatId, expanded: !isExpanded });
211
+ return;
212
+ }
213
+
214
+ if (chatId) {
215
+ emit(this, 'chat-sidebar-select', { id: chatId });
216
+ }
217
+ }
218
+
219
+ syncExpandedChatItems(chats = this.$.chats) {
220
+ for (let chat of chats || []) {
221
+ let item = [...this.querySelectorAll('chat-sidebar-item')].find((el) => el.$?.id === chat.id);
222
+ if (!item) continue;
223
+ item.$.isExpanded = Boolean(chat.isExpanded);
224
+ }
225
+ }
226
+ }
227
+
228
+ ChatSidebarShell.template = template;
229
+ ChatSidebarShell.rootStyles = css;
230
+ ChatSidebarShell.reg('chat-sidebar-shell');
@@ -0,0 +1,18 @@
1
+ import { html } from '@symbiotejs/symbiote';
2
+
3
+ export default html`
4
+ <div class="chat-nav" ${{ onclick: 'onChatClick' }}>
5
+ <div class="chat-nav-header">
6
+ <button class="nav-btn nav-btn-add" ${{ onclick: 'onNewChat', title: 'newChatTitle' }}>
7
+ <span class="material-symbols-outlined">add</span>
8
+ </button>
9
+ <span class="nav-title" ${{ textContent: 'title' }}></span>
10
+ <div class="nav-spacer"></div>
11
+ <button class="nav-btn" ${{ onclick: 'onToggleNav' }}>
12
+ <span class="material-symbols-outlined chat-nav-collapse-icon">chevron_left</span>
13
+ </button>
14
+ </div>
15
+ <div class="chat-items" itemize="chats" item-tag="chat-sidebar-item"></div>
16
+ <div class="chat-nav-resize-handle" ${{ onpointerdown: 'onResizeStart' }}></div>
17
+ </div>
18
+ `;
@@ -0,0 +1,11 @@
1
+ export const DEFAULT_NAV_WIDTH = 200;
2
+ export const MIN_NAV_WIDTH = 120;
3
+ export const MAX_NAV_WIDTH = 420;
4
+ export const COLLAPSED_NAV_WIDTH = 48;
5
+ export const COLLAPSE_DRAG_THRESHOLD = 72;
6
+ export const AUTO_COLLAPSE_WIDTH = 560;
7
+ export const AUTO_UNCOLLAPSE_WIDTH = 660;
8
+
9
+ export function clampChatSidebarWidth(width) {
10
+ return Math.max(MIN_NAV_WIDTH, Math.min(MAX_NAV_WIDTH, Math.round(width)));
11
+ }