project-graph-mcp 2.3.1 → 2.4.0

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 (226) hide show
  1. package/package.json +3 -2
  2. package/src/analysis/analysis-cache.ctx +9 -0
  3. package/src/analysis/analysis-cache.js +1 -1
  4. package/src/analysis/complexity.ctx +6 -0
  5. package/src/analysis/complexity.js +1 -1
  6. package/src/analysis/custom-rules.ctx +14 -0
  7. package/src/analysis/custom-rules.js +1 -1
  8. package/src/analysis/db-analysis.ctx +7 -0
  9. package/src/analysis/db-analysis.js +1 -1
  10. package/src/analysis/dead-code.ctx +6 -0
  11. package/src/analysis/dead-code.js +1 -1
  12. package/src/analysis/full-analysis.ctx +9 -0
  13. package/src/analysis/full-analysis.js +1 -1
  14. package/src/analysis/jsdoc-checker.ctx +10 -0
  15. package/src/analysis/jsdoc-checker.js +1 -1
  16. package/src/analysis/jsdoc-generator.ctx +9 -0
  17. package/src/analysis/jsdoc-generator.js +1 -1
  18. package/src/analysis/large-files.ctx +6 -0
  19. package/src/analysis/large-files.js +1 -1
  20. package/src/analysis/outdated-patterns.ctx +7 -0
  21. package/src/analysis/outdated-patterns.js +1 -1
  22. package/src/analysis/similar-functions.ctx +6 -0
  23. package/src/analysis/similar-functions.js +1 -1
  24. package/src/analysis/test-annotations.ctx +11 -0
  25. package/src/analysis/test-annotations.js +1 -1
  26. package/src/analysis/type-checker.ctx +6 -0
  27. package/src/analysis/type-checker.js +1 -1
  28. package/src/analysis/undocumented.ctx +8 -0
  29. package/src/analysis/undocumented.js +1 -1
  30. package/src/cli/cli-handlers.ctx +7 -0
  31. package/src/cli/cli-handlers.js +1 -1
  32. package/src/cli/cli.ctx +6 -0
  33. package/src/cli/cli.js +1 -1
  34. package/src/compact/ai-context.ctx +6 -0
  35. package/src/compact/ai-context.js +1 -1
  36. package/src/compact/compact-migrate.ctx +8 -0
  37. package/src/compact/compact-migrate.js +1 -1
  38. package/src/compact/compact.ctx +11 -0
  39. package/src/compact/compact.js +1 -1
  40. package/src/compact/compress.ctx +7 -0
  41. package/src/compact/compress.js +1 -1
  42. package/src/compact/ctx-resolver.ctx +2 -0
  43. package/src/compact/ctx-resolver.js +1 -1
  44. package/src/compact/ctx-to-jsdoc.ctx +11 -0
  45. package/src/compact/ctx-to-jsdoc.js +1 -1
  46. package/src/compact/doc-dialect.ctx +11 -0
  47. package/src/compact/doc-dialect.js +2 -2
  48. package/src/compact/expand.ctx +14 -0
  49. package/src/compact/expand.js +1 -1
  50. package/src/compact/framework-references.ctx +7 -0
  51. package/src/compact/framework-references.js +1 -1
  52. package/src/compact/instructions.ctx +6 -0
  53. package/src/compact/instructions.js +1 -1
  54. package/src/compact/jsdoc-builder.ctx +4 -0
  55. package/src/compact/jsdoc-builder.js +1 -1
  56. package/src/compact/mode-config.ctx +8 -0
  57. package/src/compact/mode-config.js +1 -1
  58. package/src/compact/split-declarations.ctx +6 -0
  59. package/src/compact/split-declarations.js +1 -1
  60. package/src/compact/validate-pipeline.ctx +12 -0
  61. package/src/compact/validate-pipeline.js +1 -1
  62. package/src/core/event-bus.ctx +9 -0
  63. package/src/core/event-bus.js +1 -1
  64. package/src/core/file-walker.ctx +1 -0
  65. package/src/core/file-walker.js +1 -1
  66. package/src/core/filters.ctx +12 -0
  67. package/src/core/filters.js +1 -1
  68. package/src/core/graph-builder.ctx +7 -0
  69. package/src/core/graph-builder.js +1 -1
  70. package/src/core/parser.ctx +12 -0
  71. package/src/core/parser.js +1 -1
  72. package/src/core/utils.ctx +1 -0
  73. package/src/core/utils.js +1 -1
  74. package/src/core/workspace.ctx +7 -0
  75. package/src/core/workspace.js +1 -1
  76. package/src/lang/lang-go.ctx +8 -0
  77. package/src/lang/lang-go.js +1 -1
  78. package/src/lang/lang-python.ctx +5 -0
  79. package/src/lang/lang-python.js +1 -1
  80. package/src/lang/lang-sql.ctx +10 -0
  81. package/src/lang/lang-sql.js +1 -1
  82. package/src/lang/lang-typescript.ctx +6 -0
  83. package/src/lang/lang-typescript.js +1 -1
  84. package/src/lang/lang-utils.ctx +5 -0
  85. package/src/lang/lang-utils.js +1 -1
  86. package/src/mcp/mcp-server.ctx +6 -0
  87. package/src/mcp/mcp-server.js +1 -1
  88. package/src/mcp/tool-defs.ctx +2 -0
  89. package/src/mcp/tool-defs.js +1 -1
  90. package/src/mcp/tools.ctx +13 -0
  91. package/src/mcp/tools.js +1 -1
  92. package/src/network/backend-lifecycle.ctx +10 -0
  93. package/src/network/backend-lifecycle.js +1 -1
  94. package/src/network/backend.ctx +5 -0
  95. package/src/network/backend.js +1 -1
  96. package/src/network/local-gateway.ctx +9 -0
  97. package/src/network/local-gateway.js +1 -1
  98. package/src/network/mdns.ctx +6 -0
  99. package/src/network/mdns.js +1 -1
  100. package/src/network/server.ctx +2 -0
  101. package/src/network/server.js +2 -2
  102. package/src/network/web-server.ctx +17 -0
  103. package/src/network/web-server.js +2 -2
  104. package/web/follow-controller.js +94 -25
  105. package/web/panels/dep-graph.js +207 -21
  106. package/project-graph-mcp-2.3.0.tgz +0 -0
  107. package/vendor/symbiote-node/CHANGELOG.md +0 -31
  108. package/vendor/symbiote-node/LICENSE +0 -21
  109. package/vendor/symbiote-node/README.md +0 -206
  110. package/vendor/symbiote-node/canvas/AutoLayout.js +0 -725
  111. package/vendor/symbiote-node/canvas/Breadcrumb/Breadcrumb.css.js +0 -73
  112. package/vendor/symbiote-node/canvas/Breadcrumb/Breadcrumb.js +0 -93
  113. package/vendor/symbiote-node/canvas/Breadcrumb/Breadcrumb.tpl.js +0 -9
  114. package/vendor/symbiote-node/canvas/CanvasConnectionRenderer.js +0 -962
  115. package/vendor/symbiote-node/canvas/ConnectionRenderer.js +0 -1468
  116. package/vendor/symbiote-node/canvas/FlowSimulator.js +0 -323
  117. package/vendor/symbiote-node/canvas/ForceLayout.js +0 -189
  118. package/vendor/symbiote-node/canvas/ForceWorker.js +0 -1325
  119. package/vendor/symbiote-node/canvas/GraphTabs/GraphTabs.css.js +0 -97
  120. package/vendor/symbiote-node/canvas/GraphTabs/GraphTabs.js +0 -176
  121. package/vendor/symbiote-node/canvas/GraphTabs/GraphTabs.tpl.js +0 -12
  122. package/vendor/symbiote-node/canvas/LODManager.js +0 -88
  123. package/vendor/symbiote-node/canvas/Minimap/Minimap.css.js +0 -71
  124. package/vendor/symbiote-node/canvas/Minimap/Minimap.js +0 -207
  125. package/vendor/symbiote-node/canvas/Minimap/Minimap.tpl.js +0 -9
  126. package/vendor/symbiote-node/canvas/NodeCanvas/NodeCanvas.css.js +0 -261
  127. package/vendor/symbiote-node/canvas/NodeCanvas/NodeCanvas.js +0 -1840
  128. package/vendor/symbiote-node/canvas/NodeCanvas/NodeCanvas.tpl.js +0 -22
  129. package/vendor/symbiote-node/canvas/NodeSearch/NodeSearch.css.js +0 -97
  130. package/vendor/symbiote-node/canvas/NodeSearch/NodeSearch.js +0 -132
  131. package/vendor/symbiote-node/canvas/NodeSearch/NodeSearch.tpl.js +0 -21
  132. package/vendor/symbiote-node/canvas/NodeViewManager.js +0 -584
  133. package/vendor/symbiote-node/canvas/PinExpansion.js +0 -131
  134. package/vendor/symbiote-node/canvas/PseudoConnection.js +0 -80
  135. package/vendor/symbiote-node/canvas/SubgraphManager.js +0 -201
  136. package/vendor/symbiote-node/canvas/SubgraphRouter.js +0 -443
  137. package/vendor/symbiote-node/canvas/ViewportActions.js +0 -446
  138. package/vendor/symbiote-node/core/Connection.js +0 -45
  139. package/vendor/symbiote-node/core/Editor.js +0 -451
  140. package/vendor/symbiote-node/core/Frame.js +0 -31
  141. package/vendor/symbiote-node/core/GraphMermaid.js +0 -348
  142. package/vendor/symbiote-node/core/GraphText.js +0 -210
  143. package/vendor/symbiote-node/core/Node.js +0 -143
  144. package/vendor/symbiote-node/core/Portal.js +0 -104
  145. package/vendor/symbiote-node/core/Socket.js +0 -185
  146. package/vendor/symbiote-node/core/SubgraphNode.js +0 -125
  147. package/vendor/symbiote-node/index.js +0 -103
  148. package/vendor/symbiote-node/inspector/InspectorPanel/InspectorPanel.css.js +0 -361
  149. package/vendor/symbiote-node/inspector/InspectorPanel/InspectorPanel.js +0 -332
  150. package/vendor/symbiote-node/inspector/InspectorPanel/InspectorPanel.tpl.js +0 -96
  151. package/vendor/symbiote-node/inspector/TemplatePreview/TemplatePreview.css.js +0 -104
  152. package/vendor/symbiote-node/inspector/TemplatePreview/TemplatePreview.js +0 -133
  153. package/vendor/symbiote-node/inspector/TemplatePreview/TemplatePreview.tpl.js +0 -33
  154. package/vendor/symbiote-node/interactions/ConnectFlow.js +0 -307
  155. package/vendor/symbiote-node/interactions/Drag.js +0 -102
  156. package/vendor/symbiote-node/interactions/Selector.js +0 -132
  157. package/vendor/symbiote-node/interactions/SnapGrid.js +0 -65
  158. package/vendor/symbiote-node/interactions/Zoom.js +0 -140
  159. package/vendor/symbiote-node/layout/ActionZone/ActionZone.css.js +0 -88
  160. package/vendor/symbiote-node/layout/ActionZone/ActionZone.js +0 -254
  161. package/vendor/symbiote-node/layout/ActionZone/ActionZone.tpl.js +0 -11
  162. package/vendor/symbiote-node/layout/Layout/Layout.css.js +0 -88
  163. package/vendor/symbiote-node/layout/Layout/Layout.js +0 -622
  164. package/vendor/symbiote-node/layout/Layout/Layout.tpl.js +0 -25
  165. package/vendor/symbiote-node/layout/LayoutNode/LayoutNode.css.js +0 -293
  166. package/vendor/symbiote-node/layout/LayoutNode/LayoutNode.js +0 -467
  167. package/vendor/symbiote-node/layout/LayoutNode/LayoutNode.tpl.js +0 -33
  168. package/vendor/symbiote-node/layout/LayoutPreview/LayoutPreview.css.js +0 -46
  169. package/vendor/symbiote-node/layout/LayoutPreview/LayoutPreview.js +0 -102
  170. package/vendor/symbiote-node/layout/LayoutPreview/LayoutPreview.tpl.js +0 -6
  171. package/vendor/symbiote-node/layout/LayoutRouter/LayoutRouter.js +0 -156
  172. package/vendor/symbiote-node/layout/LayoutRouter/routerSync.js +0 -250
  173. package/vendor/symbiote-node/layout/LayoutSidebar/LayoutSidebar.css.js +0 -379
  174. package/vendor/symbiote-node/layout/LayoutSidebar/LayoutSidebar.js +0 -263
  175. package/vendor/symbiote-node/layout/LayoutSidebar/LayoutSidebar.tpl.js +0 -20
  176. package/vendor/symbiote-node/layout/LayoutSidebar/SidebarSection.js +0 -183
  177. package/vendor/symbiote-node/layout/LayoutTree.js +0 -246
  178. package/vendor/symbiote-node/layout/PanelMenu/PanelMenu.css.js +0 -43
  179. package/vendor/symbiote-node/layout/PanelMenu/PanelMenu.js +0 -89
  180. package/vendor/symbiote-node/layout/PanelMenu/PanelMenu.tpl.js +0 -14
  181. package/vendor/symbiote-node/layout/index.js +0 -16
  182. package/vendor/symbiote-node/menu/ContextMenu/ContextMenu.css.js +0 -61
  183. package/vendor/symbiote-node/menu/ContextMenu/ContextMenu.js +0 -79
  184. package/vendor/symbiote-node/menu/ContextMenu/ContextMenu.tpl.js +0 -19
  185. package/vendor/symbiote-node/node/CtrlItem/CtrlItem.css.js +0 -41
  186. package/vendor/symbiote-node/node/CtrlItem/CtrlItem.js +0 -24
  187. package/vendor/symbiote-node/node/CtrlItem/CtrlItem.tpl.js +0 -16
  188. package/vendor/symbiote-node/node/GraphFrame/GraphFrame.css.js +0 -65
  189. package/vendor/symbiote-node/node/GraphFrame/GraphFrame.js +0 -29
  190. package/vendor/symbiote-node/node/GraphFrame/GraphFrame.tpl.js +0 -13
  191. package/vendor/symbiote-node/node/GraphNode/GraphNode.css.js +0 -683
  192. package/vendor/symbiote-node/node/GraphNode/GraphNode.js +0 -92
  193. package/vendor/symbiote-node/node/GraphNode/GraphNode.tpl.js +0 -17
  194. package/vendor/symbiote-node/node/NodeSocket/NodeSocket.js +0 -25
  195. package/vendor/symbiote-node/node/NodeSocket/NodeSocket.tpl.js +0 -7
  196. package/vendor/symbiote-node/node/PortItem/PortItem.css.js +0 -90
  197. package/vendor/symbiote-node/node/PortItem/PortItem.js +0 -87
  198. package/vendor/symbiote-node/node/PortItem/PortItem.tpl.js +0 -10
  199. package/vendor/symbiote-node/package.json +0 -59
  200. package/vendor/symbiote-node/palette/PaletteBrowser/PaletteBrowser.css.js +0 -143
  201. package/vendor/symbiote-node/palette/PaletteBrowser/PaletteBrowser.js +0 -131
  202. package/vendor/symbiote-node/palette/PaletteBrowser/PaletteBrowser.tpl.js +0 -16
  203. package/vendor/symbiote-node/plugins/History.js +0 -384
  204. package/vendor/symbiote-node/plugins/Readonly.js +0 -59
  205. package/vendor/symbiote-node/shapes/CircleShape.js +0 -80
  206. package/vendor/symbiote-node/shapes/CommentShape.js +0 -35
  207. package/vendor/symbiote-node/shapes/DiamondShape.js +0 -115
  208. package/vendor/symbiote-node/shapes/NodeShape.js +0 -80
  209. package/vendor/symbiote-node/shapes/PillShape.js +0 -91
  210. package/vendor/symbiote-node/shapes/RectShape.js +0 -72
  211. package/vendor/symbiote-node/shapes/SVGShape.js +0 -494
  212. package/vendor/symbiote-node/shapes/index.js +0 -53
  213. package/vendor/symbiote-node/themes/Palette.js +0 -32
  214. package/vendor/symbiote-node/themes/Skin.js +0 -113
  215. package/vendor/symbiote-node/themes/Theme.js +0 -84
  216. package/vendor/symbiote-node/themes/carbon.js +0 -137
  217. package/vendor/symbiote-node/themes/dark.js +0 -137
  218. package/vendor/symbiote-node/themes/ebook.js +0 -138
  219. package/vendor/symbiote-node/themes/grey.js +0 -137
  220. package/vendor/symbiote-node/themes/light.js +0 -137
  221. package/vendor/symbiote-node/themes/neon.js +0 -138
  222. package/vendor/symbiote-node/themes/pcb.js +0 -273
  223. package/vendor/symbiote-node/themes/synthwave.js +0 -137
  224. package/vendor/symbiote-node/toolbar/QuickToolbar/QuickToolbar.css.js +0 -86
  225. package/vendor/symbiote-node/toolbar/QuickToolbar/QuickToolbar.js +0 -128
  226. package/vendor/symbiote-node/toolbar/QuickToolbar/QuickToolbar.tpl.js +0 -29
@@ -1,80 +0,0 @@
1
- /**
2
- * PseudoConnection — temporary connection line during drag
3
- *
4
- * Renders dashed Bézier path + plus indicator at endpoint.
5
- * Extracted from NodeCanvas to reduce complexity.
6
- *
7
- * @module symbiote-node/canvas/PseudoConnection
8
- */
9
-
10
- export class PseudoConnection {
11
-
12
- /** @type {SVGPathElement|null} */
13
- #path = null;
14
-
15
- /** @type {SVGGElement|null} */
16
- #plusIndicator = null;
17
-
18
- /** @type {SVGElement} */
19
- #svg;
20
-
21
- /**
22
- * @param {SVGElement} svgLayer - pseudo SVG overlay
23
- */
24
- constructor(svgLayer) {
25
- this.#svg = svgLayer;
26
- }
27
-
28
- /**
29
- * Show pseudo-connection between two world-space points.
30
- * pseudo-svg is inside .content which applies CSS transform (zoom+pan),
31
- * so coordinates must be in world-space (same as ConnectionRenderer dots).
32
- * @param {number} sx - Start X (world space)
33
- * @param {number} sy - Start Y (world space)
34
- * @param {number} ex - End X (world space)
35
- * @param {number} ey - End Y (world space)
36
- */
37
- show(sx, sy, ex, ey) {
38
- if (!this.#path) {
39
- this.#path = document.createElementNS('http://www.w3.org/2000/svg', 'path');
40
- this.#path.setAttribute('class', 'pseudo-path');
41
- this.#svg.appendChild(this.#path);
42
- }
43
-
44
- const dx = Math.abs(ex - sx) * 0.5;
45
- const d = `M ${sx} ${sy} C ${sx + dx} ${sy}, ${ex - dx} ${ey}, ${ex} ${ey}`;
46
- this.#path.setAttribute('d', d);
47
-
48
- // Plus indicator at endpoint
49
- if (!this.#plusIndicator) {
50
- const g = document.createElementNS('http://www.w3.org/2000/svg', 'g');
51
- g.setAttribute('class', 'plus-indicator');
52
- const circle = document.createElementNS('http://www.w3.org/2000/svg', 'circle');
53
- circle.setAttribute('r', '8');
54
- const h = document.createElementNS('http://www.w3.org/2000/svg', 'line');
55
- h.setAttribute('x1', '-4'); h.setAttribute('y1', '0');
56
- h.setAttribute('x2', '4'); h.setAttribute('y2', '0');
57
- const v = document.createElementNS('http://www.w3.org/2000/svg', 'line');
58
- v.setAttribute('x1', '0'); v.setAttribute('y1', '-4');
59
- v.setAttribute('x2', '0'); v.setAttribute('y2', '4');
60
- g.appendChild(circle);
61
- g.appendChild(h);
62
- g.appendChild(v);
63
- this.#svg.appendChild(g);
64
- this.#plusIndicator = g;
65
- }
66
- this.#plusIndicator.setAttribute('transform', `translate(${ex}, ${ey})`);
67
- }
68
-
69
- /** Hide and clean up pseudo-connection */
70
- hide() {
71
- if (this.#path) {
72
- this.#path.remove();
73
- this.#path = null;
74
- }
75
- if (this.#plusIndicator) {
76
- this.#plusIndicator.remove();
77
- this.#plusIndicator = null;
78
- }
79
- }
80
- }
@@ -1,201 +0,0 @@
1
- /**
2
- * SubgraphManager — Navigation stack for subgraph drill-down
3
- *
4
- * Manages a stack of editor contexts. When drilling into a subgraph,
5
- * the current state is saved and the inner editor is loaded.
6
- * Breadcrumb navigation allows jumping back to any level.
7
- *
8
- * @module symbiote-node/canvas/SubgraphManager
9
- */
10
-
11
- /**
12
- * @typedef {Object} NavEntry
13
- * @property {import('../core/Editor.js').NodeEditor} editor
14
- * @property {Object<string, { x: number, y: number }>} positions
15
- * @property {{ panX: number, panY: number, zoom: number }} transform
16
- * @property {string} label - Display label for breadcrumb
17
- * @property {string|null} subgraphNodeId - SubgraphNode ID in parent (null for root)
18
- */
19
-
20
- export class SubgraphManager {
21
- /** @type {NavEntry[]} */
22
- #stack = [];
23
-
24
- /** @type {import('./NodeCanvas.js').NodeCanvas|null} */
25
- #canvas = null;
26
-
27
- /** @type {function|null} */
28
- #onNavigate = null;
29
-
30
- /**
31
- * Initialize with canvas and root editor
32
- * @param {*} canvas - NodeCanvas element
33
- * @param {import('../core/Editor.js').NodeEditor} rootEditor
34
- */
35
- initialize(canvas, rootEditor) {
36
- this.#canvas = canvas;
37
- this.#stack = [{
38
- editor: rootEditor,
39
- positions: {},
40
- transform: { panX: 0, panY: 0, zoom: 1 },
41
- label: 'Root',
42
- subgraphNodeId: null,
43
- }];
44
- }
45
-
46
- /**
47
- * Set navigation callback (for breadcrumb updates)
48
- * @param {function} callback - (path: string[]) => void
49
- */
50
- onNavigate(callback) {
51
- this.#onNavigate = callback;
52
- }
53
-
54
- /**
55
- * Drill down into a SubgraphNode
56
- * @param {import('../core/SubgraphNode.js').SubgraphNode} subgraphNode
57
- */
58
- drillDown(subgraphNode) {
59
- if (!subgraphNode?._isSubgraph || !this.#canvas) return;
60
-
61
- // Save current state
62
- const current = this.#stack[this.#stack.length - 1];
63
- current.positions = this.#capturePositions(current.editor, current.positions);
64
- current.transform = this.#captureTransform();
65
-
66
- // Push inner editor
67
- this.#stack.push({
68
- editor: subgraphNode.getInnerEditor(),
69
- positions: subgraphNode.innerPositions,
70
- transform: subgraphNode.innerTransform,
71
- label: subgraphNode.label,
72
- subgraphNodeId: subgraphNode.id,
73
- });
74
-
75
- // Switch canvas to inner editor
76
- this.#applyLevel(this.#stack.length - 1);
77
- this.#notifyNavigate();
78
- }
79
-
80
- /**
81
- * Navigate up to a specific level
82
- * @param {number} level - 0 = root
83
- */
84
- drillUp(level) {
85
- if (level === undefined) level = this.#stack.length - 2;
86
- if (level < 0 || level >= this.#stack.length || !this.#canvas) return;
87
-
88
- // Save state of levels being popped (persist inner positions)
89
- for (let i = this.#stack.length - 1; i > level; i--) {
90
- const entry = this.#stack[i];
91
- const parentEntry = this.#stack[i - 1];
92
- if (entry.subgraphNodeId) {
93
- const subNode = parentEntry.editor.getNode(entry.subgraphNodeId);
94
- if (subNode?._isSubgraph) {
95
- subNode.setInnerPositions(this.#capturePositions(entry.editor, entry.positions));
96
- subNode.setInnerTransform(entry.transform);
97
- }
98
- }
99
- }
100
-
101
- // Save current deep level positions before truncating
102
- const currentEntry = this.#stack[this.#stack.length - 1];
103
- currentEntry.positions = this.#capturePositions(currentEntry.editor, currentEntry.positions);
104
- currentEntry.transform = this.#captureTransform();
105
-
106
- // Truncate stack
107
- this.#stack.length = level + 1;
108
-
109
- // Switch canvas
110
- this.#applyLevel(level);
111
- this.#notifyNavigate();
112
- }
113
-
114
- /**
115
- * Get current breadcrumb path
116
- * @returns {Array<{ label: string, level: number }>}
117
- */
118
- getPath() {
119
- return this.#stack.map((entry, i) => ({
120
- label: entry.label,
121
- level: i,
122
- }));
123
- }
124
-
125
- /**
126
- * Get current depth (0 = root)
127
- * @returns {number}
128
- */
129
- get depth() {
130
- return this.#stack.length - 1;
131
- }
132
-
133
- /**
134
- * Get current editor
135
- * @returns {import('../core/Editor.js').NodeEditor}
136
- */
137
- get currentEditor() {
138
- return this.#stack[this.#stack.length - 1].editor;
139
- }
140
-
141
- /**
142
- * Apply a stack level to the canvas
143
- * @param {number} level
144
- */
145
- #applyLevel(level) {
146
- const entry = this.#stack[level];
147
- if (!entry || !this.#canvas) return;
148
-
149
- // Rebind editor
150
- this.#canvas.setEditor(entry.editor);
151
-
152
- // Restore positions
153
- for (const [nodeId, pos] of Object.entries(entry.positions)) {
154
- this.#canvas.setNodePosition(nodeId, pos.x, pos.y);
155
- }
156
-
157
- // Restore transform
158
- this.#canvas.$.panX = entry.transform.panX;
159
- this.#canvas.$.panY = entry.transform.panY;
160
- this.#canvas.$.zoom = entry.transform.zoom;
161
- }
162
-
163
- /**
164
- * Capture current node positions from editor.
165
- * Falls back to existing entry positions when DOM elements
166
- * have already been cleared (e.g. during setEditor transitions).
167
- * @param {import('../core/Editor.js').NodeEditor} editor
168
- * @param {Object<string, { x: number, y: number }>} [fallback={}]
169
- * @returns {Object<string, { x: number, y: number }>}
170
- */
171
- #capturePositions(editor, fallback = {}) {
172
- const positions = { ...fallback };
173
- if (!this.#canvas) return positions;
174
- for (const node of editor.getNodes()) {
175
- const el = this.#canvas.getNodeView?.(node.id);
176
- if (el?._position) {
177
- positions[node.id] = { ...el._position };
178
- }
179
- }
180
- return positions;
181
- }
182
-
183
- /**
184
- * Capture current viewport transform
185
- * @returns {{ panX: number, panY: number, zoom: number }}
186
- */
187
- #captureTransform() {
188
- if (!this.#canvas) return { panX: 0, panY: 0, zoom: 1 };
189
- return {
190
- panX: this.#canvas.$.panX ?? 0,
191
- panY: this.#canvas.$.panY ?? 0,
192
- zoom: this.#canvas.$.zoom ?? 1,
193
- };
194
- }
195
-
196
- #notifyNavigate() {
197
- if (this.#onNavigate) {
198
- this.#onNavigate(this.getPath());
199
- }
200
- }
201
- }