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,509 @@
1
+ /**
2
+ * @fileoverview LayoutNode - Universal recursive layout node
3
+ * Renders panel or split based on node type.
4
+ * Split nodes recursively create child LayoutNodes.
5
+ * Panels include action zones for split/join gestures.
6
+ */
7
+
8
+ import Symbiote from '@symbiotejs/symbiote';
9
+ import { ensureMaterialSymbols } from '../../icons/MaterialSymbols.js';
10
+ import { template } from './LayoutNode.tpl.js';
11
+ import { styles } from './LayoutNode.css.js';
12
+ import './../ActionZone/ActionZone.js';
13
+ import { translate } from '../../locale/index.js';
14
+
15
+ const LAYOUT_NODE_ICONS = [
16
+ 'arrow_drop_down',
17
+ 'chevron_left',
18
+ 'chevron_right',
19
+ 'dashboard',
20
+ 'expand_less',
21
+ 'expand_more',
22
+ 'fullscreen',
23
+ 'fullscreen_exit',
24
+ ];
25
+
26
+ export class LayoutNode extends Symbiote {
27
+ static isoMode = true;
28
+
29
+ init$ = {
30
+
31
+ nodeData: null,
32
+
33
+
34
+ nodeType: 'panel',
35
+ isPanel: true,
36
+ isSplit: false,
37
+ direction: 'horizontal',
38
+ ratio: 0.5,
39
+ panelType: 'default',
40
+ nodeId: '',
41
+
42
+
43
+ panelTitle: 'Panel',
44
+ panelIcon: 'dashboard',
45
+ panelChrome: true,
46
+
47
+
48
+ isCollapsed: false,
49
+ canCollapse: true,
50
+ collapseDirection: 'vertical',
51
+ collapseIcon: 'expand_less',
52
+ savedRatio: 0.5,
53
+ isFullscreen: false,
54
+ fullscreenIcon: 'fullscreen',
55
+ collapseTitle: translate('layout.collapse'),
56
+ fullscreenTitle: translate('layout.fullscreen'),
57
+
58
+
59
+ firstStyle: '',
60
+ secondStyle: '',
61
+
62
+
63
+ '^panelTypes': {},
64
+ '^fullscreenPanelId': null,
65
+ '^panelChrome': true,
66
+
67
+
68
+ onResizerDown: (e) => this._startResize(e),
69
+ onTypeClick: (e) => this._showTypeMenu(e),
70
+ onCollapseClick: () => this._toggleCollapse(),
71
+ onExpandClick: () => this._toggleCollapse(),
72
+ onFullscreenClick: () => this._toggleFullscreen(),
73
+ };
74
+
75
+ renderCallback() {
76
+ ensureMaterialSymbols(LAYOUT_NODE_ICONS);
77
+
78
+ this.sub('nodeData', (data) => {
79
+ if (!data) return;
80
+
81
+ this.$.nodeType = data.type || 'panel';
82
+ this.$.isPanel = this.$.nodeType === 'panel';
83
+ this.$.isSplit = this.$.nodeType === 'split';
84
+ this.$.direction = data.direction || 'horizontal';
85
+ this.$.ratio = data.ratio || 0.5;
86
+ this.$.panelType = data.panelType || 'default';
87
+ this.$.nodeId = data.id || '';
88
+
89
+
90
+ if (data.type === 'panel') {
91
+ this.$.isCollapsed = data.collapsed || false;
92
+ this.toggleAttribute('collapsed', this.$.isCollapsed);
93
+ this.#syncHostAttribute('collapse-dir', this.$.isCollapsed ? this.$.collapseDirection : '');
94
+
95
+ if (this.$.isCollapsed) {
96
+ if (this.$.collapseDirection === 'horizontal') {
97
+ this.$.collapseIcon = 'chevron_right';
98
+ } else {
99
+ this.$.collapseIcon = 'expand_more';
100
+ }
101
+ } else {
102
+ if (this.$.collapseDirection === 'horizontal') {
103
+ this.$.collapseIcon = 'chevron_left';
104
+ } else {
105
+ this.$.collapseIcon = 'expand_less';
106
+ }
107
+ }
108
+ }
109
+
110
+ this._updateStyles();
111
+ this._updatePanelInfo();
112
+ this._renderNode(data);
113
+ });
114
+
115
+
116
+ this.sub('^panelTypes', () => {
117
+ this._updatePanelInfo();
118
+ });
119
+
120
+
121
+ this.sub('panelType', () => {
122
+ this._updatePanelInfo();
123
+ });
124
+
125
+
126
+ if (this.$.nodeData) {
127
+ this.sub('nodeData', () => {});
128
+ }
129
+ }
130
+
131
+ _updateStyles() {
132
+ let ratio = this.$.ratio;
133
+ let dir = this.$.direction;
134
+ let data = this.$.nodeData;
135
+
136
+
137
+ let firstCollapsed = data?.first?.collapsed || false;
138
+ let secondCollapsed = data?.second?.collapsed || false;
139
+
140
+
141
+ const COLLAPSED_SIZE = dir === 'horizontal' ? '32px' : '28px';
142
+
143
+ if (firstCollapsed) {
144
+
145
+ if (dir === 'horizontal') {
146
+ this.$.firstStyle = `width: ${COLLAPSED_SIZE}; height: 100%; flex: 0 0 ${COLLAPSED_SIZE};`;
147
+ this.$.secondStyle = 'flex: 1; height: 100%;';
148
+ } else {
149
+ this.$.firstStyle = `height: ${COLLAPSED_SIZE}; width: 100%; flex: 0 0 ${COLLAPSED_SIZE};`;
150
+ this.$.secondStyle = 'flex: 1; width: 100%;';
151
+ }
152
+ } else if (secondCollapsed) {
153
+
154
+ if (dir === 'horizontal') {
155
+ this.$.firstStyle = 'flex: 1; height: 100%;';
156
+ this.$.secondStyle = `width: ${COLLAPSED_SIZE}; height: 100%; flex: 0 0 ${COLLAPSED_SIZE};`;
157
+ } else {
158
+ this.$.firstStyle = 'flex: 1; width: 100%;';
159
+ this.$.secondStyle = `height: ${COLLAPSED_SIZE}; width: 100%; flex: 0 0 ${COLLAPSED_SIZE};`;
160
+ }
161
+ } else {
162
+
163
+ if (dir === 'horizontal') {
164
+ this.$.firstStyle = `width: ${ratio * 100}%; height: 100%;`;
165
+ this.$.secondStyle = `width: ${(1 - ratio) * 100}%; height: 100%;`;
166
+ } else {
167
+ this.$.firstStyle = `height: ${ratio * 100}%; width: 100%;`;
168
+ this.$.secondStyle = `height: ${(1 - ratio) * 100}%; width: 100%;`;
169
+ }
170
+ }
171
+ }
172
+
173
+ _updatePanelInfo() {
174
+ let panelTypes = this.$['^panelTypes'] || {};
175
+ let config = panelTypes[this.$.panelType] || {};
176
+ this.$.panelTitle = config.title || this.$.panelType;
177
+ this.$.panelIcon = config.icon || 'dashboard';
178
+ ensureMaterialSymbols([this.$.panelIcon]);
179
+
180
+
181
+ this._injectPanelComponent(config);
182
+
183
+
184
+ let container = this.parentElement;
185
+ if (!container) return;
186
+ let isSplitChild =
187
+ container &&
188
+ (container.classList.contains('split-first') || container.classList.contains('split-second'));
189
+
190
+
191
+ let siblingExists = false;
192
+ let siblingCollapsed = false;
193
+ let isFirst = false;
194
+
195
+ if (isSplitChild) {
196
+ isFirst = container.classList.contains('split-first');
197
+
198
+ let siblingContainer = isFirst
199
+ ? container.parentElement.querySelector(':scope > .split-second')
200
+ : container.parentElement.querySelector(':scope > .split-first');
201
+ siblingExists = !!siblingContainer;
202
+
203
+
204
+ if (siblingContainer) {
205
+ let siblingNode = siblingContainer.querySelector(':scope > layout-node');
206
+
207
+ if (siblingNode?.getAttribute('node-type') === 'panel') {
208
+ siblingCollapsed = siblingNode.$.isCollapsed || false;
209
+ }
210
+ }
211
+ }
212
+
213
+
214
+ if (this.$.nodeType === 'panel') {
215
+
216
+ this.$.canCollapse = !!isSplitChild && siblingExists && !siblingCollapsed;
217
+
218
+ if (isSplitChild) {
219
+
220
+ let parentNode = container.closest('layout-node');
221
+ if (!parentNode && container.getRootNode() instanceof ShadowRoot) {
222
+ parentNode = container.getRootNode().host;
223
+ }
224
+
225
+ if (parentNode) {
226
+ let parentDir = parentNode.getAttribute('direction');
227
+ this.$.collapseDirection = parentDir;
228
+ if (this.$.isCollapsed) {
229
+ this.#syncHostAttribute('collapse-dir', parentDir);
230
+ }
231
+
232
+
233
+ if (!this.$.isCollapsed) {
234
+ if (parentDir === 'horizontal') {
235
+ this.$.collapseIcon = isFirst ? 'chevron_left' : 'chevron_right';
236
+ } else {
237
+ this.$.collapseIcon = isFirst ? 'expand_less' : 'expand_more';
238
+ }
239
+ }
240
+ }
241
+ }
242
+ }
243
+ }
244
+
245
+ /**
246
+ * Inject custom component into panel content.
247
+ * Hides existing components instead of destroying them to preserve state.
248
+ * Uses style.display instead of hidden attribute because components may have
249
+ * CSS rules (e.g. display:block) that override the hidden attribute.
250
+ * @param {Object} config - Panel type configuration
251
+ */
252
+ _injectPanelComponent(config) {
253
+ let contentEl = this.ref.panelContent;
254
+ if (!contentEl) return;
255
+
256
+ let componentTag = config.component;
257
+ if (!componentTag) return;
258
+
259
+
260
+ for (const child of contentEl.children) {
261
+ child.style.display = 'none';
262
+ }
263
+
264
+
265
+ let existing = contentEl.querySelector(componentTag);
266
+ if (existing) {
267
+ existing.style.display = '';
268
+ return;
269
+ }
270
+
271
+
272
+ let component = document.createElement(componentTag);
273
+ component.dataset.panelId = this.$.nodeData?.id || '';
274
+ contentEl.appendChild(component);
275
+ }
276
+
277
+ _renderNode(data) {
278
+ this.$.panelChrome = this.$.panelChrome !== false && this.$['^panelChrome'] !== false;
279
+ this.setAttribute('panel-chrome', this.$.panelChrome ? 'default' : 'none');
280
+
281
+ let prevType = this.getAttribute('node-type');
282
+ this.#syncHostAttribute('node-type', data.type);
283
+
284
+ if (data.type === 'split') {
285
+ this.#syncHostAttribute('direction', data.direction);
286
+ this._renderSplit(data);
287
+ } else {
288
+ this.removeAttribute('direction');
289
+
290
+
291
+ if (prevType === 'split') {
292
+ if (this.ref.first) this.ref.first.replaceChildren();
293
+ if (this.ref.second) this.ref.second.replaceChildren();
294
+ }
295
+
296
+
297
+ this._setupActionZones(data.id);
298
+
299
+ this._updatePanelInfo();
300
+ }
301
+ }
302
+
303
+ _renderSplit(data) {
304
+
305
+
306
+ if (data.first && this.ref.first) {
307
+ this._ensureChildNode(this.ref.first, data.first);
308
+ }
309
+ if (data.second && this.ref.second) {
310
+ this._ensureChildNode(this.ref.second, data.second);
311
+ }
312
+ }
313
+
314
+ /**
315
+ * @param {HTMLElement} container
316
+ * @param {Object} nodeData
317
+ */
318
+ _ensureChildNode(container, nodeData) {
319
+ let child = container.querySelector('layout-node');
320
+ if (!child) {
321
+ child = document.createElement('layout-node');
322
+ container.appendChild(child);
323
+
324
+ if (typeof setTimeout !== 'undefined') {
325
+ setTimeout(() => child._updatePanelInfo && child._updatePanelInfo());
326
+ }
327
+ }
328
+
329
+ child.$.panelChrome = this.$.panelChrome !== false;
330
+ child.setAttribute('panel-chrome', this.$.panelChrome ? 'default' : 'none');
331
+ child.$.nodeData = { ...nodeData };
332
+ }
333
+
334
+ _setupActionZones(panelId) {
335
+
336
+ let zones = this.querySelectorAll('action-zone');
337
+ zones.forEach((zone) => {
338
+ zone.$.panelId = panelId;
339
+ });
340
+ }
341
+
342
+ _startResize(e) {
343
+ if (this.$.panelChrome === false || this.$['^panelChrome'] === false) return;
344
+ e.preventDefault();
345
+ this.toggleAttribute('resizing', true);
346
+
347
+
348
+ const COLLAPSE_THRESHOLD = 0.05;
349
+ const UNCOLLAPSE_THRESHOLD = 0.08;
350
+
351
+ let onMove = (moveEvent) => {
352
+ let rect = this.getBoundingClientRect();
353
+ let currentPos = this.$.direction === 'horizontal' ? moveEvent.clientX : moveEvent.clientY;
354
+ let containerSize = this.$.direction === 'horizontal' ? rect.width : rect.height;
355
+ let startOffset = this.$.direction === 'horizontal' ? rect.left : rect.top;
356
+
357
+
358
+ let rawRatio = (currentPos - startOffset) / containerSize;
359
+
360
+
361
+ let firstChild = this.ref.first?.querySelector('layout-node');
362
+ let secondChild = this.ref.second?.querySelector('layout-node');
363
+
364
+
365
+ if (rawRatio < COLLAPSE_THRESHOLD && firstChild && !firstChild.$.isCollapsed) {
366
+
367
+ firstChild._setCollapsed(true);
368
+ return;
369
+ } else if (rawRatio > UNCOLLAPSE_THRESHOLD && firstChild?.$.isCollapsed) {
370
+
371
+ firstChild._setCollapsed(false);
372
+ }
373
+
374
+
375
+ if (rawRatio > 1 - COLLAPSE_THRESHOLD && secondChild && !secondChild.$.isCollapsed) {
376
+
377
+ secondChild._setCollapsed(true);
378
+ return;
379
+ } else if (rawRatio < 1 - UNCOLLAPSE_THRESHOLD && secondChild?.$.isCollapsed) {
380
+
381
+ secondChild._setCollapsed(false);
382
+ }
383
+
384
+
385
+ if (firstChild?.$.isCollapsed || secondChild?.$.isCollapsed) {
386
+ return;
387
+ }
388
+
389
+
390
+ let newRatio = Math.max(0.1, Math.min(0.9, rawRatio));
391
+
392
+
393
+ this.$.ratio = newRatio;
394
+ this._updateStyles();
395
+
396
+
397
+ if (this.$.nodeData) {
398
+ this.$.nodeData.ratio = newRatio;
399
+ }
400
+
401
+
402
+ this._notifyChange();
403
+ };
404
+
405
+ let onUp = () => {
406
+ this.removeAttribute('resizing');
407
+ document.removeEventListener('pointermove', onMove);
408
+ document.removeEventListener('pointerup', onUp);
409
+ };
410
+
411
+ document.addEventListener('pointermove', onMove);
412
+ document.addEventListener('pointerup', onUp);
413
+ }
414
+
415
+ #syncHostAttribute(name, value) {
416
+ if (value === undefined || value === null || value === '') {
417
+ this.removeAttribute(name);
418
+ return;
419
+ }
420
+ this.toggleAttribute(name, true);
421
+ let attr = this.getAttributeNode(name);
422
+ if (attr) attr.value = String(value);
423
+ }
424
+
425
+ _notifyChange() {
426
+ this.dispatchEvent(
427
+ new CustomEvent('layout-change', {
428
+ bubbles: true,
429
+ detail: { nodeId: this.$.nodeId },
430
+ })
431
+ );
432
+ }
433
+
434
+ _toggleCollapse() {
435
+ if (this.$.panelChrome === false || this.$['^panelChrome'] === false) return;
436
+
437
+
438
+ this.dispatchEvent(
439
+ new CustomEvent('panel-collapse-toggle', {
440
+ bubbles: true,
441
+ composed: true,
442
+ detail: {
443
+ panelId: this.$.nodeId,
444
+ collapsed: !this.$.isCollapsed,
445
+ },
446
+ })
447
+ );
448
+ }
449
+
450
+ /**
451
+ * Programmatically set collapsed state (used by resize gesture)
452
+ * @param {boolean} collapsed
453
+ */
454
+ _setCollapsed(collapsed) {
455
+ if (this.$.panelChrome === false || this.$['^panelChrome'] === false) return;
456
+ if (this.$.isCollapsed === collapsed) return;
457
+
458
+
459
+ this.dispatchEvent(
460
+ new CustomEvent('panel-collapse-toggle', {
461
+ bubbles: true,
462
+ composed: true,
463
+ detail: {
464
+ panelId: this.$.nodeId,
465
+ collapsed: collapsed,
466
+ },
467
+ })
468
+ );
469
+ }
470
+
471
+ _toggleFullscreen() {
472
+ if (this.$.panelChrome === false || this.$['^panelChrome'] === false) return;
473
+
474
+ if (this.$.isCollapsed) return;
475
+
476
+ this.dispatchEvent(
477
+ new CustomEvent('panel-fullscreen', {
478
+ bubbles: true,
479
+ composed: true,
480
+ detail: { panelId: this.$.nodeId },
481
+ })
482
+ );
483
+ }
484
+
485
+ _showTypeMenu(e) {
486
+ if (this.$.panelChrome === false || this.$['^panelChrome'] === false) return;
487
+
488
+ if (this.$.isCollapsed) return;
489
+
490
+ let rect = e.target.getBoundingClientRect();
491
+ this.dispatchEvent(
492
+ new CustomEvent('panel-type-menu', {
493
+ bubbles: true,
494
+ composed: true,
495
+ detail: {
496
+ panelId: this.$.nodeId,
497
+ currentType: this.$.panelType,
498
+ x: rect.left,
499
+ y: rect.bottom + 4,
500
+ },
501
+ })
502
+ );
503
+ }
504
+ }
505
+
506
+ LayoutNode.template = template;
507
+ LayoutNode.rootStyles = styles;
508
+
509
+ LayoutNode.reg('layout-node');
@@ -0,0 +1,39 @@
1
+ import { html } from '@symbiotejs/symbiote';
2
+
3
+ export let template = html`
4
+ <div class="panel-view" ${{ '@hidden': '!isPanel' }}>
5
+ <div class="panel-header" ${{ '@hidden': '!panelChrome' }}>
6
+ <button class="header-btn type-btn" ${{ onclick: 'onTypeClick' }}>
7
+ <span class="material-symbols-outlined panel-icon" ${{ textContent: 'panelIcon' }}></span>
8
+ <span class="panel-title" ${{ textContent: 'panelTitle' }}></span>
9
+ <span class="material-symbols-outlined dropdown-arrow">arrow_drop_down</span>
10
+ </button>
11
+ <div class="header-spacer"></div>
12
+ <button
13
+ class="header-btn collapse-btn"
14
+ ${{ onclick: 'onCollapseClick', '@hidden': '!canCollapse', title: 'collapseTitle' }}
15
+ >
16
+ <span class="material-symbols-outlined" ${{ textContent: 'collapseIcon' }}></span>
17
+ </button>
18
+ <button
19
+ class="header-btn fullscreen-btn"
20
+ ${{ onclick: 'onFullscreenClick', title: 'fullscreenTitle' }}
21
+ >
22
+ <span class="material-symbols-outlined" ${{ textContent: 'fullscreenIcon' }}></span>
23
+ </button>
24
+ </div>
25
+ <div class="panel-content" ref="panelContent" ${{ '@hidden': 'isCollapsed' }}></div>
26
+
27
+ <!-- Action zones for split/join -->
28
+ <action-zone corner="tl" ${{ '@hidden': '!panelChrome' }}></action-zone>
29
+ <action-zone corner="tr" ${{ '@hidden': '!panelChrome' }}></action-zone>
30
+ <action-zone corner="bl" ${{ '@hidden': '!panelChrome' }}></action-zone>
31
+ <action-zone corner="br" ${{ '@hidden': '!panelChrome' }}></action-zone>
32
+ </div>
33
+
34
+ <div class="split-view" ${{ '@hidden': '!isSplit', '@direction': 'direction' }}>
35
+ <div class="split-first" ref="first" ${{ '@style': 'firstStyle' }}></div>
36
+ <div class="split-resizer" ${{ onpointerdown: 'onResizerDown' }}></div>
37
+ <div class="split-second" ref="second" ${{ '@style': 'secondStyle' }}></div>
38
+ </div>
39
+ `;
@@ -0,0 +1,46 @@
1
+ import { css } from '@symbiotejs/symbiote';
2
+
3
+ export let styles = css`
4
+ layout-preview {
5
+ position: fixed;
6
+ top: 0;
7
+ left: 0;
8
+ width: 100%;
9
+ height: 100%;
10
+ pointer-events: none;
11
+ z-index: 9999;
12
+
13
+ &[hidden] {
14
+ display: none;
15
+ }
16
+
17
+ .preview-overlay {
18
+ position: absolute;
19
+ background: var(--sn-layout-preview-join-bg);
20
+ border: 2px solid var(--sn-layout-preview-join-border);
21
+ display: none;
22
+ }
23
+
24
+ &[type='join'] .preview-overlay {
25
+ display: block;
26
+ }
27
+
28
+ .preview-line {
29
+ position: absolute;
30
+ background: var(--sn-layout-preview-line);
31
+ box-shadow: var(--sn-layout-preview-line-shadow);
32
+ display: none;
33
+ }
34
+
35
+ &[type='split-h'] .preview-line,
36
+ &[type='split-v'] .preview-line {
37
+ display: block;
38
+ }
39
+
40
+ /* Hidden attribute overrides */
41
+ .preview-overlay[hidden],
42
+ .preview-line[hidden] {
43
+ display: none !important;
44
+ }
45
+ }
46
+ `;
@@ -0,0 +1,102 @@
1
+ /**
2
+ * @fileoverview LayoutPreview - Visual preview for split/join operations
3
+ * Shows where the new panel will be created or which panel will be removed.
4
+ */
5
+
6
+ import Symbiote from '@symbiotejs/symbiote';
7
+ import { template } from './LayoutPreview.tpl.js';
8
+ import { styles } from './LayoutPreview.css.js';
9
+
10
+ export class LayoutPreview extends Symbiote {
11
+ static isoMode = true;
12
+
13
+ init$ = {
14
+
15
+ previewType: null,
16
+
17
+
18
+ targetRect: null,
19
+
20
+
21
+ visible: false,
22
+
23
+
24
+ overlayStyle: '',
25
+ lineStyle: '',
26
+ };
27
+
28
+ renderCallback() {
29
+ this.sub('previewType', (type) => {
30
+ if (type) {
31
+ this.setAttribute('type', type);
32
+ } else {
33
+ this.removeAttribute('type');
34
+ }
35
+ });
36
+ }
37
+
38
+ /**
39
+ * Show split preview
40
+ * @param {'split-h' | 'split-v'} direction
41
+ * @param {DOMRect} panelRect
42
+ * @param {number} [ratio=0.5]
43
+ */
44
+ showSplit(direction, panelRect, ratio = 0.5) {
45
+ this.$.previewType = direction;
46
+ this.$.visible = true;
47
+
48
+ if (direction === 'split-h') {
49
+
50
+ let x = panelRect.left + panelRect.width * ratio;
51
+ this.$.lineStyle = `
52
+ left: ${x}px;
53
+ top: ${panelRect.top}px;
54
+ width: 4px;
55
+ height: ${panelRect.height}px;
56
+ `;
57
+ this.$.overlayStyle = '';
58
+ } else {
59
+
60
+ let y = panelRect.top + panelRect.height * ratio;
61
+ this.$.lineStyle = `
62
+ left: ${panelRect.left}px;
63
+ top: ${y}px;
64
+ width: ${panelRect.width}px;
65
+ height: 4px;
66
+ `;
67
+ this.$.overlayStyle = '';
68
+ }
69
+ }
70
+
71
+ /**
72
+ * Show join preview (overlay on target panel)
73
+ * @param {DOMRect} targetRect - Panel that will be removed
74
+ */
75
+ showJoin(targetRect) {
76
+ this.$.previewType = 'join';
77
+ this.$.visible = true;
78
+
79
+ this.$.overlayStyle = `
80
+ left: ${targetRect.left}px;
81
+ top: ${targetRect.top}px;
82
+ width: ${targetRect.width}px;
83
+ height: ${targetRect.height}px;
84
+ `;
85
+ this.$.lineStyle = '';
86
+ }
87
+
88
+ /**
89
+ * Hide preview
90
+ */
91
+ hide() {
92
+ this.$.visible = false;
93
+ this.$.previewType = null;
94
+ this.$.overlayStyle = '';
95
+ this.$.lineStyle = '';
96
+ }
97
+ }
98
+
99
+ LayoutPreview.template = template;
100
+ LayoutPreview.rootStyles = styles;
101
+
102
+ LayoutPreview.reg('layout-preview');
@@ -0,0 +1,6 @@
1
+ import { html } from '@symbiotejs/symbiote';
2
+
3
+ export let template = html`
4
+ <div class="preview-overlay" ${{ '@style': 'overlayStyle', '@hidden': '!visible' }}></div>
5
+ <div class="preview-line" ${{ '@style': 'lineStyle', '@hidden': '!visible' }}></div>
6
+ `;