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,173 @@
1
+ /**
2
+ * GraphNode — visual node card component
3
+ *
4
+ * Renders node header, input/output ports with sockets,
5
+ * and embedded controls. Receives data via _nodeData property.
6
+ *
7
+ * @module symbiote-node/components/GraphNode
8
+ */
9
+
10
+ import Symbiote from '@symbiotejs/symbiote';
11
+ import { ensureMaterialSymbols } from '../../icons/MaterialSymbols.js';
12
+ import { template } from './GraphNode.tpl.js';
13
+ import { styles } from './GraphNode.css.js';
14
+ import '../PortItem/PortItem.js';
15
+ import '../CtrlItem/CtrlItem.js';
16
+
17
+ /** @type {Object<string, string>} */
18
+ const CATEGORY_ICONS = {
19
+ server: 'dns',
20
+ instance: 'memory',
21
+ control: 'tune',
22
+ data: 'database',
23
+
24
+ directory: 'folder',
25
+ file: 'description',
26
+ function: 'functions',
27
+ class: 'class',
28
+ module: 'package_2',
29
+ default: 'radio_button_checked',
30
+ };
31
+
32
+ function setOptionalAttribute(el, name, value) {
33
+ if (value) {
34
+ el.setAttribute(name, String(value));
35
+ } else {
36
+ el.removeAttribute(name);
37
+ }
38
+ }
39
+
40
+ function normalizeNodeTone(value) {
41
+ let tone = String(value || '').trim().toLowerCase();
42
+ if (tone === 'inverted') return 'inverse';
43
+ return tone;
44
+ }
45
+
46
+ export class GraphNode extends Symbiote {
47
+ destructionDelay = 200;
48
+
49
+ init$ = {
50
+ '@node-label': '',
51
+ '@node-category': 'default',
52
+ nodeLabel: '',
53
+ nodeIcon: 'radio_button_checked',
54
+ mediaSrc: '',
55
+ mediaAlt: '',
56
+ summary: '',
57
+ href: '',
58
+ linkLabel: 'Open',
59
+ linkIcon: 'arrow_forward',
60
+ hasItems: false,
61
+ itemsList: [],
62
+ inputPorts: [],
63
+ outputPorts: [],
64
+ controlsList: [],
65
+ };
66
+
67
+ renderCallback() {
68
+ ensureMaterialSymbols(Object.values(CATEGORY_ICONS));
69
+
70
+ this.sub('@node-label', (val) => {
71
+ this.$.nodeLabel = val || '';
72
+ });
73
+ this.sub('@node-category', (val) => {
74
+ this.$.nodeIcon = CATEGORY_ICONS[val] || CATEGORY_ICONS.default;
75
+ ensureMaterialSymbols([this.$.nodeIcon]);
76
+ });
77
+
78
+
79
+ if (this._nodeData) {
80
+ this.#populateFromNodeData(this._nodeData);
81
+ }
82
+ }
83
+
84
+ #syncMedia() {
85
+ let src = this.$.mediaSrc;
86
+ if (src && this.ref.mediaImage) {
87
+ this.ref.mediaImage.src = src;
88
+ this.ref.mediaImage.alt = this.$.mediaAlt || '';
89
+ this.ref.mediaImage.draggable = false;
90
+ }
91
+ this.toggleAttribute('data-has-media', Boolean(src));
92
+ }
93
+
94
+ #syncLink() {
95
+ let href = this.$.href;
96
+ if (!this.ref.contentLink) return;
97
+ if (!href) {
98
+ this.ref.contentLink.removeAttribute('href');
99
+ return;
100
+ }
101
+ this.ref.contentLink.href = href;
102
+ if (href.startsWith('http://') || href.startsWith('https://')) {
103
+ this.ref.contentLink.target = '_blank';
104
+ this.ref.contentLink.rel = 'noopener noreferrer';
105
+ this.$.linkIcon = 'open_in_new';
106
+ } else {
107
+ this.ref.contentLink.removeAttribute('target');
108
+ this.ref.contentLink.removeAttribute('rel');
109
+ this.$.linkIcon = 'arrow_forward';
110
+ }
111
+ ensureMaterialSymbols([this.$.linkIcon]);
112
+ }
113
+
114
+ /**
115
+ * Populate ports and controls from Node instance
116
+ * @param {import('../core/Node.js').Node} node
117
+ */
118
+ #populateFromNodeData(node) {
119
+ let params = node.params || {};
120
+ let contentHidden = Boolean(params.hideContent || params.contentHidden);
121
+ this.toggleAttribute('data-header-hidden', Boolean(params.hideHeader || params.headerHidden));
122
+ this.toggleAttribute('data-content-hidden', contentHidden);
123
+ setOptionalAttribute(this, 'data-node-tone', normalizeNodeTone(params.tone || params.nodeTone));
124
+ this.set$({
125
+ nodeIcon: node.icon || CATEGORY_ICONS[node.category] || CATEGORY_ICONS.default,
126
+ mediaSrc: params.media || params.image || params.avatar || '',
127
+ mediaAlt: params.mediaAlt || params.imageAlt || params.avatarAlt || node.label || '',
128
+ summary: contentHidden ? '' : params.summary || '',
129
+ href: params.href || '',
130
+ linkLabel: params.linkLabel || 'Open',
131
+ hasItems: !contentHidden && Array.isArray(params.items) && params.items.length > 0,
132
+ itemsList: !contentHidden && Array.isArray(params.items)
133
+ ? params.items.map((item) => ({
134
+ href: item.href || '#',
135
+ target: item.external ? '_blank' : '',
136
+ rel: item.external ? 'noopener noreferrer' : '',
137
+ kicker: item.kicker || '',
138
+ title: item.title || '',
139
+ summary: item.summary || '',
140
+ }))
141
+ : [],
142
+ inputPorts: Object.entries(node.inputs).map(([key, input]) => ({
143
+ key,
144
+ label: input.label || key,
145
+ socketColor: input.socket?.color || 'var(--sn-node-accent)',
146
+ socketName: input.socket?.name || 'any',
147
+ side: 'input',
148
+ })),
149
+ outputPorts: Object.entries(node.outputs).map(([key, output]) => ({
150
+ key,
151
+ label: output.label || key,
152
+ socketColor: output.socket?.color || 'var(--sn-node-accent)',
153
+ socketName: output.socket?.name || 'any',
154
+ side: 'output',
155
+ })),
156
+ controlsList: Object.entries(node.controls).map(([key, ctrl]) => ({
157
+ key,
158
+ label: key,
159
+ inputType: ctrl.type || 'text',
160
+ value: ctrl.value !== undefined ? String(ctrl.value) : '',
161
+ isReadonly: ctrl.readonly || false,
162
+ })),
163
+ });
164
+ ensureMaterialSymbols([this.$.nodeIcon]);
165
+ this.#syncMedia();
166
+ this.#syncLink();
167
+ }
168
+ }
169
+
170
+ GraphNode.template = template;
171
+ GraphNode.rootStyles = styles;
172
+
173
+ GraphNode.reg('graph-node');
@@ -0,0 +1,33 @@
1
+ import { html } from '@symbiotejs/symbiote';
2
+
3
+ export let template = html`
4
+ <div class="sn-node-header">
5
+ <span class="sn-node-icon material-symbols-outlined">{{nodeIcon}}</span>
6
+ <span class="sn-node-label">{{nodeLabel}}</span>
7
+ </div>
8
+ <div class="sn-node-media" ${{ '@hidden': '!mediaSrc' }}>
9
+ <img ref="mediaImage" class="sn-node-media-img" alt="{{mediaAlt}}">
10
+ </div>
11
+ <div class="sn-node-body">
12
+ <div class="inputs" ${{ itemize: 'inputPorts', 'item-tag': 'port-item' }}></div>
13
+ <div class="sn-node-content" ${{ '@hidden': '!summary' }}>
14
+ <p class="sn-node-summary">{{summary}}</p>
15
+ <a ref="contentLink" class="sn-node-link" ${{ '@hidden': '!href' }}>
16
+ <span>{{linkLabel}}</span>
17
+ <span class="material-symbols-outlined">{{linkIcon}}</span>
18
+ </a>
19
+ </div>
20
+ <div class="sn-node-items" itemize="itemsList" ${{ '@hidden': '!hasItems' }}>
21
+ <template>
22
+ <a class="sn-node-item" ${{ '@href': 'href', '@target': 'target', '@rel': 'rel' }}>
23
+ <span class="sn-node-item-kicker">{{kicker}}</span>
24
+ <span class="sn-node-item-title">{{title}}</span>
25
+ <span class="sn-node-item-summary">{{summary}}</span>
26
+ </a>
27
+ </template>
28
+ </div>
29
+ <div class="controls" ${{ itemize: 'controlsList', 'item-tag': 'ctrl-item' }}></div>
30
+ <div class="outputs" ${{ itemize: 'outputPorts', 'item-tag': 'port-item' }}></div>
31
+ </div>
32
+ <div ref="previewArea" class="sn-preview" hidden></div>
33
+ `;
@@ -0,0 +1,91 @@
1
+ import { css } from '@symbiotejs/symbiote';
2
+
3
+ export const styles = css`
4
+ node-callout {
5
+ position: fixed;
6
+ inset: 0 auto auto 0;
7
+ z-index: var(--sn-node-callout-z, var(--sn-overlay-z-base, 20000));
8
+ display: block;
9
+ inline-size: max-content;
10
+ max-inline-size: min(var(--sn-node-callout-max-width, 520px), calc(100vw - 32px));
11
+ pointer-events: none;
12
+ contain: layout style;
13
+
14
+ &[hidden] {
15
+ display: none !important;
16
+ }
17
+
18
+ &[trigger='hover']:not([data-open]) {
19
+ visibility: hidden;
20
+ }
21
+
22
+ .sn-node-callout {
23
+ position: relative;
24
+ box-sizing: border-box;
25
+ max-inline-size: inherit;
26
+ padding: var(--sn-node-callout-padding, 10px 12px);
27
+ border: 1px solid var(--sn-node-callout-border, var(--sn-node-border));
28
+ border-radius: var(--sn-node-callout-radius, var(--sn-node-radius));
29
+ background: var(--sn-node-callout-bg, var(--sn-node-bg));
30
+ box-shadow: var(--sn-node-callout-shadow, var(--sn-shadow-lg));
31
+ color: var(--sn-node-callout-color, var(--sn-text));
32
+ font-family: var(--sn-font);
33
+ font-size: var(--sn-node-callout-font-size, 12px);
34
+ font-weight: var(--sn-node-callout-font-weight, 700);
35
+ line-height: var(--sn-node-callout-line-height, 1.35);
36
+ text-align: center;
37
+ text-wrap: balance;
38
+ white-space: normal;
39
+ overflow-wrap: anywhere;
40
+ opacity: 1;
41
+ transform: translateY(0);
42
+ transition:
43
+ opacity 120ms ease,
44
+ transform 120ms ease;
45
+ -webkit-font-smoothing: antialiased;
46
+ }
47
+
48
+ &[trigger='hover']:not([data-open]) .sn-node-callout {
49
+ opacity: 0;
50
+ transform: translateY(2px);
51
+ }
52
+
53
+ .sn-node-callout::after {
54
+ content: '';
55
+ position: absolute;
56
+ inline-size: var(--sn-node-callout-arrow-size, 10px);
57
+ block-size: var(--sn-node-callout-arrow-size, 10px);
58
+ border-inline-end: 1px solid var(--sn-node-callout-border, var(--sn-node-border));
59
+ border-block-end: 1px solid var(--sn-node-callout-border, var(--sn-node-border));
60
+ background: var(--sn-node-callout-bg, var(--sn-node-bg));
61
+ transform: rotate(45deg);
62
+ }
63
+
64
+ &[placement='top'] .sn-node-callout::after {
65
+ inset-block-end: calc(var(--sn-node-callout-arrow-size, 10px) / -2 - 1px);
66
+ inset-inline-start: calc(50% - var(--sn-node-callout-arrow-size, 10px) / 2);
67
+ }
68
+
69
+ &[placement='bottom'] .sn-node-callout::after {
70
+ inset-block-start: calc(var(--sn-node-callout-arrow-size, 10px) / -2 - 1px);
71
+ inset-inline-start: calc(50% - var(--sn-node-callout-arrow-size, 10px) / 2);
72
+ transform: rotate(225deg);
73
+ }
74
+
75
+ &[placement='left'] .sn-node-callout::after {
76
+ inset-inline-end: calc(var(--sn-node-callout-arrow-size, 10px) / -2 - 1px);
77
+ inset-block-start: calc(50% - var(--sn-node-callout-arrow-size, 10px) / 2);
78
+ transform: rotate(-45deg);
79
+ }
80
+
81
+ &[placement='right'] .sn-node-callout::after {
82
+ inset-inline-start: calc(var(--sn-node-callout-arrow-size, 10px) / -2 - 1px);
83
+ inset-block-start: calc(50% - var(--sn-node-callout-arrow-size, 10px) / 2);
84
+ transform: rotate(135deg);
85
+ }
86
+
87
+ .sn-node-callout-text[hidden] {
88
+ display: none;
89
+ }
90
+ }
91
+ `;
@@ -0,0 +1,281 @@
1
+ import Symbiote from '@symbiotejs/symbiote';
2
+ import { slotProcessor } from '@symbiotejs/symbiote/core/slotProcessor.js';
3
+ import { bringOverlayToFront } from '../../ui/overlay-stack.js';
4
+ import { template } from './NodeCallout.tpl.js';
5
+ import { styles } from './NodeCallout.css.js';
6
+
7
+ /**
8
+ * Floating label anchored to a graph node or any DOM element.
9
+ */
10
+ export class NodeCallout extends Symbiote {
11
+ static observedAttributes = ['anchor-node-id', 'anchor-selector', 'placement', 'offset', 'text', 'trigger', 'hidden'];
12
+
13
+ #box = null;
14
+ #text = null;
15
+ #anchorElement = null;
16
+ #resizeObserver = null;
17
+ #mutationObserver = null;
18
+ #observedAnchor = null;
19
+ #hoverActive = false;
20
+ #raf = 0;
21
+ #lastLayout = '';
22
+ #isMounted = false;
23
+
24
+ constructor() {
25
+ super();
26
+ this.templateProcessors.add(slotProcessor);
27
+ }
28
+
29
+ connectedCallback() {
30
+ super.connectedCallback?.();
31
+ if (this.#isMounted) return;
32
+ this.#isMounted = true;
33
+
34
+ if (!this.hasAttribute('placement')) this.setAttribute('placement', 'top');
35
+ if (!this.hasAttribute('role')) this.setAttribute('role', 'tooltip');
36
+
37
+ this.#box = this.querySelector('.sn-node-callout');
38
+ this.#text = this.querySelector('.sn-node-callout-text');
39
+ this.#syncText();
40
+
41
+ if (typeof window === 'undefined' || typeof document === 'undefined') return;
42
+
43
+ if (typeof ResizeObserver !== 'undefined') {
44
+ this.#resizeObserver = new ResizeObserver(() => this.requestUpdate());
45
+ }
46
+ if (typeof MutationObserver !== 'undefined') {
47
+ this.#mutationObserver = new MutationObserver(() => this.requestUpdate());
48
+ this.#mutationObserver.observe(document.documentElement, {
49
+ attributes: true,
50
+ attributeFilter: ['style', 'class', 'hidden'],
51
+ childList: true,
52
+ subtree: true,
53
+ });
54
+ }
55
+
56
+ window.addEventListener('resize', this.#onWindowChange, { passive: true });
57
+ window.addEventListener('scroll', this.#onWindowChange, { passive: true, capture: true });
58
+ this.requestUpdate();
59
+ if (typeof requestAnimationFrame !== 'undefined') {
60
+ this.#track();
61
+ }
62
+ }
63
+
64
+ disconnectedCallback() {
65
+ this.#isMounted = false;
66
+ this.#clearAnchorObservation();
67
+ this.#resizeObserver?.disconnect();
68
+ this.#resizeObserver = null;
69
+ this.#mutationObserver?.disconnect();
70
+ this.#mutationObserver = null;
71
+
72
+ if (typeof window !== 'undefined') {
73
+ window.removeEventListener('resize', this.#onWindowChange);
74
+ window.removeEventListener('scroll', this.#onWindowChange, { capture: true });
75
+ }
76
+
77
+ if (typeof cancelAnimationFrame !== 'undefined') {
78
+ cancelAnimationFrame(this.#raf);
79
+ }
80
+ this.#raf = 0;
81
+ super.disconnectedCallback?.();
82
+ }
83
+
84
+ attributeChangedCallback(name, oldValue, newValue) {
85
+ if (oldValue === newValue) return;
86
+ if (name === 'text') this.#syncText();
87
+ if (name === 'trigger' && newValue !== 'hover') {
88
+ this.#hoverActive = false;
89
+ }
90
+ this.requestUpdate();
91
+ }
92
+
93
+ show() {
94
+ this.hidden = false;
95
+ bringOverlayToFront(this);
96
+ this.requestUpdate();
97
+ }
98
+
99
+ hide() {
100
+ this.hidden = true;
101
+ }
102
+
103
+ get anchorElement() {
104
+ return this.#anchorElement;
105
+ }
106
+
107
+ set anchorElement(value) {
108
+ this.#anchorElement = typeof Element !== 'undefined' && value instanceof Element ? value : null;
109
+ this.requestUpdate();
110
+ }
111
+
112
+ requestUpdate() {
113
+ this.#render();
114
+ }
115
+
116
+ #onWindowChange = () => this.requestUpdate();
117
+
118
+ #onAnchorEnter = () => {
119
+ if (this.#trigger() !== 'hover') return;
120
+ this.#hoverActive = true;
121
+ bringOverlayToFront(this);
122
+ this.requestUpdate();
123
+ };
124
+
125
+ #onAnchorLeave = (event) => {
126
+ if (this.#trigger() !== 'hover') return;
127
+ if (event?.relatedTarget && this.#observedAnchor?.contains(event.relatedTarget)) return;
128
+ this.#hoverActive = false;
129
+ this.requestUpdate();
130
+ };
131
+
132
+ #track() {
133
+ if (typeof requestAnimationFrame === 'undefined') return;
134
+ this.#render();
135
+ this.#raf = requestAnimationFrame(() => this.#track());
136
+ }
137
+
138
+ #syncText() {
139
+ this.#text ||= this.querySelector('.sn-node-callout-text');
140
+ if (!this.#text) return;
141
+ const text = this.getAttribute('text') || '';
142
+ this.#text.textContent = text;
143
+ this.#text.hidden = !text;
144
+ }
145
+
146
+ #render() {
147
+ if (!this.isConnected || this.hidden) return;
148
+ this.#box ||= this.querySelector('.sn-node-callout');
149
+ const anchor = this.#anchor();
150
+ if (!anchor || !this.#box) {
151
+ this.#clearAnchorObservation();
152
+ this.style.visibility = 'hidden';
153
+ this.#lastLayout = '';
154
+ return;
155
+ }
156
+
157
+ this.#observeAnchor(anchor);
158
+ const isOpen = this.#isOpen();
159
+ const wasOpen = this.hasAttribute('data-open');
160
+ this.toggleAttribute('data-open', isOpen);
161
+ this.style.visibility = isOpen ? '' : 'hidden';
162
+ if (isOpen && !wasOpen) bringOverlayToFront(this);
163
+
164
+ const anchorRect = anchor.getBoundingClientRect();
165
+ const boxRect = this.#box.getBoundingClientRect();
166
+ const placement = this.#placement();
167
+ const offset = this.#offset();
168
+ const padding = 12;
169
+ let x = 0;
170
+ let y = 0;
171
+
172
+ if (placement === 'bottom') {
173
+ x = anchorRect.left + anchorRect.width / 2 - boxRect.width / 2;
174
+ y = anchorRect.bottom + offset;
175
+ } else if (placement === 'left') {
176
+ x = anchorRect.left - boxRect.width - offset;
177
+ y = anchorRect.top + anchorRect.height / 2 - boxRect.height / 2;
178
+ } else if (placement === 'right') {
179
+ x = anchorRect.right + offset;
180
+ y = anchorRect.top + anchorRect.height / 2 - boxRect.height / 2;
181
+ } else {
182
+ x = anchorRect.left + anchorRect.width / 2 - boxRect.width / 2;
183
+ y = anchorRect.top - boxRect.height - offset;
184
+ }
185
+
186
+ x = Math.min(Math.max(padding, x), window.innerWidth - boxRect.width - padding);
187
+ y = Math.min(Math.max(padding, y), window.innerHeight - boxRect.height - padding);
188
+
189
+ const nextLayout = `${Math.round(x)}:${Math.round(y)}:${placement}`;
190
+ if (nextLayout === this.#lastLayout) return;
191
+ this.#lastLayout = nextLayout;
192
+ this.style.transform = `translate3d(${Math.round(x)}px, ${Math.round(y)}px, 0)`;
193
+ }
194
+
195
+ #observeAnchor(anchor) {
196
+ if (!this.#resizeObserver || this.#observedAnchor === anchor) return;
197
+ this.#clearAnchorObservation();
198
+ this.#resizeObserver.observe(anchor);
199
+ anchor.addEventListener('mouseenter', this.#onAnchorEnter);
200
+ anchor.addEventListener('mouseover', this.#onAnchorEnter);
201
+ anchor.addEventListener('pointerenter', this.#onAnchorEnter);
202
+ anchor.addEventListener('pointerover', this.#onAnchorEnter);
203
+ anchor.addEventListener('mouseleave', this.#onAnchorLeave);
204
+ anchor.addEventListener('mouseout', this.#onAnchorLeave);
205
+ anchor.addEventListener('pointerleave', this.#onAnchorLeave);
206
+ anchor.addEventListener('pointerout', this.#onAnchorLeave);
207
+ this.#observedAnchor = anchor;
208
+ }
209
+
210
+ #clearAnchorObservation() {
211
+ if (this.#resizeObserver && this.#observedAnchor) {
212
+ this.#resizeObserver.unobserve(this.#observedAnchor);
213
+ }
214
+ this.#observedAnchor?.removeEventListener('mouseenter', this.#onAnchorEnter);
215
+ this.#observedAnchor?.removeEventListener('mouseover', this.#onAnchorEnter);
216
+ this.#observedAnchor?.removeEventListener('pointerenter', this.#onAnchorEnter);
217
+ this.#observedAnchor?.removeEventListener('pointerover', this.#onAnchorEnter);
218
+ this.#observedAnchor?.removeEventListener('mouseleave', this.#onAnchorLeave);
219
+ this.#observedAnchor?.removeEventListener('mouseout', this.#onAnchorLeave);
220
+ this.#observedAnchor?.removeEventListener('pointerleave', this.#onAnchorLeave);
221
+ this.#observedAnchor?.removeEventListener('pointerout', this.#onAnchorLeave);
222
+ this.#observedAnchor = null;
223
+ this.#hoverActive = false;
224
+ }
225
+
226
+ #anchor() {
227
+ if (this.#anchorElement?.isConnected) {
228
+ this.toggleAttribute('data-anchor-selector-invalid', false);
229
+ return this.#anchorElement;
230
+ }
231
+
232
+ const nodeId = this.getAttribute('anchor-node-id') || '';
233
+ if (nodeId) {
234
+ for (const node of document.querySelectorAll('graph-node[node-id]')) {
235
+ if (node.getAttribute('node-id') === nodeId) {
236
+ this.toggleAttribute('data-anchor-selector-invalid', false);
237
+ return node;
238
+ }
239
+ }
240
+ this.toggleAttribute('data-anchor-selector-invalid', false);
241
+ return null;
242
+ }
243
+
244
+ const selector = this.getAttribute('anchor-selector') || '';
245
+ if (!selector) {
246
+ this.toggleAttribute('data-anchor-selector-invalid', false);
247
+ return null;
248
+ }
249
+ try {
250
+ const anchor = document.querySelector(selector);
251
+ this.toggleAttribute('data-anchor-selector-invalid', false);
252
+ return anchor;
253
+ } catch {
254
+ this.toggleAttribute('data-anchor-selector-invalid', true);
255
+ return null;
256
+ }
257
+ }
258
+
259
+ #placement() {
260
+ const placement = this.getAttribute('placement') || 'top';
261
+ return ['top', 'bottom', 'left', 'right'].includes(placement) ? placement : 'top';
262
+ }
263
+
264
+ #offset() {
265
+ const value = Number.parseFloat(this.getAttribute('offset') || '');
266
+ return Number.isFinite(value) ? value : 14;
267
+ }
268
+
269
+ #trigger() {
270
+ const trigger = this.getAttribute('trigger') || 'always';
271
+ return trigger === 'hover' ? 'hover' : 'always';
272
+ }
273
+
274
+ #isOpen() {
275
+ return this.#trigger() !== 'hover' || this.#hoverActive;
276
+ }
277
+ }
278
+
279
+ NodeCallout.template = template;
280
+ NodeCallout.rootStyles = styles;
281
+ NodeCallout.reg('node-callout');
@@ -0,0 +1,8 @@
1
+ import { html } from '@symbiotejs/symbiote';
2
+
3
+ export const template = html`
4
+ <div class="sn-node-callout" part="box">
5
+ <span class="sn-node-callout-text" part="text" hidden></span>
6
+ <slot></slot>
7
+ </div>
8
+ `;
@@ -0,0 +1,68 @@
1
+ /**
2
+ * NodeSocket styles
3
+ * @module symbiote-node/node/NodeSocket.css
4
+ */
5
+ import { css } from '@symbiotejs/symbiote';
6
+
7
+ export let styles = css`
8
+ node-socket {
9
+ display: block;
10
+ width: var(--sn-socket-size);
11
+ height: var(--sn-socket-size);
12
+ border-radius: 50%;
13
+ background: var(--socket-color, var(--sn-node-accent));
14
+ border: var(--sn-socket-border-width) solid var(--sn-node-bg);
15
+ cursor: crosshair;
16
+ flex-shrink: 0;
17
+ transition:
18
+ transform 0.2s ease-out,
19
+ box-shadow 0.2s ease-out;
20
+ z-index: 10;
21
+ position: relative;
22
+
23
+ &::before {
24
+ content: '';
25
+ position: absolute;
26
+ top: 50%;
27
+ left: 50%;
28
+ width: 44px;
29
+ height: 44px;
30
+ transform: translate(-50%, -50%);
31
+ }
32
+
33
+ &:hover {
34
+ transform: scale(1.3);
35
+ box-shadow: 0 0 8px var(--socket-color, var(--sn-node-accent));
36
+ }
37
+
38
+ &[data-socket-shape='square'] {
39
+ border-radius: 2px;
40
+ }
41
+
42
+ &[data-socket-shape='diamond'] {
43
+ border-radius: 1px;
44
+ transform: rotate(45deg) scale(0.85);
45
+
46
+ &:hover {
47
+ transform: rotate(45deg) scale(1.1);
48
+ }
49
+ }
50
+
51
+ &[data-socket-shape='triangle'] {
52
+ border-radius: 0;
53
+ background: transparent;
54
+ width: 0;
55
+ height: 0;
56
+ border: calc(var(--sn-socket-size) / 2) solid transparent;
57
+ border-left: calc(var(--sn-socket-size) - var(--sn-socket-border-width)) solid
58
+ var(--socket-color, var(--sn-node-accent));
59
+ border-right: none;
60
+ }
61
+ }
62
+
63
+ @media (prefers-reduced-motion: reduce) {
64
+ node-socket {
65
+ transition: none;
66
+ }
67
+ }
68
+ `;
@@ -0,0 +1,26 @@
1
+ /**
2
+ * NodeSocket — connection endpoint circle
3
+ *
4
+ * Visual socket indicator for node ports.
5
+ * Changes color via socket data attribute.
6
+ *
7
+ * @module symbiote-node/components/NodeSocket
8
+ */
9
+
10
+ import Symbiote from '@symbiotejs/symbiote';
11
+ import { template } from './NodeSocket.tpl.js';
12
+ import { styles } from './NodeSocket.css.js';
13
+
14
+ export class NodeSocket extends Symbiote {
15
+ renderCallback() {
16
+ this.sub('@data-socket-color', (val) => {
17
+ if (val) {
18
+ this.style.setProperty('--socket-color', val);
19
+ }
20
+ });
21
+ }
22
+ }
23
+
24
+ NodeSocket.template = template;
25
+ NodeSocket.rootStyles = styles;
26
+ NodeSocket.reg('node-socket');
@@ -0,0 +1,7 @@
1
+ /**
2
+ * NodeSocket template
3
+ * @module symbiote-node/node/NodeSocket.tpl
4
+ */
5
+ import { html } from '@symbiotejs/symbiote';
6
+
7
+ export let template = html`<slot></slot>`;