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,637 @@
1
+ /**
2
+ * @fileoverview Layout - Root container for Blender-style panel layout
3
+ * Uses LayoutNode for recursive BSP tree rendering.
4
+ * Handles action zone events for split/join operations.
5
+ */
6
+
7
+ import Symbiote from '@symbiotejs/symbiote';
8
+ import { ensureMaterialSymbols } from '../../icons/MaterialSymbols.js';
9
+ import * as LayoutTree from './../LayoutTree.js';
10
+ import { template } from './Layout.tpl.js';
11
+ import { styles } from './Layout.css.js';
12
+ import './../LayoutNode/LayoutNode.js';
13
+ import './../LayoutPreview/LayoutPreview.js';
14
+ import './../PanelMenu/PanelMenu.js';
15
+
16
+ export class Layout extends Symbiote {
17
+ static isoMode = true;
18
+
19
+ init$ = {
20
+
21
+ '@storage-key': '',
22
+ '@min-panel-size': 50,
23
+
24
+
25
+ layoutTree: null,
26
+
27
+
28
+ panelTypes: {},
29
+
30
+ panelChrome: true,
31
+
32
+ activeGesture: null,
33
+
34
+
35
+ fullscreenPanelId: null,
36
+
37
+
38
+ hasFullscreenTabs: false,
39
+ tabItems: [],
40
+
41
+
42
+ onTabClick: (e) => {
43
+ let panelId = e.target.closest('[data-panel-id]')?.dataset.panelId;
44
+ if (panelId && panelId !== this.$.fullscreenPanelId) {
45
+ this._switchFullscreenPanel(panelId);
46
+ }
47
+ },
48
+
49
+
50
+ onLayoutChange: () => this._saveLayout(),
51
+ };
52
+
53
+ /**
54
+ * Register panel type
55
+ * @param {string} name - Panel type name
56
+ * @param {Object} config - Panel configuration
57
+ * @param {string} [config.title] - Default title
58
+ * @param {string} [config.icon] - Material Symbols icon name
59
+ * @param {string} [config.component] - Custom element tag name
60
+ */
61
+ registerPanelType(name, config) {
62
+ ensureMaterialSymbols([config.icon || 'dashboard']);
63
+ this.$.panelTypes = {
64
+ ...this.$.panelTypes,
65
+ [name]: config,
66
+ };
67
+ }
68
+
69
+ initCallback() {
70
+ ensureMaterialSymbols(['dashboard']);
71
+ this._loadLayout();
72
+
73
+
74
+ this.addEventListener('layout-change', () => this._saveLayout());
75
+
76
+
77
+ this.addEventListener('action-zone-start', (e) => this._onActionZoneStart(e));
78
+ this.addEventListener('action-zone-gesture', (e) => this._onActionZoneGesture(e));
79
+ this.addEventListener('action-zone-execute', (e) => this._onActionZoneExecute(e));
80
+ this.addEventListener('action-zone-end', (e) => this._onActionZoneEnd(e));
81
+
82
+
83
+ this.addEventListener('panel-type-menu', (e) => this._onPanelTypeMenu(e));
84
+ this.addEventListener('panel-type-select', (e) => this._onPanelTypeSelect(e));
85
+ this.addEventListener('panel-fullscreen', (e) => this._onPanelFullscreen(e));
86
+ this.addEventListener('panel-collapse-toggle', (e) => this._onPanelCollapseToggle(e));
87
+
88
+
89
+ this._globalPointerFallback = () => {
90
+ if (this.$.activeGesture) {
91
+ this.$.activeGesture = null;
92
+ if (this.ref.preview) {
93
+ this.ref.preview.hide();
94
+ }
95
+ }
96
+ };
97
+ if (typeof document !== 'undefined') {
98
+ document.addEventListener('pointerup', this._globalPointerFallback);
99
+ document.addEventListener('pointercancel', this._globalPointerFallback);
100
+ }
101
+ }
102
+
103
+ disconnectedCallback() {
104
+ if (this._globalPointerFallback && typeof document !== 'undefined') {
105
+ document.removeEventListener('pointerup', this._globalPointerFallback);
106
+ document.removeEventListener('pointercancel', this._globalPointerFallback);
107
+ }
108
+ }
109
+
110
+ renderCallback() {
111
+ this._renderRoot();
112
+ this.sub('layoutTree', () => {
113
+ this._renderRoot();
114
+
115
+ if (this.$.fullscreenPanelId) {
116
+
117
+ if (typeof requestAnimationFrame !== 'undefined') {
118
+ requestAnimationFrame(() => {
119
+ let allPanels = this.querySelectorAll('layout-node[node-type="panel"]');
120
+
121
+ let panelExists = Array.from(allPanels).some(
122
+ (p) => p.$.nodeId === this.$.fullscreenPanelId
123
+ );
124
+ if (panelExists) {
125
+ this._updateTabItems(allPanels, this.$.fullscreenPanelId);
126
+ } else {
127
+
128
+ this.$.fullscreenPanelId = null;
129
+ this.$.hasFullscreenTabs = false;
130
+ this.$.tabItems = [];
131
+ allPanels.forEach((p) => {
132
+ p.removeAttribute('fullscreen');
133
+ p.$.isFullscreen = false;
134
+ this.#setPanelVisible(p, true);
135
+ });
136
+ }
137
+ });
138
+ }
139
+ }
140
+ });
141
+ }
142
+
143
+ _loadLayout() {
144
+ let storageKey = this.$['@storage-key'];
145
+
146
+
147
+ if (storageKey && typeof localStorage !== 'undefined') {
148
+ let stored = localStorage.getItem(storageKey);
149
+ if (stored) {
150
+ try {
151
+ this.$.layoutTree = LayoutTree.deserialize(stored);
152
+ return;
153
+ } catch (e) {
154
+ void e;
155
+ }
156
+ }
157
+ }
158
+
159
+
160
+ let layoutAttr = this.getAttribute('layout');
161
+ if (layoutAttr) {
162
+ try {
163
+ this.$.layoutTree = JSON.parse(layoutAttr);
164
+ return;
165
+ } catch (e) {
166
+ void e;
167
+ }
168
+ }
169
+
170
+
171
+ this.$.layoutTree = LayoutTree.createPanel('default');
172
+ }
173
+
174
+ _saveLayout() {
175
+ let storageKey = this.$['@storage-key'];
176
+ if (storageKey && this.$.layoutTree && typeof localStorage !== 'undefined') {
177
+ localStorage.setItem(storageKey, LayoutTree.serialize(this.$.layoutTree));
178
+ }
179
+ }
180
+
181
+ _renderRoot() {
182
+ if (!this.$.layoutTree || !this.ref.root) return;
183
+
184
+
185
+ let rootNode = this.ref.root.querySelector('layout-node');
186
+ if (!rootNode) {
187
+ rootNode = document.createElement('layout-node');
188
+ this.ref.root.appendChild(rootNode);
189
+ }
190
+
191
+
192
+ let chromeEnabled = this.$.panelChrome !== false;
193
+ rootNode.$.panelChrome = chromeEnabled;
194
+ rootNode.setAttribute('panel-chrome', chromeEnabled ? 'default' : 'none');
195
+ rootNode.$.nodeData = this.$.layoutTree;
196
+ }
197
+
198
+
199
+ /**
200
+ * Called when action zone drag starts
201
+ * @param {CustomEvent} e
202
+ */
203
+ _onActionZoneStart(e) {
204
+ if (this.$.panelChrome === false) return;
205
+ let { panelId, corner } = e.detail;
206
+ this.$.activeGesture = { panelId, corner };
207
+ }
208
+
209
+ /**
210
+ * Called during action zone drag with gesture type
211
+ * @param {CustomEvent} e
212
+ */
213
+ _onActionZoneGesture(e) {
214
+ if (this.$.panelChrome === false) return;
215
+ let { panelId, gesture, dx, dy } = e.detail;
216
+
217
+
218
+ let panelNode = this._findPanelNode(panelId);
219
+ if (!panelNode) return;
220
+
221
+ let panelRect = panelNode.getBoundingClientRect();
222
+
223
+
224
+ let preview = this.ref.preview;
225
+ if (!preview) return;
226
+
227
+ if (gesture === 'split-h' || gesture === 'split-v') {
228
+ preview.showSplit(gesture, panelRect, 0.5);
229
+ } else if (gesture === 'join') {
230
+
231
+ let neighborInfo = this._findJoinTarget(panelId, dx, dy);
232
+ if (neighborInfo) {
233
+ let neighborNode = this._findPanelNode(neighborInfo.id);
234
+ if (neighborNode) {
235
+ preview.showJoin(neighborNode.getBoundingClientRect());
236
+ }
237
+ }
238
+ }
239
+ }
240
+
241
+ /**
242
+ * Called when action zone gesture is completed
243
+ * @param {CustomEvent} e
244
+ */
245
+ _onActionZoneExecute(e) {
246
+ if (this.$.panelChrome === false) return;
247
+ let { panelId, corner: _corner, gesture } = e.detail;
248
+
249
+ if (gesture === 'split-h') {
250
+ this.splitPanel(panelId, 'horizontal', 0.5);
251
+ } else if (gesture === 'split-v') {
252
+ this.splitPanel(panelId, 'vertical', 0.5);
253
+ } else if (gesture === 'join') {
254
+
255
+ this.joinPanels(panelId);
256
+ }
257
+ }
258
+
259
+ /**
260
+ * Called when action zone drag ends
261
+ * @param {CustomEvent} _e
262
+ */
263
+ _onActionZoneEnd(_e) {
264
+ this.$.activeGesture = null;
265
+
266
+
267
+ let preview = this.ref.preview;
268
+ if (preview) {
269
+ preview.hide();
270
+ }
271
+ }
272
+
273
+ /**
274
+ * Show panel type selection menu
275
+ * @param {CustomEvent} e
276
+ */
277
+ _onPanelTypeMenu(e) {
278
+ if (this.$.panelChrome === false) return;
279
+ let { panelId, currentType, x, y } = e.detail;
280
+ let menu = this.ref.menu;
281
+ if (!menu) return;
282
+
283
+
284
+ let items = Object.entries(this.$.panelTypes).map(([type, config]) => ({
285
+ type,
286
+ title: config.title || type,
287
+ icon: config.icon || 'dashboard',
288
+ }));
289
+
290
+ menu.show(x, y, panelId, currentType, items);
291
+ }
292
+
293
+ /**
294
+ * Handle panel type change
295
+ * @param {CustomEvent} e
296
+ */
297
+ _onPanelTypeSelect(e) {
298
+ if (this.$.panelChrome === false) return;
299
+ let { panelId, type } = e.detail;
300
+
301
+
302
+ let tree = this.$.layoutTree;
303
+ if (!tree) return;
304
+
305
+ let updateNode = (node) => {
306
+ if (!node) return;
307
+ if (node.id === panelId) {
308
+ node.panelType = type;
309
+ return;
310
+ }
311
+ if (node.first) updateNode(node.first);
312
+ if (node.second) updateNode(node.second);
313
+ };
314
+
315
+ updateNode(tree);
316
+ this.$.layoutTree = { ...tree };
317
+ this._saveLayout();
318
+ }
319
+
320
+ /**
321
+ * Toggle panel collapse state
322
+ * @param {CustomEvent} e
323
+ */
324
+ _onPanelCollapseToggle(e) {
325
+ if (this.$.panelChrome === false) return;
326
+ let { panelId, collapsed } = e.detail;
327
+ let tree = this.$.layoutTree;
328
+ if (!tree) return;
329
+
330
+
331
+ LayoutTree.updateNode(tree, panelId, { collapsed });
332
+
333
+
334
+ this.$.layoutTree = { ...tree };
335
+ this._renderRoot();
336
+ this._saveLayout();
337
+
338
+
339
+ if (typeof requestAnimationFrame !== 'undefined') {
340
+ requestAnimationFrame(() => {
341
+ let panelNode = this._findPanelNode(panelId);
342
+ if (panelNode) {
343
+
344
+ let container = panelNode.parentElement;
345
+ if (
346
+ container?.classList.contains('split-first') ||
347
+ container?.classList.contains('split-second')
348
+ ) {
349
+ let siblingContainer = container.classList.contains('split-first')
350
+ ? container.parentElement?.querySelector('.split-second')
351
+ : container.parentElement?.querySelector('.split-first');
352
+
353
+
354
+ if (siblingContainer) {
355
+ let siblingPanel = siblingContainer.querySelector('layout-node[node-type="panel"]');
356
+ if (siblingPanel?._updatePanelInfo) {
357
+ siblingPanel._updatePanelInfo();
358
+ }
359
+ }
360
+
361
+
362
+ if (panelNode._updatePanelInfo) {
363
+ panelNode._updatePanelInfo();
364
+ }
365
+ }
366
+ }
367
+ });
368
+ }
369
+ }
370
+
371
+ /**
372
+ * Toggle panel fullscreen
373
+ * @param {CustomEvent} e
374
+ */
375
+ _onPanelFullscreen(e) {
376
+ if (this.$.panelChrome === false) return;
377
+ let { panelId } = e.detail;
378
+ let panelNode = this._findPanelNode(panelId);
379
+ if (!panelNode) return;
380
+
381
+ let allPanels = this.querySelectorAll('layout-node[node-type="panel"]');
382
+
383
+ if (this.$.fullscreenPanelId === panelId) {
384
+
385
+ this.$.fullscreenPanelId = null;
386
+ this.$.hasFullscreenTabs = false;
387
+ this.$.tabItems = [];
388
+
389
+ panelNode.removeAttribute('fullscreen');
390
+ panelNode.$.isFullscreen = false;
391
+ panelNode.$.fullscreenIcon = 'fullscreen';
392
+
393
+
394
+ allPanels.forEach((p) => {
395
+ p.removeAttribute('fullscreen');
396
+ p.$.isFullscreen = false;
397
+ p.$.fullscreenIcon = 'fullscreen';
398
+ this.#setPanelVisible(p, true);
399
+ });
400
+
401
+
402
+ this._renderRoot();
403
+ this.dispatchEvent(new CustomEvent('layout-change', { bubbles: true }));
404
+ } else {
405
+
406
+ this.$.fullscreenPanelId = panelId;
407
+
408
+
409
+ allPanels.forEach((p) => {
410
+ if (p === panelNode) {
411
+ p.toggleAttribute('fullscreen', true);
412
+ p.$.isFullscreen = true;
413
+ p.$.fullscreenIcon = 'fullscreen_exit';
414
+ this.#setPanelVisible(p, true);
415
+ } else {
416
+ this.#setPanelVisible(p, false);
417
+ }
418
+ });
419
+
420
+
421
+ this._updateTabItems(allPanels, panelId);
422
+ this.$.hasFullscreenTabs = true;
423
+ }
424
+ }
425
+
426
+ /**
427
+ * Update tabItems array for Itemize-based tab bar
428
+ * @param {NodeListOf<Element>} [allPanels] - Optional, will query DOM if not provided
429
+ * @param {string} [activePanelId] - Optional, defaults to fullscreenPanelId
430
+ * @returns {void}
431
+ */
432
+ _updateTabItems(allPanels, activePanelId) {
433
+ let panels = allPanels || this.querySelectorAll('layout-node[node-type="panel"]');
434
+ let activeId = activePanelId || this.$.fullscreenPanelId;
435
+
436
+ this.$.tabItems = Array.from(panels).map((p) => {
437
+ let nodeData = p.$.nodeData;
438
+ let panelType = nodeData?.panelType || 'panel';
439
+ let typeConfig = this.$.panelTypes[panelType] || {};
440
+
441
+ return {
442
+ panelId: p.$.nodeId,
443
+ icon: typeConfig.icon || 'dashboard',
444
+ title: typeConfig.title || panelType,
445
+ isActive: p.$.nodeId === activeId,
446
+ };
447
+ });
448
+ }
449
+
450
+ /**
451
+ * Switch fullscreen to another panel
452
+ * @param {string} panelId - Panel ID to switch to
453
+ */
454
+ _switchFullscreenPanel(panelId) {
455
+ let allPanels = this.querySelectorAll('layout-node[node-type="panel"]');
456
+ let newPanel = this._findPanelNode(panelId);
457
+ if (!newPanel) return;
458
+
459
+
460
+ allPanels.forEach((p) => {
461
+ if (p.$.nodeId === panelId) {
462
+ p.toggleAttribute('fullscreen', true);
463
+ p.$.isFullscreen = true;
464
+ p.$.fullscreenIcon = 'fullscreen_exit';
465
+ this.#setPanelVisible(p, true);
466
+ } else {
467
+ p.removeAttribute('fullscreen');
468
+ p.$.isFullscreen = false;
469
+ p.$.fullscreenIcon = 'fullscreen';
470
+ this.#setPanelVisible(p, false);
471
+ }
472
+ });
473
+
474
+ this.$.fullscreenPanelId = panelId;
475
+
476
+
477
+ this._updateTabItems(allPanels, panelId);
478
+ }
479
+
480
+ /**
481
+ * Find a panel node by ID
482
+ * @param {string} panelId
483
+ * @returns {HTMLElement|null}
484
+ */
485
+ _findPanelNode(panelId) {
486
+ let nodes = this.querySelectorAll('layout-node[node-type="panel"]');
487
+ for (const node of nodes) {
488
+ if (node.$.nodeId === panelId) {
489
+ return node;
490
+ }
491
+ }
492
+ return null;
493
+ }
494
+
495
+ /**
496
+ * Find the neighbor panel for join operation
497
+ * @param {string} panelId
498
+ * @param {number} _dx
499
+ * @param {number} _dy
500
+ * @returns {{id: string, direction: string}|null}
501
+ */
502
+ _findJoinTarget(panelId, _dx, _dy) {
503
+
504
+ let parentInfo = LayoutTree.findParent(this.$.layoutTree, panelId);
505
+ if (!parentInfo) return null;
506
+
507
+ let { parent, which } = parentInfo;
508
+
509
+
510
+ let sibling = which === 'first' ? parent.second : parent.first;
511
+ if (!sibling) return null;
512
+
513
+
514
+ let siblingId = this._getFirstPanelId(sibling);
515
+
516
+ return { id: siblingId, direction: parent.direction };
517
+ }
518
+
519
+ /**
520
+ * Get the first panel ID from a node (handles nested splits)
521
+ * @param {Object} node
522
+ * @returns {string}
523
+ */
524
+ _getFirstPanelId(node) {
525
+ if (node.type === 'panel') return node.id;
526
+
527
+ return this._getFirstPanelId(node.first);
528
+ }
529
+
530
+
531
+ /**
532
+ * Split a panel
533
+ * @param {string} panelId - Panel ID to split
534
+ * @param {'horizontal' | 'vertical'} direction - Split direction
535
+ * @param {number} [ratio=0.5] - Split ratio
536
+ * @param {string} [newPanelType] - Type for new panel
537
+ */
538
+ splitPanel(panelId, direction, ratio = 0.5, newPanelType) {
539
+ let newTree = LayoutTree.splitPanel(
540
+ LayoutTree.clone(this.$.layoutTree),
541
+ panelId,
542
+ direction,
543
+ ratio,
544
+ newPanelType
545
+ );
546
+
547
+ if (newTree) {
548
+ this.$.layoutTree = newTree;
549
+ this._saveLayout();
550
+ }
551
+ }
552
+
553
+ /**
554
+ * Join panels (remove one)
555
+ * @param {string} panelToRemove - Panel ID to remove
556
+ */
557
+ joinPanels(panelToRemove) {
558
+ let newTree = LayoutTree.joinPanels(LayoutTree.clone(this.$.layoutTree), panelToRemove);
559
+
560
+ if (newTree) {
561
+ this.$.layoutTree = newTree;
562
+ this._saveLayout();
563
+ }
564
+ }
565
+
566
+ /**
567
+ * Get current layout
568
+ * @returns {import('./../LayoutTree.js').LayoutNode}
569
+ */
570
+ getLayout() {
571
+ return LayoutTree.clone(this.$.layoutTree);
572
+ }
573
+
574
+ /**
575
+ * Set layout
576
+ * @param {import('./../LayoutTree.js').LayoutNode} layout
577
+ */
578
+ setLayout(layout) {
579
+
580
+ if (this.$.fullscreenPanelId) {
581
+ let panelNode = this._findPanelNode(this.$.fullscreenPanelId);
582
+ if (panelNode) {
583
+ panelNode.removeAttribute('fullscreen');
584
+ panelNode.$.isFullscreen = false;
585
+ panelNode.$.fullscreenIcon = 'fullscreen';
586
+ this.#clearInlineProperties(panelNode, ['left', 'width']);
587
+ }
588
+ this.$.fullscreenPanelId = null;
589
+ this.$.hasFullscreenTabs = false;
590
+ this.$.tabItems = [];
591
+ }
592
+
593
+
594
+ this.querySelectorAll('layout-node[stripe]').forEach((node) => {
595
+ node.removeAttribute('stripe');
596
+ this.#clearInlineProperties(node, ['left', 'top', 'width', 'height']);
597
+ });
598
+
599
+
600
+ this.querySelectorAll('layout-node[collapsed]').forEach((node) => {
601
+ node.removeAttribute('collapsed');
602
+ node.removeAttribute('collapse-dir');
603
+ node.$.isCollapsed = false;
604
+
605
+ if (node.$.collapseDirection === 'horizontal') {
606
+ node.$.collapseIcon = 'chevron_left';
607
+ } else {
608
+ node.$.collapseIcon = 'expand_less';
609
+ }
610
+ });
611
+
612
+
613
+ this.querySelectorAll('[collapsed-child]').forEach((el) => {
614
+ el.removeAttribute('collapsed-child');
615
+ el.removeAttribute('saved-ratio');
616
+ this.#clearInlineProperties(el, ['width', 'height', 'flex']);
617
+ });
618
+
619
+ this.$.layoutTree = layout;
620
+ this._saveLayout();
621
+ }
622
+
623
+ #setPanelVisible(panel, visible) {
624
+ let style = panel.style;
625
+ style.display = visible ? '' : 'none';
626
+ }
627
+
628
+ #clearInlineProperties(el, properties) {
629
+ let style = el.style;
630
+ properties.forEach((property) => style.removeProperty(property));
631
+ }
632
+ }
633
+
634
+ Layout.template = template;
635
+ Layout.rootStyles = styles;
636
+
637
+ Layout.reg('panel-layout');
@@ -0,0 +1,27 @@
1
+ import { html } from '@symbiotejs/symbiote';
2
+
3
+ export let template = html`
4
+ <div class="layout-root" ref="root"></div>
5
+ <layout-preview ref="preview"></layout-preview>
6
+ <panel-menu ref="menu"></panel-menu>
7
+
8
+ <!-- Fullscreen tab bar (hidden by default) -->
9
+ <div class="fullscreen-tab-bar" ${{ '@hidden': '!hasFullscreenTabs' }}>
10
+ <div class="tab-list" itemize="tabItems">
11
+ <template>
12
+ <button
13
+ class="fullscreen-tab"
14
+ ${{
15
+ onclick: '^onTabClick',
16
+ '@data-panel-id': 'panelId',
17
+ '@active': 'isActive',
18
+ }}
19
+ >
20
+ <span class="material-symbols-outlined">{{icon}}</span>
21
+ <span>{{title}}</span>
22
+ </button>
23
+ </template>
24
+ </div>
25
+ <div class="tab-filler"></div>
26
+ </div>
27
+ `;