mirror of
				https://github.com/mermaid-js/mermaid.git
				synced 2025-11-04 04:44:08 +01:00 
			
		
		
		
	Compare commits
	
		
			55 Commits
		
	
	
		
			@mermaid-j
			...
			a716a525c3
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 
						 | 
					a716a525c3 | ||
| 
						 | 
					11abfc9ae5 | ||
| 
						 | 
					81b0ffb92a | ||
| 
						 | 
					dd36046e23 | ||
| 
						 | 
					1507435e15 | ||
| 
						 | 
					68c01b76bf | ||
| 
						 | 
					28717e108d | ||
| 
						 | 
					688d9b383d | ||
| 
						 | 
					e68424d748 | ||
| 
						 | 
					204a9a338f | ||
| 
						 | 
					6a6a39ff33 | ||
| 
						 | 
					b296db9a33 | ||
| 
						 | 
					01ce84d8ee | ||
| 
						 | 
					f48e663d4c | ||
| 
						 | 
					a4aa2bd355 | ||
| 
						 | 
					b51b9d50c2 | ||
| 
						 | 
					b61780f735 | ||
| 
						 | 
					074f18dfb8 | ||
| 
						 | 
					d7308b0f43 | ||
| 
						 | 
					2f1860386a | ||
| 
						 | 
					f0bca7da55 | ||
| 
						 | 
					6fcdf5bfcc | ||
| 
						 | 
					e2ce0450c1 | ||
| 
						 | 
					c95c64139d | ||
| 
						 | 
					a7f12f1baa | ||
| 
						 | 
					2a8653de2b | ||
| 
						 | 
					a92c3bb251 | ||
| 
						 | 
					3677abe9e5 | ||
| 
						 | 
					95847ad236 | ||
| 
						 | 
					e0152fb873 | ||
| 
						 | 
					2298b96d8e | ||
| 
						 | 
					5db83365b6 | ||
| 
						 | 
					341a81a113 | ||
| 
						 | 
					8a62b4cace | ||
| 
						 | 
					ccafc20917 | ||
| 
						 | 
					d5cb4eaa59 | ||
| 
						 | 
					425fb7ee33 | ||
| 
						 | 
					cd6f8e5a24 | ||
| 
						 | 
					8314554eb5 | ||
| 
						 | 
					b7c03dc27e | ||
| 
						 | 
					c7f2f609a9 | ||
| 
						 | 
					4c3de3a1ec | ||
| 
						 | 
					f0445b74d1 | ||
| 
						 | 
					ba52eef257 | ||
| 
						 | 
					c13ce2a5c0 | ||
| 
						 | 
					d2463f41b5 | ||
| 
						 | 
					eadb343292 | ||
| 
						 | 
					e7208622f7 | ||
| 
						 | 
					fbae611406 | ||
| 
						 | 
					34027bc589 | ||
| 
						 | 
					f2eef37599 | ||
| 
						 | 
					1e3ea13323 | ||
| 
						 | 
					4c8c48cde9 | ||
| 
						 | 
					c8e50276e8 | ||
| 
						 | 
					1e6419a63f | 
@@ -33,6 +33,11 @@ export const packageOptions = {
 | 
			
		||||
    packageName: 'mermaid-layout-elk',
 | 
			
		||||
    file: 'layouts.ts',
 | 
			
		||||
  },
 | 
			
		||||
  'mermaid-layout-tidy-tree': {
 | 
			
		||||
    name: 'mermaid-layout-tidy-tree',
 | 
			
		||||
    packageName: 'mermaid-layout-tidy-tree',
 | 
			
		||||
    file: 'index.ts',
 | 
			
		||||
  },
 | 
			
		||||
  examples: {
 | 
			
		||||
    name: 'mermaid-examples',
 | 
			
		||||
    packageName: 'examples',
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										7
									
								
								.changeset/hungry-guests-drive.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								.changeset/hungry-guests-drive.md
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,7 @@
 | 
			
		||||
---
 | 
			
		||||
'mermaid': minor
 | 
			
		||||
'@mermaid-js/layout-tidy-tree': minor
 | 
			
		||||
'@mermaid-js/layout-elk': minor
 | 
			
		||||
---
 | 
			
		||||
 | 
			
		||||
feat: Update mindmap rendering to support multiple layouts, improved edge intersections, and new shapes
 | 
			
		||||
@@ -7,6 +7,7 @@ catmull
 | 
			
		||||
compositTitleSize
 | 
			
		||||
curv
 | 
			
		||||
doublecircle
 | 
			
		||||
elem
 | 
			
		||||
elems
 | 
			
		||||
gantt
 | 
			
		||||
gitgraph
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										2
									
								
								.github/workflows/validate-lockfile.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/validate-lockfile.yml
									
									
									
									
										vendored
									
									
								
							@@ -35,7 +35,7 @@ jobs:
 | 
			
		||||
 | 
			
		||||
          # 2) No unwanted vitepress paths
 | 
			
		||||
          if grep -qF 'packages/mermaid/src/vitepress' pnpm-lock.yaml; then
 | 
			
		||||
            issues+=("• Disallowed path 'packages/mermaid/src/vitepress' present. Run `rm -rf packages/mermaid/src/vitepress && pnpm install` to regenerate.")
 | 
			
		||||
            issues+=("• Disallowed path 'packages/mermaid/src/vitepress' present. Run \`rm -rf packages/mermaid/src/vitepress && pnpm install\` to regenerate.")
 | 
			
		||||
          fi
 | 
			
		||||
 | 
			
		||||
          # 3) Lockfile only changes when package.json changes
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										1
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							@@ -4,6 +4,7 @@ node_modules/
 | 
			
		||||
coverage/
 | 
			
		||||
.idea/
 | 
			
		||||
.pnpm-store/
 | 
			
		||||
.instructions/
 | 
			
		||||
 | 
			
		||||
dist
 | 
			
		||||
v8-compile-cache-0
 | 
			
		||||
 
 | 
			
		||||
@@ -159,12 +159,10 @@ root
 | 
			
		||||
  });
 | 
			
		||||
  it('square shape', () => {
 | 
			
		||||
    imgSnapshotTest(
 | 
			
		||||
      `
 | 
			
		||||
mindmap
 | 
			
		||||
      `mindmap
 | 
			
		||||
    root[
 | 
			
		||||
      The root
 | 
			
		||||
    ]
 | 
			
		||||
      `,
 | 
			
		||||
    ]`,
 | 
			
		||||
      {},
 | 
			
		||||
      undefined,
 | 
			
		||||
      shouldHaveRoot
 | 
			
		||||
@@ -172,12 +170,10 @@ mindmap
 | 
			
		||||
  });
 | 
			
		||||
  it('rounded rect shape', () => {
 | 
			
		||||
    imgSnapshotTest(
 | 
			
		||||
      `
 | 
			
		||||
mindmap
 | 
			
		||||
      `mindmap
 | 
			
		||||
    root((
 | 
			
		||||
      The root
 | 
			
		||||
    ))
 | 
			
		||||
      `,
 | 
			
		||||
    ))`,
 | 
			
		||||
      {},
 | 
			
		||||
      undefined,
 | 
			
		||||
      shouldHaveRoot
 | 
			
		||||
@@ -185,12 +181,10 @@ mindmap
 | 
			
		||||
  });
 | 
			
		||||
  it('circle shape', () => {
 | 
			
		||||
    imgSnapshotTest(
 | 
			
		||||
      `
 | 
			
		||||
mindmap
 | 
			
		||||
      `mindmap
 | 
			
		||||
    root(
 | 
			
		||||
      The root
 | 
			
		||||
    )
 | 
			
		||||
      `,
 | 
			
		||||
    )`,
 | 
			
		||||
      {},
 | 
			
		||||
      undefined,
 | 
			
		||||
      shouldHaveRoot
 | 
			
		||||
@@ -198,10 +192,8 @@ mindmap
 | 
			
		||||
  });
 | 
			
		||||
  it('default shape', () => {
 | 
			
		||||
    imgSnapshotTest(
 | 
			
		||||
      `
 | 
			
		||||
mindmap
 | 
			
		||||
  The root
 | 
			
		||||
      `,
 | 
			
		||||
      `mindmap
 | 
			
		||||
  The root`,
 | 
			
		||||
      {},
 | 
			
		||||
      undefined,
 | 
			
		||||
      shouldHaveRoot
 | 
			
		||||
@@ -209,12 +201,10 @@ mindmap
 | 
			
		||||
  });
 | 
			
		||||
  it('adding children', () => {
 | 
			
		||||
    imgSnapshotTest(
 | 
			
		||||
      `
 | 
			
		||||
mindmap
 | 
			
		||||
      `mindmap
 | 
			
		||||
  The root
 | 
			
		||||
    child1
 | 
			
		||||
    child2
 | 
			
		||||
      `,
 | 
			
		||||
    child2`,
 | 
			
		||||
      {},
 | 
			
		||||
      undefined,
 | 
			
		||||
      shouldHaveRoot
 | 
			
		||||
@@ -222,13 +212,11 @@ mindmap
 | 
			
		||||
  });
 | 
			
		||||
  it('adding grand children', () => {
 | 
			
		||||
    imgSnapshotTest(
 | 
			
		||||
      `
 | 
			
		||||
mindmap
 | 
			
		||||
      `mindmap
 | 
			
		||||
  The root
 | 
			
		||||
    child1
 | 
			
		||||
      child2
 | 
			
		||||
      child3
 | 
			
		||||
      `,
 | 
			
		||||
      child3`,
 | 
			
		||||
      {},
 | 
			
		||||
      undefined,
 | 
			
		||||
      shouldHaveRoot
 | 
			
		||||
 
 | 
			
		||||
@@ -131,6 +131,19 @@
 | 
			
		||||
 | 
			
		||||
  <body>
 | 
			
		||||
    <pre id="diagram4" class="mermaid">
 | 
			
		||||
      ---
 | 
			
		||||
      config:
 | 
			
		||||
        layout: elk
 | 
			
		||||
        flowchart:
 | 
			
		||||
          curve: linear
 | 
			
		||||
      ---
 | 
			
		||||
      flowchart LR
 | 
			
		||||
          D["Use<br/>the<br/>editor"] -- Mermaid js --> I["fa:fa-code Text"]
 | 
			
		||||
          I --> D & D
 | 
			
		||||
          D@{ shape: question}
 | 
			
		||||
          I@{ shape: question}
 | 
			
		||||
</pre>
 | 
			
		||||
    <pre id="diagram4" class="mermaid2">
 | 
			
		||||
---
 | 
			
		||||
config:
 | 
			
		||||
  layout: elk
 | 
			
		||||
@@ -157,84 +170,149 @@ treemap
 | 
			
		||||
    "Leaf 2.2": 25
 | 
			
		||||
    "Leaf 2.3": 12
 | 
			
		||||
 | 
			
		||||
classDef class1   fill:red,color:blue,stroke:#FFD600;
 | 
			
		||||
    </pre>
 | 
			
		||||
    <pre id="diagram5" class="mermaid">
 | 
			
		||||
      ---
 | 
			
		||||
      config:
 | 
			
		||||
        layout: elk
 | 
			
		||||
        flowchart:
 | 
			
		||||
          curve: rounded
 | 
			
		||||
      ---
 | 
			
		||||
      flowchart LR
 | 
			
		||||
          I["fa:fa-code Text"] -- Mermaid js --> D["Use<br/>the<br/>editor!"]
 | 
			
		||||
          I --> D & D
 | 
			
		||||
          D@{ shape: question}
 | 
			
		||||
          I@{ shape: question}
 | 
			
		||||
 | 
			
		||||
    </pre>
 | 
			
		||||
    <pre id="diagram4" class="mermaid">
 | 
			
		||||
      ---
 | 
			
		||||
      config:
 | 
			
		||||
        layout: tidy-tree
 | 
			
		||||
      ---
 | 
			
		||||
      mindmap
 | 
			
		||||
      root((mindmap))
 | 
			
		||||
        Origins
 | 
			
		||||
          Long history
 | 
			
		||||
          ::icon(fa fa-book)
 | 
			
		||||
          Popularisation
 | 
			
		||||
            British popular psychology author Tony Buzan
 | 
			
		||||
        Research
 | 
			
		||||
          On effectiveness<br/>and features
 | 
			
		||||
          On Automatic creation
 | 
			
		||||
            Uses
 | 
			
		||||
                Creative techniques
 | 
			
		||||
                Strategic planning
 | 
			
		||||
                Argument mapping
 | 
			
		||||
        Tools
 | 
			
		||||
          Pen and paper
 | 
			
		||||
          Mermaid
 | 
			
		||||
 | 
			
		||||
</pre
 | 
			
		||||
    >
 | 
			
		||||
    <pre id="diagram4" class="mermaid2">
 | 
			
		||||
    </pre>
 | 
			
		||||
    <pre id="diagram4" class="mermaid">
 | 
			
		||||
      ---
 | 
			
		||||
      config:
 | 
			
		||||
        layout: elk
 | 
			
		||||
        flowchart:
 | 
			
		||||
          curve: linear
 | 
			
		||||
      ---
 | 
			
		||||
      flowchart LR
 | 
			
		||||
          A[A] --> B[B]
 | 
			
		||||
          A[A] --- B([C])
 | 
			
		||||
          A@{ shape: diamond}
 | 
			
		||||
          %%B@{ shape: diamond}
 | 
			
		||||
 | 
			
		||||
    </pre>
 | 
			
		||||
    <pre id="diagram4" class="mermaid">
 | 
			
		||||
      ---
 | 
			
		||||
      config:
 | 
			
		||||
        layout: elk
 | 
			
		||||
        flowchart:
 | 
			
		||||
          curve: linear
 | 
			
		||||
      ---
 | 
			
		||||
      flowchart LR
 | 
			
		||||
          A[A] -- Mermaid js --> B[B]
 | 
			
		||||
          A[A] -- Mermaid js --- B[B]
 | 
			
		||||
          A@{ shape: diamond}
 | 
			
		||||
          B@{ shape: diamond}
 | 
			
		||||
 | 
			
		||||
    </pre>
 | 
			
		||||
    <pre id="diagram4" class="mermaid">
 | 
			
		||||
      ---
 | 
			
		||||
      config:
 | 
			
		||||
        layout: elk
 | 
			
		||||
        flowchart:
 | 
			
		||||
          curve: rounded
 | 
			
		||||
      ---
 | 
			
		||||
      flowchart LR
 | 
			
		||||
          D["Use the editor"] -- Mermaid js --> I["fa:fa-code Text"]
 | 
			
		||||
          I --> D & D
 | 
			
		||||
          D@{ shape: question}
 | 
			
		||||
          I@{ shape: question}
 | 
			
		||||
    </pre>
 | 
			
		||||
    <pre id="diagram4" class="mermaid">
 | 
			
		||||
      ---
 | 
			
		||||
      config:
 | 
			
		||||
        layout: elk
 | 
			
		||||
        flowchart:
 | 
			
		||||
          curve: rounded
 | 
			
		||||
        elk:
 | 
			
		||||
          nodePlacementStrategy: NETWORK_SIMPLEX
 | 
			
		||||
      ---
 | 
			
		||||
      flowchart LR
 | 
			
		||||
          D["Use the editor"] -- Mermaid js --> I["fa:fa-code Text"]
 | 
			
		||||
          D --> I & I
 | 
			
		||||
          a["a"]
 | 
			
		||||
          D@{ shape: trap-b}
 | 
			
		||||
          I@{ shape: lean-l}
 | 
			
		||||
    </pre>
 | 
			
		||||
    <pre id="diagram4" class="mermaid">
 | 
			
		||||
---
 | 
			
		||||
config:
 | 
			
		||||
  treemap:
 | 
			
		||||
    valueFormat: '$0,0'
 | 
			
		||||
  layout: elk
 | 
			
		||||
 | 
			
		||||
---
 | 
			
		||||
treemap
 | 
			
		||||
"Budget"
 | 
			
		||||
    "Operations"
 | 
			
		||||
        "Salaries": 7000
 | 
			
		||||
        "Equipment": 2000
 | 
			
		||||
        "Supplies": 1000
 | 
			
		||||
    "Marketing"
 | 
			
		||||
        "Advertising": 4000
 | 
			
		||||
        "Events": 1000
 | 
			
		||||
flowchart LR
 | 
			
		||||
 %% subgraph s1["Untitled subgraph"]
 | 
			
		||||
        C["Evaluate"]
 | 
			
		||||
 %% end
 | 
			
		||||
 | 
			
		||||
</pre
 | 
			
		||||
    >
 | 
			
		||||
    B --> C
 | 
			
		||||
    </pre>
 | 
			
		||||
    <pre id="diagram4" class="mermaid">
 | 
			
		||||
    treemap
 | 
			
		||||
      title Accessible Treemap Title
 | 
			
		||||
      "Category A"
 | 
			
		||||
          "Item A1": 10
 | 
			
		||||
          "Item A2": 20
 | 
			
		||||
      "Category B"
 | 
			
		||||
          "Item B1": 15
 | 
			
		||||
          "Item B2": 25
 | 
			
		||||
    </pre>
 | 
			
		||||
    <pre id="diagram4" class="mermaid2">
 | 
			
		||||
      flowchart LR
 | 
			
		||||
        AB["apa@apa@"] --> B(("`apa@apa`"))
 | 
			
		||||
    </pre>
 | 
			
		||||
    <pre id="diagram4" class="mermaid2">
 | 
			
		||||
      flowchart
 | 
			
		||||
        D(("for D"))
 | 
			
		||||
    </pre>
 | 
			
		||||
    <pre id="diagram4" class="mermaid2">
 | 
			
		||||
      flowchart LR
 | 
			
		||||
        A e1@==> B
 | 
			
		||||
        e1@{ animate: true}
 | 
			
		||||
    </pre>
 | 
			
		||||
    <pre id="diagram4" class="mermaid2">
 | 
			
		||||
---
 | 
			
		||||
config:
 | 
			
		||||
  layout: elk
 | 
			
		||||
  flowchart:
 | 
			
		||||
    //curve: linear
 | 
			
		||||
---
 | 
			
		||||
flowchart LR
 | 
			
		||||
  A e1@--> B
 | 
			
		||||
  classDef animate stroke-width:2,stroke-dasharray:10\,8,stroke-dashoffset:-180,animation: edge-animation-frame 6s linear infinite, stroke-linecap: round
 | 
			
		||||
  class e1 animate
 | 
			
		||||
    </pre>
 | 
			
		||||
    <h2>infinite</h2>
 | 
			
		||||
    <pre id="diagram4" class="mermaid2">
 | 
			
		||||
flowchart LR
 | 
			
		||||
  A e1@--> B
 | 
			
		||||
  classDef animate stroke-dasharray: 9\,5,stroke-dashoffset: 900,animation: dash 25s linear infinite;
 | 
			
		||||
  class e1 animate
 | 
			
		||||
    </pre>
 | 
			
		||||
    <h2>Mermaid - edge-animation-slow</h2>
 | 
			
		||||
    <pre id="diagram4" class="mermaid2">
 | 
			
		||||
flowchart LR
 | 
			
		||||
  A e1@--> B
 | 
			
		||||
e1@{ animation: fast}
 | 
			
		||||
    </pre>
 | 
			
		||||
    <h2>Mermaid - edge-animation-fast</h2>
 | 
			
		||||
    <pre id="diagram4" class="mermaid2">
 | 
			
		||||
flowchart LR
 | 
			
		||||
  A e1@--> B
 | 
			
		||||
  classDef animate stroke-dasharray: 1000,stroke-dashoffset: 1000,animation: dash 10s linear;
 | 
			
		||||
  class e1 edge-animation-fast
 | 
			
		||||
    </pre>
 | 
			
		||||
%% A ==> B
 | 
			
		||||
%% A2 --> B2
 | 
			
		||||
A{A} --> B((Bo boo)) & B & B & B
 | 
			
		||||
 | 
			
		||||
    <pre id="diagram4" class="mermaid2">
 | 
			
		||||
 | 
			
		||||
info    </pre
 | 
			
		||||
    >
 | 
			
		||||
    <pre id="diagram4" class="mermaid2">
 | 
			
		||||
    </pre>
 | 
			
		||||
    <pre id="diagram4" class="mermaid">
 | 
			
		||||
      ---
 | 
			
		||||
      config:
 | 
			
		||||
        layout: elk
 | 
			
		||||
        theme: default
 | 
			
		||||
        look: classic
 | 
			
		||||
      ---
 | 
			
		||||
      flowchart LR
 | 
			
		||||
       subgraph s1["APA"]
 | 
			
		||||
              D{"Use the editor"}
 | 
			
		||||
        end
 | 
			
		||||
       subgraph S2["S2"]
 | 
			
		||||
              s1
 | 
			
		||||
              I>"fa:fa-code Text"]
 | 
			
		||||
              E["E"]
 | 
			
		||||
        end
 | 
			
		||||
          D -- Mermaid js --> I
 | 
			
		||||
          D --> I & E
 | 
			
		||||
          E --> I
 | 
			
		||||
    </pre>
 | 
			
		||||
    <pre id="diagram4" class="mermaid">
 | 
			
		||||
---
 | 
			
		||||
config:
 | 
			
		||||
  layout: elk
 | 
			
		||||
@@ -259,7 +337,7 @@ config:
 | 
			
		||||
      end
 | 
			
		||||
      end
 | 
			
		||||
    </pre>
 | 
			
		||||
    <pre id="diagram4" class="mermaid2">
 | 
			
		||||
    <pre id="diagram4" class="mermaid">
 | 
			
		||||
---
 | 
			
		||||
config:
 | 
			
		||||
  layout: elk
 | 
			
		||||
@@ -272,7 +350,7 @@ config:
 | 
			
		||||
      D-->I
 | 
			
		||||
      D-->I
 | 
			
		||||
    </pre>
 | 
			
		||||
    <pre id="diagram4" class="mermaid2">
 | 
			
		||||
    <pre id="diagram4" class="mermaid">
 | 
			
		||||
---
 | 
			
		||||
config:
 | 
			
		||||
  layout: elk
 | 
			
		||||
@@ -311,7 +389,7 @@ flowchart LR
 | 
			
		||||
    n8@{ shape: rect}
 | 
			
		||||
 | 
			
		||||
    </pre>
 | 
			
		||||
    <pre id="diagram4" class="mermaid2">
 | 
			
		||||
    <pre id="diagram4" class="mermaid">
 | 
			
		||||
---
 | 
			
		||||
config:
 | 
			
		||||
  layout: elk
 | 
			
		||||
@@ -327,7 +405,7 @@ flowchart LR
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
    </pre>
 | 
			
		||||
    <pre id="diagram4" class="mermaid2">
 | 
			
		||||
    <pre id="diagram4" class="mermaid">
 | 
			
		||||
---
 | 
			
		||||
config:
 | 
			
		||||
  layout: elk
 | 
			
		||||
@@ -336,7 +414,7 @@ flowchart LR
 | 
			
		||||
    A{A} --> B & C
 | 
			
		||||
</pre
 | 
			
		||||
    >
 | 
			
		||||
    <pre id="diagram4" class="mermaid2">
 | 
			
		||||
    <pre id="diagram4" class="mermaid">
 | 
			
		||||
---
 | 
			
		||||
config:
 | 
			
		||||
  layout: elk
 | 
			
		||||
@@ -348,7 +426,7 @@ flowchart LR
 | 
			
		||||
    end
 | 
			
		||||
</pre
 | 
			
		||||
    >
 | 
			
		||||
    <pre id="diagram4" class="mermaid2">
 | 
			
		||||
    <pre id="diagram4" class="mermaid">
 | 
			
		||||
---
 | 
			
		||||
config:
 | 
			
		||||
  layout: elk
 | 
			
		||||
@@ -366,7 +444,7 @@ flowchart LR
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
    </pre>
 | 
			
		||||
    <pre id="diagram4" class="mermaid2">
 | 
			
		||||
    <pre id="diagram4" class="mermaid">
 | 
			
		||||
---
 | 
			
		||||
config:
 | 
			
		||||
  kanban:
 | 
			
		||||
@@ -385,81 +463,81 @@ kanban
 | 
			
		||||
    task3[💻 Develop login feature]@{ ticket: 103 }
 | 
			
		||||
 | 
			
		||||
    </pre>
 | 
			
		||||
    <pre id="diagram4" class="mermaid2">
 | 
			
		||||
    <pre id="diagram4" class="mermaid">
 | 
			
		||||
flowchart LR
 | 
			
		||||
nA[Default] --> A@{ icon: 'fa:bell', form: 'rounded' }
 | 
			
		||||
 | 
			
		||||
    </pre>
 | 
			
		||||
    <pre id="diagram4" class="mermaid2">
 | 
			
		||||
    <pre id="diagram4" class="mermaid">
 | 
			
		||||
flowchart LR
 | 
			
		||||
nA[Style] --> A@{ icon: 'fa:bell', form: 'rounded' }
 | 
			
		||||
style A fill:#f9f,stroke:#333,stroke-width:4px
 | 
			
		||||
    </pre>
 | 
			
		||||
    <pre id="diagram4" class="mermaid2">
 | 
			
		||||
    <pre id="diagram4" class="mermaid">
 | 
			
		||||
flowchart LR
 | 
			
		||||
nA[Class] --> A@{ icon: 'fa:bell', form: 'rounded' }
 | 
			
		||||
A:::AClass
 | 
			
		||||
classDef AClass fill:#f9f,stroke:#333,stroke-width:4px
 | 
			
		||||
    </pre>
 | 
			
		||||
    <pre id="diagram4" class="mermaid2">
 | 
			
		||||
    <pre id="diagram4" class="mermaid">
 | 
			
		||||
flowchart LR
 | 
			
		||||
  nA[Class] --> A@{ icon: 'logos:aws', form: 'rounded' }
 | 
			
		||||
 | 
			
		||||
    </pre>
 | 
			
		||||
    <pre id="diagram4" class="mermaid2">
 | 
			
		||||
    <pre id="diagram4" class="mermaid">
 | 
			
		||||
flowchart LR
 | 
			
		||||
nA[Default] --> A@{ icon: 'fa:bell', form: 'square' }
 | 
			
		||||
 | 
			
		||||
    </pre>
 | 
			
		||||
    <pre id="diagram4" class="mermaid2">
 | 
			
		||||
    <pre id="diagram4" class="mermaid">
 | 
			
		||||
flowchart LR
 | 
			
		||||
nA[Style] --> A@{ icon: 'fa:bell', form: 'square' }
 | 
			
		||||
style A fill:#f9f,stroke:#333,stroke-width:4px
 | 
			
		||||
    </pre>
 | 
			
		||||
    <pre id="diagram4" class="mermaid2">
 | 
			
		||||
    <pre id="diagram4" class="mermaid">
 | 
			
		||||
flowchart LR
 | 
			
		||||
nA[Class] --> A@{ icon: 'fa:bell', form: 'square' }
 | 
			
		||||
A:::AClass
 | 
			
		||||
classDef AClass fill:#f9f,stroke:#333,stroke-width:4px
 | 
			
		||||
    </pre>
 | 
			
		||||
    <pre id="diagram4" class="mermaid2">
 | 
			
		||||
    <pre id="diagram4" class="mermaid">
 | 
			
		||||
flowchart LR
 | 
			
		||||
  nA[Class] --> A@{ icon: 'logos:aws', form: 'square' }
 | 
			
		||||
 | 
			
		||||
    </pre>
 | 
			
		||||
    <pre id="diagram4" class="mermaid2">
 | 
			
		||||
    <pre id="diagram4" class="mermaid">
 | 
			
		||||
flowchart LR
 | 
			
		||||
nA[Default] --> A@{ icon: 'fa:bell', form: 'circle' }
 | 
			
		||||
 | 
			
		||||
    </pre>
 | 
			
		||||
    <pre id="diagram4" class="mermaid2">
 | 
			
		||||
    <pre id="diagram4" class="mermaid">
 | 
			
		||||
flowchart LR
 | 
			
		||||
nA[Style] --> A@{ icon: 'fa:bell', form: 'circle' }
 | 
			
		||||
style A fill:#f9f,stroke:#333,stroke-width:4px
 | 
			
		||||
    </pre>
 | 
			
		||||
    <pre id="diagram4" class="mermaid2">
 | 
			
		||||
    <pre id="diagram4" class="mermaid">
 | 
			
		||||
flowchart LR
 | 
			
		||||
nA[Class] --> A@{ icon: 'fa:bell', form: 'circle' }
 | 
			
		||||
A:::AClass
 | 
			
		||||
classDef AClass fill:#f9f,stroke:#333,stroke-width:4px
 | 
			
		||||
    </pre>
 | 
			
		||||
    <pre id="diagram4" class="mermaid2">
 | 
			
		||||
    <pre id="diagram4" class="mermaid">
 | 
			
		||||
flowchart LR
 | 
			
		||||
  nA[Class] --> A@{ icon: 'logos:aws', form: 'circle' }
 | 
			
		||||
  A:::AClass
 | 
			
		||||
  classDef AClass fill:#f9f,stroke:#333,stroke-width:4px
 | 
			
		||||
    </pre>
 | 
			
		||||
    <pre id="diagram4" class="mermaid2">
 | 
			
		||||
    <pre id="diagram4" class="mermaid">
 | 
			
		||||
flowchart LR
 | 
			
		||||
  nA[Style] --> A@{ icon: 'logos:aws', form: 'circle' }
 | 
			
		||||
  style A fill:#f9f,stroke:#333,stroke-width:4px
 | 
			
		||||
    </pre>
 | 
			
		||||
    <pre id="diagram4" class="mermaid2">
 | 
			
		||||
    <pre id="diagram4" class="mermaid">
 | 
			
		||||
kanban
 | 
			
		||||
  id2[In progress]
 | 
			
		||||
    docs[Create Blog about the new diagram]@{ priority: 'Very Low', ticket: MC-2037, assigned: 'knsv' }
 | 
			
		||||
    </pre>
 | 
			
		||||
    <pre id="diagram4" class="mermaid2">
 | 
			
		||||
    <pre id="diagram4" class="mermaid">
 | 
			
		||||
---
 | 
			
		||||
config:
 | 
			
		||||
  kanban:
 | 
			
		||||
@@ -488,6 +566,7 @@ kanban
 | 
			
		||||
    <script type="module">
 | 
			
		||||
      import mermaid from './mermaid.esm.mjs';
 | 
			
		||||
      import layouts from './mermaid-layout-elk.esm.mjs';
 | 
			
		||||
      import tidyTreeLayouts from './mermaid-layout-tidy-tree.esm.mjs';
 | 
			
		||||
 | 
			
		||||
      const staticBellIconPack = {
 | 
			
		||||
        prefix: 'fa6-regular',
 | 
			
		||||
@@ -513,6 +592,7 @@ kanban
 | 
			
		||||
        },
 | 
			
		||||
      ]);
 | 
			
		||||
      mermaid.registerLayoutLoaders(layouts);
 | 
			
		||||
      mermaid.registerLayoutLoaders(tidyTreeLayouts);
 | 
			
		||||
      mermaid.parseError = function (err, hash) {
 | 
			
		||||
        console.error('Mermaid error: ', err);
 | 
			
		||||
      };
 | 
			
		||||
@@ -530,7 +610,7 @@ kanban
 | 
			
		||||
        // look: 'handDrawn',
 | 
			
		||||
        // 'elk.nodePlacement.strategy': 'NETWORK_SIMPLEX',
 | 
			
		||||
        // layout: 'dagre',
 | 
			
		||||
        // layout: 'elk',
 | 
			
		||||
        layout: 'elk',
 | 
			
		||||
        // layout: 'fixed',
 | 
			
		||||
        // htmlLabels: false,
 | 
			
		||||
        flowchart: { titleTopMargin: 10 },
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										379
									
								
								cypress/platform/mindmap-layouts.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										379
									
								
								cypress/platform/mindmap-layouts.html
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,379 @@
 | 
			
		||||
<!doctype html>
 | 
			
		||||
<html lang="en">
 | 
			
		||||
  <head>
 | 
			
		||||
    <meta charset="utf-8" />
 | 
			
		||||
    <meta http-equiv="X-UA-Compatible" content="IE=edge" />
 | 
			
		||||
    <title>Mermaid Quick Test Page</title>
 | 
			
		||||
    <link rel="icon" type="image/png" href="" />
 | 
			
		||||
    <style>
 | 
			
		||||
      div.mermaid {
 | 
			
		||||
        font-family: 'Courier New', Courier, monospace !important;
 | 
			
		||||
      }
 | 
			
		||||
    </style>
 | 
			
		||||
  </head>
 | 
			
		||||
 | 
			
		||||
  <body>
 | 
			
		||||
    <pre class="mermaid">
 | 
			
		||||
 ---
 | 
			
		||||
      config:
 | 
			
		||||
        layout: tidy-tree
 | 
			
		||||
      ---
 | 
			
		||||
      mindmap
 | 
			
		||||
      root((mindmap))
 | 
			
		||||
        A
 | 
			
		||||
        B
 | 
			
		||||
    </pre>
 | 
			
		||||
    <pre class="mermaid">
 | 
			
		||||
 ---
 | 
			
		||||
      config:
 | 
			
		||||
        layout: dagre
 | 
			
		||||
      ---
 | 
			
		||||
      mindmap
 | 
			
		||||
      root((mindmap))
 | 
			
		||||
        A
 | 
			
		||||
        B
 | 
			
		||||
    </pre>
 | 
			
		||||
    <pre class="mermaid">
 | 
			
		||||
 ---
 | 
			
		||||
      config:
 | 
			
		||||
        layout: elk
 | 
			
		||||
      ---
 | 
			
		||||
      mindmap
 | 
			
		||||
      root((mindmap))
 | 
			
		||||
        A
 | 
			
		||||
        B
 | 
			
		||||
    </pre>
 | 
			
		||||
    <pre class="mermaid">
 | 
			
		||||
 ---
 | 
			
		||||
      config:
 | 
			
		||||
        layout: cose-bilkent
 | 
			
		||||
      ---
 | 
			
		||||
      mindmap
 | 
			
		||||
      root((mindmap))
 | 
			
		||||
        A
 | 
			
		||||
        B
 | 
			
		||||
    </pre>
 | 
			
		||||
    <pre class="mermaid">
 | 
			
		||||
    ---
 | 
			
		||||
      config:
 | 
			
		||||
        layout: tidy-tree
 | 
			
		||||
      ---
 | 
			
		||||
      mindmap
 | 
			
		||||
      root((mindmap is a long thing))
 | 
			
		||||
        A
 | 
			
		||||
        B
 | 
			
		||||
        C
 | 
			
		||||
        D
 | 
			
		||||
    </pre>
 | 
			
		||||
    <pre class="mermaid">
 | 
			
		||||
    ---
 | 
			
		||||
      config:
 | 
			
		||||
        layout: dagre
 | 
			
		||||
      ---
 | 
			
		||||
      mindmap
 | 
			
		||||
      root((mindmap is a long thing))
 | 
			
		||||
        A
 | 
			
		||||
        B
 | 
			
		||||
        C
 | 
			
		||||
        D
 | 
			
		||||
    </pre>
 | 
			
		||||
    <pre class="mermaid">
 | 
			
		||||
    ---
 | 
			
		||||
      config:
 | 
			
		||||
        layout: elk
 | 
			
		||||
      ---
 | 
			
		||||
      mindmap
 | 
			
		||||
      root((mindmap is a long thing))
 | 
			
		||||
        A
 | 
			
		||||
        B
 | 
			
		||||
        C
 | 
			
		||||
        D
 | 
			
		||||
    </pre>
 | 
			
		||||
    <pre class="mermaid">
 | 
			
		||||
    ---
 | 
			
		||||
      config:
 | 
			
		||||
        layout: cose-bilkent
 | 
			
		||||
      ---
 | 
			
		||||
      mindmap
 | 
			
		||||
      root((mindmap is a long thing))
 | 
			
		||||
        A
 | 
			
		||||
        B
 | 
			
		||||
        C
 | 
			
		||||
        D
 | 
			
		||||
    </pre>
 | 
			
		||||
    <pre class="mermaid">
 | 
			
		||||
      mindmap
 | 
			
		||||
      root((mindmap is a long thing))
 | 
			
		||||
        A
 | 
			
		||||
        B
 | 
			
		||||
        C
 | 
			
		||||
        D
 | 
			
		||||
    </pre>
 | 
			
		||||
    <pre class="mermaid">
 | 
			
		||||
    ---
 | 
			
		||||
      config:
 | 
			
		||||
        layout: tidy-tree
 | 
			
		||||
      ---
 | 
			
		||||
      mindmap
 | 
			
		||||
      root((mindmap))
 | 
			
		||||
        Origins
 | 
			
		||||
          Long history
 | 
			
		||||
          ::icon(fa fa-book)
 | 
			
		||||
          Popularisation
 | 
			
		||||
            British popular psychology author Tony Buzan
 | 
			
		||||
        Research
 | 
			
		||||
          On effectiveness<br/>and features
 | 
			
		||||
          On Automatic creation
 | 
			
		||||
            Uses
 | 
			
		||||
                Creative techniques
 | 
			
		||||
                Strategic planning
 | 
			
		||||
                Argument mapping
 | 
			
		||||
        Tools
 | 
			
		||||
              id)I am a cloud(
 | 
			
		||||
                  id))I am a bang((
 | 
			
		||||
                    Tools
 | 
			
		||||
    </pre>
 | 
			
		||||
    <pre class="mermaid">
 | 
			
		||||
    ---
 | 
			
		||||
      config:
 | 
			
		||||
        layout: dagre
 | 
			
		||||
      ---
 | 
			
		||||
      mindmap
 | 
			
		||||
      root((mindmap))
 | 
			
		||||
        Origins
 | 
			
		||||
          Long history
 | 
			
		||||
          ::icon(fa fa-book)
 | 
			
		||||
          Popularisation
 | 
			
		||||
            British popular psychology author Tony Buzan
 | 
			
		||||
        Research
 | 
			
		||||
          On effectiveness<br/>and features
 | 
			
		||||
          On Automatic creation
 | 
			
		||||
            Uses
 | 
			
		||||
                Creative techniques
 | 
			
		||||
                Strategic planning
 | 
			
		||||
                Argument mapping
 | 
			
		||||
        Tools
 | 
			
		||||
              id)I am a cloud(
 | 
			
		||||
                  id))I am a bang((
 | 
			
		||||
                    Tools
 | 
			
		||||
    </pre>
 | 
			
		||||
    <pre class="mermaid">
 | 
			
		||||
    ---
 | 
			
		||||
      config:
 | 
			
		||||
        layout: elk
 | 
			
		||||
      ---
 | 
			
		||||
      mindmap
 | 
			
		||||
      root((mindmap))
 | 
			
		||||
        Origins
 | 
			
		||||
          Long history
 | 
			
		||||
          ::icon(fa fa-book)
 | 
			
		||||
          Popularisation
 | 
			
		||||
            British popular psychology author Tony Buzan
 | 
			
		||||
        Research
 | 
			
		||||
          On effectiveness<br/>and features
 | 
			
		||||
          On Automatic creation
 | 
			
		||||
            Uses
 | 
			
		||||
                Creative techniques
 | 
			
		||||
                Strategic planning
 | 
			
		||||
                Argument mapping
 | 
			
		||||
        Tools
 | 
			
		||||
              id)I am a cloud(
 | 
			
		||||
                  id))I am a bang((
 | 
			
		||||
                    Tools
 | 
			
		||||
    </pre>
 | 
			
		||||
    <pre class="mermaid">
 | 
			
		||||
    ---
 | 
			
		||||
      config:
 | 
			
		||||
        layout: cose-bilkent
 | 
			
		||||
      ---
 | 
			
		||||
      mindmap
 | 
			
		||||
      root((mindmap))
 | 
			
		||||
        Origins
 | 
			
		||||
          Long history
 | 
			
		||||
          ::icon(fa fa-book)
 | 
			
		||||
          Popularisation
 | 
			
		||||
            British popular psychology author Tony Buzan
 | 
			
		||||
        Research
 | 
			
		||||
          On effectiveness<br/>and features
 | 
			
		||||
          On Automatic creation
 | 
			
		||||
            Uses
 | 
			
		||||
                Creative techniques
 | 
			
		||||
                Strategic planning
 | 
			
		||||
                Argument mapping
 | 
			
		||||
        Tools
 | 
			
		||||
              id)I am a cloud(
 | 
			
		||||
                  id))I am a bang((
 | 
			
		||||
                    Tools
 | 
			
		||||
    </pre>
 | 
			
		||||
    <pre class="mermaid">
 | 
			
		||||
      ---
 | 
			
		||||
      config:
 | 
			
		||||
        layout: tidy-tree
 | 
			
		||||
      ---
 | 
			
		||||
      mindmap
 | 
			
		||||
      root((mindmap))
 | 
			
		||||
        A
 | 
			
		||||
          a
 | 
			
		||||
            apa[I am a long long multline string passing several levels of text. Lorum ipsum and so on. I am a long long multline string passing several levels of text. Lorum ipsum and so on. I am a long long multline string passing several levels of text. Lorum ipsum and so on. I am a long long multline string passing several levels of text. Lorum ipsum and so on]
 | 
			
		||||
            apa2[I am a long long multline string passing several levels of text. Lorum ipsum and so on. I am a long long multline string passing several levels of text. Lorum ipsum and so on. I am a long long multline string passing several levels of text. Lorum ipsum and so on. I am a long long multline string passing several levels of text. Lorum ipsum and so on]
 | 
			
		||||
          b
 | 
			
		||||
          c
 | 
			
		||||
          d
 | 
			
		||||
        B
 | 
			
		||||
            apa3[I am a long long multline string passing several levels of text. Lorum ipsum and so on. I am a long long multline string passing several levels of text. Lorum ipsum and so on. I am a long long multline string passing several levels of text. Lorum ipsum and so on. I am a long long multline string passing several levels of text. Lorum ipsum and so on]
 | 
			
		||||
        D
 | 
			
		||||
          apa5[I am a long long multline string passing several levels of text. Lorum ipsum and so on. I am a long long multline string passing several levels of text. Lorum ipsum and so on. I am a long long multline string passing several levels of text. Lorum ipsum and so on. I am a long long multline string passing several levels of text. Lorum ipsum and so on]
 | 
			
		||||
          apa4[I am a long long multline string passing several levels of text. Lorum ipsum and so on. I am a long long multline string passing several levels of text. Lorum ipsum and so on. I am a long long multline string passing several levels of text. Lorum ipsum and so on. I am a long long multline string passing several levels of text. Lorum ipsum and so on]
 | 
			
		||||
 | 
			
		||||
    </pre>
 | 
			
		||||
    <pre class="mermaid">
 | 
			
		||||
      ---
 | 
			
		||||
      config:
 | 
			
		||||
        layout: dagre
 | 
			
		||||
      ---
 | 
			
		||||
      mindmap
 | 
			
		||||
      root((mindmap))
 | 
			
		||||
        A
 | 
			
		||||
          a
 | 
			
		||||
            apa[I am a long long multline string passing several levels of text. Lorum ipsum and so on. I am a long long multline string passing several levels of text. Lorum ipsum and so on. I am a long long multline string passing several levels of text. Lorum ipsum and so on. I am a long long multline string passing several levels of text. Lorum ipsum and so on]
 | 
			
		||||
            apa2[I am a long long multline string passing several levels of text. Lorum ipsum and so on. I am a long long multline string passing several levels of text. Lorum ipsum and so on. I am a long long multline string passing several levels of text. Lorum ipsum and so on. I am a long long multline string passing several levels of text. Lorum ipsum and so on]
 | 
			
		||||
          b
 | 
			
		||||
          c
 | 
			
		||||
          d
 | 
			
		||||
        B
 | 
			
		||||
            apa3[I am a long long multline string passing several levels of text. Lorum ipsum and so on. I am a long long multline string passing several levels of text. Lorum ipsum and so on. I am a long long multline string passing several levels of text. Lorum ipsum and so on. I am a long long multline string passing several levels of text. Lorum ipsum and so on]
 | 
			
		||||
        D
 | 
			
		||||
          apa5[I am a long long multline string passing several levels of text. Lorum ipsum and so on. I am a long long multline string passing several levels of text. Lorum ipsum and so on. I am a long long multline string passing several levels of text. Lorum ipsum and so on. I am a long long multline string passing several levels of text. Lorum ipsum and so on]
 | 
			
		||||
          apa4[I am a long long multline string passing several levels of text. Lorum ipsum and so on. I am a long long multline string passing several levels of text. Lorum ipsum and so on. I am a long long multline string passing several levels of text. Lorum ipsum and so on. I am a long long multline string passing several levels of text. Lorum ipsum and so on]
 | 
			
		||||
 | 
			
		||||
    </pre>
 | 
			
		||||
    <pre class="mermaid">
 | 
			
		||||
      ---
 | 
			
		||||
      config:
 | 
			
		||||
        layout: elk
 | 
			
		||||
      ---
 | 
			
		||||
      mindmap
 | 
			
		||||
      root((mindmap))
 | 
			
		||||
        A
 | 
			
		||||
          a
 | 
			
		||||
            apa[I am a long long multline string passing several levels of text. Lorum ipsum and so on. I am a long long multline string passing several levels of text. Lorum ipsum and so on. I am a long long multline string passing several levels of text. Lorum ipsum and so on. I am a long long multline string passing several levels of text. Lorum ipsum and so on]
 | 
			
		||||
            apa2[I am a long long multline string passing several levels of text. Lorum ipsum and so on. I am a long long multline string passing several levels of text. Lorum ipsum and so on. I am a long long multline string passing several levels of text. Lorum ipsum and so on. I am a long long multline string passing several levels of text. Lorum ipsum and so on]
 | 
			
		||||
          b
 | 
			
		||||
          c
 | 
			
		||||
          d
 | 
			
		||||
        B
 | 
			
		||||
            apa3[I am a long long multline string passing several levels of text. Lorum ipsum and so on. I am a long long multline string passing several levels of text. Lorum ipsum and so on. I am a long long multline string passing several levels of text. Lorum ipsum and so on. I am a long long multline string passing several levels of text. Lorum ipsum and so on]
 | 
			
		||||
        D
 | 
			
		||||
          apa5[I am a long long multline string passing several levels of text. Lorum ipsum and so on. I am a long long multline string passing several levels of text. Lorum ipsum and so on. I am a long long multline string passing several levels of text. Lorum ipsum and so on. I am a long long multline string passing several levels of text. Lorum ipsum and so on]
 | 
			
		||||
          apa4[I am a long long multline string passing several levels of text. Lorum ipsum and so on. I am a long long multline string passing several levels of text. Lorum ipsum and so on. I am a long long multline string passing several levels of text. Lorum ipsum and so on. I am a long long multline string passing several levels of text. Lorum ipsum and so on]
 | 
			
		||||
 | 
			
		||||
    </pre>
 | 
			
		||||
    <pre class="mermaid">
 | 
			
		||||
      ---
 | 
			
		||||
      config:
 | 
			
		||||
        layout: cose-bilkent
 | 
			
		||||
      ---
 | 
			
		||||
      mindmap
 | 
			
		||||
      root((mindmap))
 | 
			
		||||
        A
 | 
			
		||||
          a
 | 
			
		||||
            apa[I am a long long multline string passing several levels of text. Lorum ipsum and so on. I am a long long multline string passing several levels of text. Lorum ipsum and so on. I am a long long multline string passing several levels of text. Lorum ipsum and so on. I am a long long multline string passing several levels of text. Lorum ipsum and so on]
 | 
			
		||||
            apa2[I am a long long multline string passing several levels of text. Lorum ipsum and so on. I am a long long multline string passing several levels of text. Lorum ipsum and so on. I am a long long multline string passing several levels of text. Lorum ipsum and so on. I am a long long multline string passing several levels of text. Lorum ipsum and so on]
 | 
			
		||||
          b
 | 
			
		||||
          c
 | 
			
		||||
          d
 | 
			
		||||
        B
 | 
			
		||||
            apa3[I am a long long multline string passing several levels of text. Lorum ipsum and so on. I am a long long multline string passing several levels of text. Lorum ipsum and so on. I am a long long multline string passing several levels of text. Lorum ipsum and so on. I am a long long multline string passing several levels of text. Lorum ipsum and so on]
 | 
			
		||||
        D
 | 
			
		||||
          apa5[I am a long long multline string passing several levels of text. Lorum ipsum and so on. I am a long long multline string passing several levels of text. Lorum ipsum and so on. I am a long long multline string passing several levels of text. Lorum ipsum and so on. I am a long long multline string passing several levels of text. Lorum ipsum and so on]
 | 
			
		||||
          apa4[I am a long long multline string passing several levels of text. Lorum ipsum and so on. I am a long long multline string passing several levels of text. Lorum ipsum and so on. I am a long long multline string passing several levels of text. Lorum ipsum and so on. I am a long long multline string passing several levels of text. Lorum ipsum and so on]
 | 
			
		||||
 | 
			
		||||
    </pre>
 | 
			
		||||
 | 
			
		||||
    <pre class="mermaid">
 | 
			
		||||
 ---
 | 
			
		||||
      config:
 | 
			
		||||
        layout: tidy-tree
 | 
			
		||||
      ---
 | 
			
		||||
      mindmap
 | 
			
		||||
      ((This is a mindmap))
 | 
			
		||||
        child1
 | 
			
		||||
         grandchild 1
 | 
			
		||||
         grandchild 2
 | 
			
		||||
        child2
 | 
			
		||||
         grandchild 3
 | 
			
		||||
         grandchild 4
 | 
			
		||||
        child3
 | 
			
		||||
         grandchild 5
 | 
			
		||||
         grandchild 6
 | 
			
		||||
      
 | 
			
		||||
    </pre>
 | 
			
		||||
 | 
			
		||||
    <pre class="mermaid">
 | 
			
		||||
 ---
 | 
			
		||||
      config:
 | 
			
		||||
        layout: dagre
 | 
			
		||||
      ---
 | 
			
		||||
      mindmap
 | 
			
		||||
      ((This is a mindmap))
 | 
			
		||||
        child1
 | 
			
		||||
         grandchild 1
 | 
			
		||||
         grandchild 2
 | 
			
		||||
        child2
 | 
			
		||||
         grandchild 3
 | 
			
		||||
         grandchild 4
 | 
			
		||||
        child3
 | 
			
		||||
         grandchild 5
 | 
			
		||||
         grandchild 6
 | 
			
		||||
      
 | 
			
		||||
    </pre>
 | 
			
		||||
 | 
			
		||||
    <pre class="mermaid">
 | 
			
		||||
 ---
 | 
			
		||||
      config:
 | 
			
		||||
        layout: elk
 | 
			
		||||
      ---
 | 
			
		||||
      mindmap
 | 
			
		||||
      ((This is a mindmap))
 | 
			
		||||
        child1
 | 
			
		||||
         grandchild 1
 | 
			
		||||
         grandchild 2
 | 
			
		||||
        child2
 | 
			
		||||
         grandchild 3
 | 
			
		||||
         grandchild 4
 | 
			
		||||
        child3
 | 
			
		||||
         grandchild 5
 | 
			
		||||
         grandchild 6
 | 
			
		||||
      
 | 
			
		||||
    </pre>
 | 
			
		||||
 | 
			
		||||
    <pre class="mermaid">
 | 
			
		||||
 ---
 | 
			
		||||
      config:
 | 
			
		||||
        layout: cose-bilkent
 | 
			
		||||
      ---
 | 
			
		||||
      mindmap
 | 
			
		||||
      ((This is a mindmap))
 | 
			
		||||
        child1
 | 
			
		||||
         grandchild 1
 | 
			
		||||
         grandchild 2
 | 
			
		||||
        child2
 | 
			
		||||
         grandchild 3
 | 
			
		||||
         grandchild 4
 | 
			
		||||
        child3
 | 
			
		||||
         grandchild 5
 | 
			
		||||
         grandchild 6
 | 
			
		||||
      
 | 
			
		||||
    </pre>
 | 
			
		||||
 | 
			
		||||
    <hr />
 | 
			
		||||
    <script type="module">
 | 
			
		||||
      import mermaid from '/mermaid.esm.mjs';
 | 
			
		||||
      mermaid.initialize({
 | 
			
		||||
        theme: 'default',
 | 
			
		||||
        logLevel: 3,
 | 
			
		||||
        securityLevel: 'loose',
 | 
			
		||||
      });
 | 
			
		||||
    </script>
 | 
			
		||||
  </body>
 | 
			
		||||
</html>
 | 
			
		||||
@@ -10,7 +10,7 @@
 | 
			
		||||
 | 
			
		||||
# Interface: LayoutData
 | 
			
		||||
 | 
			
		||||
Defined in: [packages/mermaid/src/rendering-util/types.ts:145](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/rendering-util/types.ts#L145)
 | 
			
		||||
Defined in: [packages/mermaid/src/rendering-util/types.ts:170](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/rendering-util/types.ts#L170)
 | 
			
		||||
 | 
			
		||||
## Indexable
 | 
			
		||||
 | 
			
		||||
@@ -22,7 +22,7 @@ Defined in: [packages/mermaid/src/rendering-util/types.ts:145](https://github.co
 | 
			
		||||
 | 
			
		||||
> **config**: [`MermaidConfig`](MermaidConfig.md)
 | 
			
		||||
 | 
			
		||||
Defined in: [packages/mermaid/src/rendering-util/types.ts:148](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/rendering-util/types.ts#L148)
 | 
			
		||||
Defined in: [packages/mermaid/src/rendering-util/types.ts:173](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/rendering-util/types.ts#L173)
 | 
			
		||||
 | 
			
		||||
---
 | 
			
		||||
 | 
			
		||||
@@ -30,7 +30,7 @@ Defined in: [packages/mermaid/src/rendering-util/types.ts:148](https://github.co
 | 
			
		||||
 | 
			
		||||
> **edges**: `Edge`\[]
 | 
			
		||||
 | 
			
		||||
Defined in: [packages/mermaid/src/rendering-util/types.ts:147](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/rendering-util/types.ts#L147)
 | 
			
		||||
Defined in: [packages/mermaid/src/rendering-util/types.ts:172](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/rendering-util/types.ts#L172)
 | 
			
		||||
 | 
			
		||||
---
 | 
			
		||||
 | 
			
		||||
@@ -38,4 +38,4 @@ Defined in: [packages/mermaid/src/rendering-util/types.ts:147](https://github.co
 | 
			
		||||
 | 
			
		||||
> **nodes**: `Node`\[]
 | 
			
		||||
 | 
			
		||||
Defined in: [packages/mermaid/src/rendering-util/types.ts:146](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/rendering-util/types.ts#L146)
 | 
			
		||||
Defined in: [packages/mermaid/src/rendering-util/types.ts:171](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/rendering-util/types.ts#L171)
 | 
			
		||||
 
 | 
			
		||||
@@ -10,7 +10,7 @@
 | 
			
		||||
 | 
			
		||||
# Interface: LayoutLoaderDefinition
 | 
			
		||||
 | 
			
		||||
Defined in: [packages/mermaid/src/rendering-util/render.ts:21](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/rendering-util/render.ts#L21)
 | 
			
		||||
Defined in: [packages/mermaid/src/rendering-util/render.ts:24](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/rendering-util/render.ts#L24)
 | 
			
		||||
 | 
			
		||||
## Properties
 | 
			
		||||
 | 
			
		||||
@@ -18,7 +18,7 @@ Defined in: [packages/mermaid/src/rendering-util/render.ts:21](https://github.co
 | 
			
		||||
 | 
			
		||||
> `optional` **algorithm**: `string`
 | 
			
		||||
 | 
			
		||||
Defined in: [packages/mermaid/src/rendering-util/render.ts:24](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/rendering-util/render.ts#L24)
 | 
			
		||||
Defined in: [packages/mermaid/src/rendering-util/render.ts:27](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/rendering-util/render.ts#L27)
 | 
			
		||||
 | 
			
		||||
---
 | 
			
		||||
 | 
			
		||||
@@ -26,7 +26,7 @@ Defined in: [packages/mermaid/src/rendering-util/render.ts:24](https://github.co
 | 
			
		||||
 | 
			
		||||
> **loader**: `LayoutLoader`
 | 
			
		||||
 | 
			
		||||
Defined in: [packages/mermaid/src/rendering-util/render.ts:23](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/rendering-util/render.ts#L23)
 | 
			
		||||
Defined in: [packages/mermaid/src/rendering-util/render.ts:26](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/rendering-util/render.ts#L26)
 | 
			
		||||
 | 
			
		||||
---
 | 
			
		||||
 | 
			
		||||
@@ -34,4 +34,4 @@ Defined in: [packages/mermaid/src/rendering-util/render.ts:23](https://github.co
 | 
			
		||||
 | 
			
		||||
> **name**: `string`
 | 
			
		||||
 | 
			
		||||
Defined in: [packages/mermaid/src/rendering-util/render.ts:22](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/rendering-util/render.ts#L22)
 | 
			
		||||
Defined in: [packages/mermaid/src/rendering-util/render.ts:25](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/rendering-util/render.ts#L25)
 | 
			
		||||
 
 | 
			
		||||
@@ -10,7 +10,7 @@
 | 
			
		||||
 | 
			
		||||
# Interface: ParseOptions
 | 
			
		||||
 | 
			
		||||
Defined in: [packages/mermaid/src/types.ts:72](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/types.ts#L72)
 | 
			
		||||
Defined in: [packages/mermaid/src/types.ts:76](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/types.ts#L76)
 | 
			
		||||
 | 
			
		||||
## Properties
 | 
			
		||||
 | 
			
		||||
@@ -18,7 +18,7 @@ Defined in: [packages/mermaid/src/types.ts:72](https://github.com/mermaid-js/mer
 | 
			
		||||
 | 
			
		||||
> `optional` **suppressErrors**: `boolean`
 | 
			
		||||
 | 
			
		||||
Defined in: [packages/mermaid/src/types.ts:77](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/types.ts#L77)
 | 
			
		||||
Defined in: [packages/mermaid/src/types.ts:81](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/types.ts#L81)
 | 
			
		||||
 | 
			
		||||
If `true`, parse will return `false` instead of throwing error when the diagram is invalid.
 | 
			
		||||
The `parseError` function will not be called.
 | 
			
		||||
 
 | 
			
		||||
@@ -10,7 +10,7 @@
 | 
			
		||||
 | 
			
		||||
# Interface: ParseResult
 | 
			
		||||
 | 
			
		||||
Defined in: [packages/mermaid/src/types.ts:80](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/types.ts#L80)
 | 
			
		||||
Defined in: [packages/mermaid/src/types.ts:84](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/types.ts#L84)
 | 
			
		||||
 | 
			
		||||
## Properties
 | 
			
		||||
 | 
			
		||||
@@ -18,7 +18,7 @@ Defined in: [packages/mermaid/src/types.ts:80](https://github.com/mermaid-js/mer
 | 
			
		||||
 | 
			
		||||
> **config**: [`MermaidConfig`](MermaidConfig.md)
 | 
			
		||||
 | 
			
		||||
Defined in: [packages/mermaid/src/types.ts:88](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/types.ts#L88)
 | 
			
		||||
Defined in: [packages/mermaid/src/types.ts:92](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/types.ts#L92)
 | 
			
		||||
 | 
			
		||||
The config passed as YAML frontmatter or directives
 | 
			
		||||
 | 
			
		||||
@@ -28,6 +28,6 @@ The config passed as YAML frontmatter or directives
 | 
			
		||||
 | 
			
		||||
> **diagramType**: `string`
 | 
			
		||||
 | 
			
		||||
Defined in: [packages/mermaid/src/types.ts:84](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/types.ts#L84)
 | 
			
		||||
Defined in: [packages/mermaid/src/types.ts:88](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/types.ts#L88)
 | 
			
		||||
 | 
			
		||||
The diagram type, e.g. 'flowchart', 'sequence', etc.
 | 
			
		||||
 
 | 
			
		||||
@@ -10,7 +10,7 @@
 | 
			
		||||
 | 
			
		||||
# Interface: RenderOptions
 | 
			
		||||
 | 
			
		||||
Defined in: [packages/mermaid/src/rendering-util/render.ts:7](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/rendering-util/render.ts#L7)
 | 
			
		||||
Defined in: [packages/mermaid/src/rendering-util/render.ts:10](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/rendering-util/render.ts#L10)
 | 
			
		||||
 | 
			
		||||
## Properties
 | 
			
		||||
 | 
			
		||||
@@ -18,4 +18,4 @@ Defined in: [packages/mermaid/src/rendering-util/render.ts:7](https://github.com
 | 
			
		||||
 | 
			
		||||
> `optional` **algorithm**: `string`
 | 
			
		||||
 | 
			
		||||
Defined in: [packages/mermaid/src/rendering-util/render.ts:8](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/rendering-util/render.ts#L8)
 | 
			
		||||
Defined in: [packages/mermaid/src/rendering-util/render.ts:11](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/rendering-util/render.ts#L11)
 | 
			
		||||
 
 | 
			
		||||
@@ -10,7 +10,7 @@
 | 
			
		||||
 | 
			
		||||
# Interface: RenderResult
 | 
			
		||||
 | 
			
		||||
Defined in: [packages/mermaid/src/types.ts:98](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/types.ts#L98)
 | 
			
		||||
Defined in: [packages/mermaid/src/types.ts:102](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/types.ts#L102)
 | 
			
		||||
 | 
			
		||||
## Properties
 | 
			
		||||
 | 
			
		||||
@@ -18,7 +18,7 @@ Defined in: [packages/mermaid/src/types.ts:98](https://github.com/mermaid-js/mer
 | 
			
		||||
 | 
			
		||||
> `optional` **bindFunctions**: (`element`) => `void`
 | 
			
		||||
 | 
			
		||||
Defined in: [packages/mermaid/src/types.ts:116](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/types.ts#L116)
 | 
			
		||||
Defined in: [packages/mermaid/src/types.ts:120](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/types.ts#L120)
 | 
			
		||||
 | 
			
		||||
Bind function to be called after the svg has been inserted into the DOM.
 | 
			
		||||
This is necessary for adding event listeners to the elements in the svg.
 | 
			
		||||
@@ -45,7 +45,7 @@ bindFunctions?.(div); // To call bindFunctions only if it's present.
 | 
			
		||||
 | 
			
		||||
> **diagramType**: `string`
 | 
			
		||||
 | 
			
		||||
Defined in: [packages/mermaid/src/types.ts:106](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/types.ts#L106)
 | 
			
		||||
Defined in: [packages/mermaid/src/types.ts:110](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/types.ts#L110)
 | 
			
		||||
 | 
			
		||||
The diagram type, e.g. 'flowchart', 'sequence', etc.
 | 
			
		||||
 | 
			
		||||
@@ -55,6 +55,6 @@ The diagram type, e.g. 'flowchart', 'sequence', etc.
 | 
			
		||||
 | 
			
		||||
> **svg**: `string`
 | 
			
		||||
 | 
			
		||||
Defined in: [packages/mermaid/src/types.ts:102](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/types.ts#L102)
 | 
			
		||||
Defined in: [packages/mermaid/src/types.ts:106](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/types.ts#L106)
 | 
			
		||||
 | 
			
		||||
The svg code for the rendered graph.
 | 
			
		||||
 
 | 
			
		||||
@@ -326,7 +326,9 @@ Below is a comprehensive list of the newly introduced shapes and their correspon
 | 
			
		||||
 | 
			
		||||
| **Semantic Name**                 | **Shape Name**         | **Short Name** | **Description**                | **Alias Supported**                                              |
 | 
			
		||||
| --------------------------------- | ---------------------- | -------------- | ------------------------------ | ---------------------------------------------------------------- |
 | 
			
		||||
| Bang                              | Bang                   | `bang`         | Bang                           | `bang`                                                           |
 | 
			
		||||
| Card                              | Notched Rectangle      | `notch-rect`   | Represents a card              | `card`, `notched-rectangle`                                      |
 | 
			
		||||
| Cloud                             | Cloud                  | `cloud`        | cloud                          | `cloud`                                                          |
 | 
			
		||||
| Collate                           | Hourglass              | `hourglass`    | Represents a collate operation | `collate`, `hourglass`                                           |
 | 
			
		||||
| Com Link                          | Lightning Bolt         | `bolt`         | Communication link             | `com-link`, `lightning-bolt`                                     |
 | 
			
		||||
| Comment                           | Curly Brace            | `brace`        | Adds a comment                 | `brace-l`, `comment`                                             |
 | 
			
		||||
 
 | 
			
		||||
@@ -17,6 +17,7 @@ export default tseslint.config(
 | 
			
		||||
  ...tseslint.configs.stylisticTypeChecked,
 | 
			
		||||
  {
 | 
			
		||||
    ignores: [
 | 
			
		||||
      '**/*.d.ts',
 | 
			
		||||
      '**/dist/',
 | 
			
		||||
      '**/node_modules/',
 | 
			
		||||
      '.git/',
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										145
									
								
								instructions.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										145
									
								
								instructions.md
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,145 @@
 | 
			
		||||
Please help me to implement the tidy-tree algorithm.
 | 
			
		||||
 | 
			
		||||
- I have added a placeholder with the correct signature in `packages/mermaid/src/rendering-util/layout-algorithms/tidy-tree/`.
 | 
			
		||||
 | 
			
		||||
Replace the cytoscape layout with one using tidy-tree.
 | 
			
		||||
 | 
			
		||||
This is the API for tidy-tree:
 | 
			
		||||
 | 
			
		||||
```
 | 
			
		||||
# non-layered-tidy-tree-layout
 | 
			
		||||
 | 
			
		||||
Draw non-layered tidy trees in linear time.
 | 
			
		||||
 | 
			
		||||
> This a JavaScript port from the project [cwi-swat/non-layered-tidy-trees](https://github.com/cwi-swat/non-layered-tidy-trees), which is written in Java. The algorithm used in that project is from the paper by _A.J. van der Ploeg_, [Drawing Non-layered Tidy Trees in Linear Time](http://oai.cwi.nl/oai/asset/21856/21856B.pdf). There is another JavaScript port from that project [d3-flextree](https://github.com/Klortho/d3-flextree), which depends on _d3-hierarchy_. This project is dependency free.
 | 
			
		||||
 | 
			
		||||
## Getting started
 | 
			
		||||
 | 
			
		||||
### Installation
 | 
			
		||||
 | 
			
		||||
```
 | 
			
		||||
 | 
			
		||||
npm install non-layered-tidy-tree-layout
 | 
			
		||||
 | 
			
		||||
```
 | 
			
		||||
 | 
			
		||||
Or
 | 
			
		||||
 | 
			
		||||
```
 | 
			
		||||
 | 
			
		||||
yarn add non-layered-tidy-tree-layout
 | 
			
		||||
 | 
			
		||||
````
 | 
			
		||||
 | 
			
		||||
There's also a built verison: `dist/non-layered-tidy-tree-layout.js` for use with browser `<script>` tag, or as a Javascript module.
 | 
			
		||||
 | 
			
		||||
### Usage
 | 
			
		||||
 | 
			
		||||
```js
 | 
			
		||||
import { BoundingBox, Layout } from 'non-layered-tidy-tree-layout'
 | 
			
		||||
 | 
			
		||||
// BoundingBox(gap, bottomPadding)
 | 
			
		||||
const bb = new BoundingBox(10, 20)
 | 
			
		||||
const layout = new Layout(bb)
 | 
			
		||||
const treeData = {
 | 
			
		||||
  id: 0,
 | 
			
		||||
  width: 40,
 | 
			
		||||
  height: 40,
 | 
			
		||||
  children: [
 | 
			
		||||
    {
 | 
			
		||||
      id: 1,
 | 
			
		||||
      width: 40,
 | 
			
		||||
      height: 40,
 | 
			
		||||
      children: [{ id: 6, width: 400, height: 40 }]
 | 
			
		||||
    },
 | 
			
		||||
    { id: 2, width: 40, height: 40 },
 | 
			
		||||
    { id: 3, width: 40, height: 40 },
 | 
			
		||||
    { id: 4, width: 40, height: 40 },
 | 
			
		||||
    { id: 5, width: 40, height: 80 }
 | 
			
		||||
  ]
 | 
			
		||||
}
 | 
			
		||||
const { result, boundingBox } = layout.layout(treeData)
 | 
			
		||||
 | 
			
		||||
// result:
 | 
			
		||||
// {
 | 
			
		||||
//   id: 0,
 | 
			
		||||
//   x: 300,
 | 
			
		||||
//   y: 0,
 | 
			
		||||
//   width: 40,
 | 
			
		||||
//   height: 40,
 | 
			
		||||
//   children: [
 | 
			
		||||
//     {
 | 
			
		||||
//       id: 1,
 | 
			
		||||
//       x: 185,
 | 
			
		||||
//       y: 60,
 | 
			
		||||
//       width: 40,
 | 
			
		||||
//       height: 40,
 | 
			
		||||
//       children: [
 | 
			
		||||
//         { id: 6, x: 5, y: 120, width: 400, height: 40 }
 | 
			
		||||
//       ]
 | 
			
		||||
//     },
 | 
			
		||||
//     { id: 2, x: 242.5, y: 60, width: 40, height: 40 },
 | 
			
		||||
//     { id: 3, x: 300, y: 60, width: 40, height: 40 },
 | 
			
		||||
//     { id: 4, x: 357.5, y: 60, width: 40, height: 40 },
 | 
			
		||||
//     { id: 5, x: 415, y: 60, width: 40, height: 80 }
 | 
			
		||||
//   ]
 | 
			
		||||
// }
 | 
			
		||||
//
 | 
			
		||||
// boundingBox:
 | 
			
		||||
// {
 | 
			
		||||
//   left: 5,
 | 
			
		||||
//   right: 455,
 | 
			
		||||
//   top: 0,
 | 
			
		||||
//   bottom: 160
 | 
			
		||||
// }
 | 
			
		||||
````
 | 
			
		||||
 | 
			
		||||
The method `Layout.layout` modifies `treeData` inplace. It returns an object like `{ result: treeData, boundingBox: {left: num, right: num, top: num, bottom: num} }`. `result` is the same object `treeData` with calculated coordinates, `boundingBox` are the coordinates for the whole tree:
 | 
			
		||||
 | 
			
		||||

 | 
			
		||||
 | 
			
		||||
The red dashed lines are the bounding boxes for each node. `Layout.layout()` produces coordinates to draw nodes, which are the grey boxes with black border.
 | 
			
		||||
 | 
			
		||||
The library also provides a class `Tree` and a method `layout`.
 | 
			
		||||
 | 
			
		||||
```js
 | 
			
		||||
/**
 | 
			
		||||
 * Constructor for Tree.
 | 
			
		||||
 * @param {number} width - width of bounding box
 | 
			
		||||
 * @param {number} height - height of bounding box
 | 
			
		||||
 * @param {number} y - veritcal coordinate of bounding box
 | 
			
		||||
 * @param {array} children - a list of Tree instances
 | 
			
		||||
 */
 | 
			
		||||
new Tree(width, height, y, children);
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Calculate x, y coordindates and assign them to tree.
 | 
			
		||||
 * @param {Object} tree - a Tree object
 | 
			
		||||
 */
 | 
			
		||||
layout(tree);
 | 
			
		||||
```
 | 
			
		||||
 | 
			
		||||
In case your data structure are not the same as provided by the example above, you can refer to `src/helpers.js` to implement a `Layout` class that converts your data to a `Tree`, then call `layout` to calculate the coordinates for drawing.
 | 
			
		||||
 | 
			
		||||
## License
 | 
			
		||||
 | 
			
		||||
[MIT](./LICENSE)
 | 
			
		||||
 | 
			
		||||
## Changelog
 | 
			
		||||
 | 
			
		||||
### [2.0.1]
 | 
			
		||||
 | 
			
		||||
- Fixed bounding box calculation in `Layout.getSize` and `Layout.assignLayout` and `Layout.layout`
 | 
			
		||||
 | 
			
		||||
### [2.0.0]
 | 
			
		||||
 | 
			
		||||
- Added `Layout.layout`
 | 
			
		||||
- Removed `Layout.layoutTreeData`
 | 
			
		||||
 | 
			
		||||
### [1.0.0]
 | 
			
		||||
 | 
			
		||||
- Added `Layout`, `BoundingBox`, `layout`, `Tree`
 | 
			
		||||
 | 
			
		||||
```
 | 
			
		||||
 | 
			
		||||
```
 | 
			
		||||
@@ -2,7 +2,7 @@
 | 
			
		||||
 | 
			
		||||
This package provides a layout engine for Mermaid based on the [ELK](https://www.eclipse.org/elk/) layout engine.
 | 
			
		||||
 | 
			
		||||
> [!NOTE]  
 | 
			
		||||
> [!NOTE]
 | 
			
		||||
> The ELK Layout engine will not be available in all providers that support mermaid by default.
 | 
			
		||||
> The websites will have to install the `@mermaid-js/layout-elk` package to use the ELK layout engine.
 | 
			
		||||
 | 
			
		||||
@@ -69,4 +69,4 @@ mermaid.registerLayoutLoaders(elkLayouts);
 | 
			
		||||
- `elk.mrtree`: Multi-root tree layout
 | 
			
		||||
- `elk.sporeOverlap`: Spore overlap layout
 | 
			
		||||
 | 
			
		||||
<!-- TODO: Add images for these layouts, as GitHub doesn't support natively -->
 | 
			
		||||
<!-- TODO: Add images for these layouts, as GitHub doesn't support natively. -->
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										9
									
								
								packages/mermaid-layout-elk/src/find-common-ancestor.d.ts
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								packages/mermaid-layout-elk/src/find-common-ancestor.d.ts
									
									
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,9 @@
 | 
			
		||||
export interface TreeData {
 | 
			
		||||
  parentById: Record<string, string>;
 | 
			
		||||
  childrenById: Record<string, string[]>;
 | 
			
		||||
}
 | 
			
		||||
export declare const findCommonAncestor: (
 | 
			
		||||
  id1: string,
 | 
			
		||||
  id2: string,
 | 
			
		||||
  { parentById }: TreeData
 | 
			
		||||
) => string;
 | 
			
		||||
							
								
								
									
										19
									
								
								packages/mermaid-layout-elk/src/render.d.ts
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										19
									
								
								packages/mermaid-layout-elk/src/render.d.ts
									
									
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,19 @@
 | 
			
		||||
import type { InternalHelpers, LayoutData, RenderOptions, SVG } from 'mermaid';
 | 
			
		||||
export declare const render: (
 | 
			
		||||
  data4Layout: LayoutData,
 | 
			
		||||
  svg: SVG,
 | 
			
		||||
  {
 | 
			
		||||
    common,
 | 
			
		||||
    getConfig,
 | 
			
		||||
    insertCluster,
 | 
			
		||||
    insertEdge,
 | 
			
		||||
    insertEdgeLabel,
 | 
			
		||||
    insertMarkers,
 | 
			
		||||
    insertNode,
 | 
			
		||||
    interpolateToCurve,
 | 
			
		||||
    labelHelper,
 | 
			
		||||
    log,
 | 
			
		||||
    positionEdgeLabel,
 | 
			
		||||
  }: InternalHelpers,
 | 
			
		||||
  { algorithm }: RenderOptions
 | 
			
		||||
) => Promise<void>;
 | 
			
		||||
@@ -1,10 +1,23 @@
 | 
			
		||||
import type {
 | 
			
		||||
  InternalHelpers,
 | 
			
		||||
  LayoutData,
 | 
			
		||||
  RenderOptions,
 | 
			
		||||
  SVG,
 | 
			
		||||
  SVGGroup,
 | 
			
		||||
} from '@mermaid-chart/mermaid';
 | 
			
		||||
// @ts-ignore TODO: Investigate D3 issue
 | 
			
		||||
import { curveLinear } from 'd3';
 | 
			
		||||
import ELK from 'elkjs/lib/elk.bundled.js';
 | 
			
		||||
import type { InternalHelpers, LayoutData, RenderOptions, SVG, SVGGroup } from 'mermaid';
 | 
			
		||||
import { type TreeData, findCommonAncestor } from './find-common-ancestor.js';
 | 
			
		||||
 | 
			
		||||
type Node = LayoutData['nodes'][number];
 | 
			
		||||
 | 
			
		||||
// Minimal structural type to avoid depending on d3 Selection typings
 | 
			
		||||
interface D3Selection<T extends Element> {
 | 
			
		||||
  node(): T | null;
 | 
			
		||||
  attr(name: string, value: string): D3Selection<T>;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
interface LabelData {
 | 
			
		||||
  width: number;
 | 
			
		||||
  height: number;
 | 
			
		||||
@@ -13,9 +26,9 @@ interface LabelData {
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
interface NodeWithVertex extends Omit<Node, 'domId'> {
 | 
			
		||||
  children?: unknown[];
 | 
			
		||||
  children?: LayoutData['nodes'];
 | 
			
		||||
  labelData?: LabelData;
 | 
			
		||||
  domId?: Node['domId'] | SVGGroup | d3.Selection<SVGAElement, unknown, Element | null, unknown>;
 | 
			
		||||
  domId?: D3Selection<SVGAElement | SVGGElement>;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export const render = async (
 | 
			
		||||
@@ -51,14 +64,13 @@ export const render = async (
 | 
			
		||||
 | 
			
		||||
    // Add the element to the DOM
 | 
			
		||||
    if (!node.isGroup) {
 | 
			
		||||
      const child: NodeWithVertex = {
 | 
			
		||||
        ...node,
 | 
			
		||||
      };
 | 
			
		||||
      const child = node as NodeWithVertex;
 | 
			
		||||
      graph.children.push(child);
 | 
			
		||||
      nodeDb[node.id] = child;
 | 
			
		||||
      nodeDb[node.id] = node;
 | 
			
		||||
 | 
			
		||||
      const childNodeEl = await insertNode(nodeEl, node, { config, dir: node.dir });
 | 
			
		||||
      const boundingBox = childNodeEl.node()!.getBBox();
 | 
			
		||||
      // Store the domId separately for rendering, not in the ELK graph
 | 
			
		||||
      child.domId = childNodeEl;
 | 
			
		||||
      child.width = boundingBox.width;
 | 
			
		||||
      child.height = boundingBox.height;
 | 
			
		||||
@@ -68,7 +80,9 @@ export const render = async (
 | 
			
		||||
        ...node,
 | 
			
		||||
        children: [],
 | 
			
		||||
      };
 | 
			
		||||
      // Let elk render with the copy
 | 
			
		||||
      graph.children.push(child);
 | 
			
		||||
      // Save the original containing the intersection function
 | 
			
		||||
      nodeDb[node.id] = child;
 | 
			
		||||
      await addVertices(nodeEl, nodeArr, child, node.id);
 | 
			
		||||
 | 
			
		||||
@@ -143,7 +157,7 @@ export const render = async (
 | 
			
		||||
            height: node.height,
 | 
			
		||||
          };
 | 
			
		||||
          if (node.isGroup) {
 | 
			
		||||
            log.debug('id abc88 subgraph = ', node.id, node.x, node.y, node.labelData);
 | 
			
		||||
            log.debug('Id abc88 subgraph = ', node.id, node.x, node.y, node.labelData);
 | 
			
		||||
            const subgraphEl = subgraphsEl.insert('g').attr('class', 'subgraph');
 | 
			
		||||
            // TODO use faster way of cloning
 | 
			
		||||
            const clusterNode = JSON.parse(JSON.stringify(node));
 | 
			
		||||
@@ -152,10 +166,10 @@ export const render = async (
 | 
			
		||||
            clusterNode.width = Math.max(clusterNode.width, node.labelData.width);
 | 
			
		||||
            await insertCluster(subgraphEl, clusterNode);
 | 
			
		||||
 | 
			
		||||
            log.debug('id (UIO)= ', node.id, node.width, node.shape, node.labels);
 | 
			
		||||
            log.debug('Id (UIO)= ', node.id, node.width, node.shape, node.labels);
 | 
			
		||||
          } else {
 | 
			
		||||
            log.info(
 | 
			
		||||
              'id NODE = ',
 | 
			
		||||
              'Id NODE = ',
 | 
			
		||||
              node.id,
 | 
			
		||||
              node.x,
 | 
			
		||||
              node.y,
 | 
			
		||||
@@ -197,25 +211,19 @@ export const render = async (
 | 
			
		||||
      });
 | 
			
		||||
    });
 | 
			
		||||
 | 
			
		||||
    subgraphs.forEach(function (subgraph: { id: string | number }) {
 | 
			
		||||
      const data: any = { id: subgraph.id };
 | 
			
		||||
      if (parentLookupDb.parentById[subgraph.id] !== undefined) {
 | 
			
		||||
        data.parent = parentLookupDb.parentById[subgraph.id];
 | 
			
		||||
      }
 | 
			
		||||
    });
 | 
			
		||||
    return parentLookupDb;
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  const getEdgeStartEndPoint = (edge: any) => {
 | 
			
		||||
    const source: any = edge.start;
 | 
			
		||||
    const target: any = edge.end;
 | 
			
		||||
    // edge.start and edge.end are IDs (string/number) in our layout data
 | 
			
		||||
    const sourceId: string | number = edge.start;
 | 
			
		||||
    const targetId: string | number = edge.end;
 | 
			
		||||
 | 
			
		||||
    // Save the original source and target
 | 
			
		||||
    const sourceId = source;
 | 
			
		||||
    const targetId = target;
 | 
			
		||||
    const source = sourceId;
 | 
			
		||||
    const target = targetId;
 | 
			
		||||
 | 
			
		||||
    const startNode = nodeDb[edge.start.id];
 | 
			
		||||
    const endNode = nodeDb[edge.end.id];
 | 
			
		||||
    const startNode = nodeDb[sourceId];
 | 
			
		||||
    const endNode = nodeDb[targetId];
 | 
			
		||||
 | 
			
		||||
    if (!startNode || !endNode) {
 | 
			
		||||
      return { source, target };
 | 
			
		||||
@@ -259,7 +267,6 @@ export const render = async (
 | 
			
		||||
    const edges = dataForLayout.edges;
 | 
			
		||||
    const labelsEl = svg.insert('g').attr('class', 'edgeLabels');
 | 
			
		||||
    const linkIdCnt: any = {};
 | 
			
		||||
    const dir = dataForLayout.direction || 'DOWN';
 | 
			
		||||
    let defaultStyle: string | undefined;
 | 
			
		||||
    let defaultLabelStyle: string | undefined;
 | 
			
		||||
 | 
			
		||||
@@ -289,7 +296,7 @@ export const render = async (
 | 
			
		||||
          linkIdCnt[linkIdBase]++;
 | 
			
		||||
          log.info('abc78 new entry', linkIdBase, linkIdCnt[linkIdBase]);
 | 
			
		||||
        }
 | 
			
		||||
        const linkId = linkIdBase + '_' + linkIdCnt[linkIdBase];
 | 
			
		||||
        const linkId = linkIdBase; // + '_' + linkIdCnt[linkIdBase];
 | 
			
		||||
        edge.id = linkId;
 | 
			
		||||
        log.info('abc78 new link id to be used is', linkIdBase, linkId, linkIdCnt[linkIdBase]);
 | 
			
		||||
        const linkNameStart = 'LS_' + edge.start;
 | 
			
		||||
@@ -358,8 +365,8 @@ export const render = async (
 | 
			
		||||
            break;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        edgeData.style = edgeData.style += style;
 | 
			
		||||
        edgeData.labelStyle = edgeData.labelStyle += labelStyle;
 | 
			
		||||
        edgeData.style += style;
 | 
			
		||||
        edgeData.labelStyle += labelStyle;
 | 
			
		||||
 | 
			
		||||
        const conf = getConfig();
 | 
			
		||||
        if (edge.interpolate !== undefined) {
 | 
			
		||||
@@ -396,13 +403,11 @@ export const render = async (
 | 
			
		||||
 | 
			
		||||
        // calculate start and end points of the edge, note that the source and target
 | 
			
		||||
        // can be modified for shapes that have ports
 | 
			
		||||
        // @ts-ignore TODO: fix this
 | 
			
		||||
        const { source, target, sourceId, targetId } = getEdgeStartEndPoint(edge, dir);
 | 
			
		||||
 | 
			
		||||
        const { source, target, sourceId, targetId } = getEdgeStartEndPoint(edge);
 | 
			
		||||
        log.debug('abc78 source and target', source, target);
 | 
			
		||||
        // Add the edge to the graph
 | 
			
		||||
        graph.edges.push({
 | 
			
		||||
          // @ts-ignore TODO: fix this
 | 
			
		||||
          id: 'e' + edge.start + edge.end,
 | 
			
		||||
          ...edge,
 | 
			
		||||
          sources: [source],
 | 
			
		||||
          targets: [target],
 | 
			
		||||
@@ -436,6 +441,7 @@ export const render = async (
 | 
			
		||||
      case 'RL':
 | 
			
		||||
        return 'LEFT';
 | 
			
		||||
      case 'TB':
 | 
			
		||||
      case 'TD': // TD is an alias for TB in Mermaid
 | 
			
		||||
        return 'DOWN';
 | 
			
		||||
      case 'BT':
 | 
			
		||||
        return 'UP';
 | 
			
		||||
@@ -459,155 +465,6 @@ export const render = async (
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  function intersectLine(
 | 
			
		||||
    p1: { y: number; x: number },
 | 
			
		||||
    p2: { y: number; x: number },
 | 
			
		||||
    q1: { x: any; y: any },
 | 
			
		||||
    q2: { x: any; y: any }
 | 
			
		||||
  ) {
 | 
			
		||||
    log.debug('UIO intersectLine', p1, p2, q1, q2);
 | 
			
		||||
    // Algorithm from J. Avro, (ed.) Graphics Gems, No 2, Morgan Kaufmann, 1994,
 | 
			
		||||
    // p7 and p473.
 | 
			
		||||
 | 
			
		||||
    // let a1, a2, b1, b2, c1, c2;
 | 
			
		||||
    // let r1, r2, r3, r4;
 | 
			
		||||
    // let denom, offset, num;
 | 
			
		||||
    // let x, y;
 | 
			
		||||
 | 
			
		||||
    // Compute a1, b1, c1, where line joining points 1 and 2 is F(x,y) = a1 x +
 | 
			
		||||
    // b1 y + c1 = 0.
 | 
			
		||||
    const a1 = p2.y - p1.y;
 | 
			
		||||
    const b1 = p1.x - p2.x;
 | 
			
		||||
    const c1 = p2.x * p1.y - p1.x * p2.y;
 | 
			
		||||
 | 
			
		||||
    // Compute r3 and r4.
 | 
			
		||||
    const r3 = a1 * q1.x + b1 * q1.y + c1;
 | 
			
		||||
    const r4 = a1 * q2.x + b1 * q2.y + c1;
 | 
			
		||||
 | 
			
		||||
    const epsilon = 1e-6;
 | 
			
		||||
 | 
			
		||||
    // Check signs of r3 and r4. If both point 3 and point 4 lie on
 | 
			
		||||
    // same side of line 1, the line segments do not intersect.
 | 
			
		||||
    if (r3 !== 0 && r4 !== 0 && sameSign(r3, r4)) {
 | 
			
		||||
      return /*DON'T_INTERSECT*/;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    // Compute a2, b2, c2 where line joining points 3 and 4 is G(x,y) = a2 x + b2 y + c2 = 0
 | 
			
		||||
    const a2 = q2.y - q1.y;
 | 
			
		||||
    const b2 = q1.x - q2.x;
 | 
			
		||||
    const c2 = q2.x * q1.y - q1.x * q2.y;
 | 
			
		||||
 | 
			
		||||
    // Compute r1 and r2
 | 
			
		||||
    const r1 = a2 * p1.x + b2 * p1.y + c2;
 | 
			
		||||
    const r2 = a2 * p2.x + b2 * p2.y + c2;
 | 
			
		||||
 | 
			
		||||
    // Check signs of r1 and r2. If both point 1 and point 2 lie
 | 
			
		||||
    // on same side of second line segment, the line segments do
 | 
			
		||||
    // not intersect.
 | 
			
		||||
    if (Math.abs(r1) < epsilon && Math.abs(r2) < epsilon && sameSign(r1, r2)) {
 | 
			
		||||
      return /*DON'T_INTERSECT*/;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    // Line segments intersect: compute intersection point.
 | 
			
		||||
    const denom = a1 * b2 - a2 * b1;
 | 
			
		||||
    if (denom === 0) {
 | 
			
		||||
      return /*COLLINEAR*/;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    const offset = Math.abs(denom / 2);
 | 
			
		||||
 | 
			
		||||
    // The denom/2 is to get rounding instead of truncating. It
 | 
			
		||||
    // is added or subtracted to the numerator, depending upon the
 | 
			
		||||
    // sign of the numerator.
 | 
			
		||||
    let num = b1 * c2 - b2 * c1;
 | 
			
		||||
    const x = num < 0 ? (num - offset) / denom : (num + offset) / denom;
 | 
			
		||||
 | 
			
		||||
    num = a2 * c1 - a1 * c2;
 | 
			
		||||
    const y = num < 0 ? (num - offset) / denom : (num + offset) / denom;
 | 
			
		||||
 | 
			
		||||
    return { x: x, y: y };
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  function sameSign(r1: number, r2: number) {
 | 
			
		||||
    return r1 * r2 > 0;
 | 
			
		||||
  }
 | 
			
		||||
  const diamondIntersection = (
 | 
			
		||||
    bounds: { x: any; y: any; width: any; height: any },
 | 
			
		||||
    outsidePoint: { x: number; y: number },
 | 
			
		||||
    insidePoint: any
 | 
			
		||||
  ) => {
 | 
			
		||||
    const x1 = bounds.x;
 | 
			
		||||
    const y1 = bounds.y;
 | 
			
		||||
 | 
			
		||||
    const w = bounds.width; //+ bounds.padding;
 | 
			
		||||
    const h = bounds.height; // + bounds.padding;
 | 
			
		||||
 | 
			
		||||
    const polyPoints = [
 | 
			
		||||
      { x: x1, y: y1 - h / 2 },
 | 
			
		||||
      { x: x1 + w / 2, y: y1 },
 | 
			
		||||
      { x: x1, y: y1 + h / 2 },
 | 
			
		||||
      { x: x1 - w / 2, y: y1 },
 | 
			
		||||
    ];
 | 
			
		||||
    log.debug(
 | 
			
		||||
      `APA16 diamondIntersection calc abc89:
 | 
			
		||||
  outsidePoint: ${JSON.stringify(outsidePoint)}
 | 
			
		||||
  insidePoint : ${JSON.stringify(insidePoint)}
 | 
			
		||||
  node-bounds       : x:${bounds.x} y:${bounds.y} w:${bounds.width} h:${bounds.height}`,
 | 
			
		||||
      JSON.stringify(polyPoints)
 | 
			
		||||
    );
 | 
			
		||||
 | 
			
		||||
    const intersections = [];
 | 
			
		||||
 | 
			
		||||
    let minX = Number.POSITIVE_INFINITY;
 | 
			
		||||
    let minY = Number.POSITIVE_INFINITY;
 | 
			
		||||
 | 
			
		||||
    polyPoints.forEach(function (entry) {
 | 
			
		||||
      minX = Math.min(minX, entry.x);
 | 
			
		||||
      minY = Math.min(minY, entry.y);
 | 
			
		||||
    });
 | 
			
		||||
 | 
			
		||||
    const left = x1 - w / 2 - minX;
 | 
			
		||||
    const top = y1 - h / 2 - minY;
 | 
			
		||||
 | 
			
		||||
    for (let i = 0; i < polyPoints.length; i++) {
 | 
			
		||||
      const p1 = polyPoints[i];
 | 
			
		||||
      const p2 = polyPoints[i < polyPoints.length - 1 ? i + 1 : 0];
 | 
			
		||||
      const intersect = intersectLine(
 | 
			
		||||
        bounds,
 | 
			
		||||
        outsidePoint,
 | 
			
		||||
        { x: left + p1.x, y: top + p1.y },
 | 
			
		||||
        { x: left + p2.x, y: top + p2.y }
 | 
			
		||||
      );
 | 
			
		||||
 | 
			
		||||
      if (intersect) {
 | 
			
		||||
        intersections.push(intersect);
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    if (!intersections.length) {
 | 
			
		||||
      return bounds;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    log.debug('UIO intersections', intersections);
 | 
			
		||||
 | 
			
		||||
    if (intersections.length > 1) {
 | 
			
		||||
      // More intersections, find the one nearest to edge end point
 | 
			
		||||
      intersections.sort(function (p, q) {
 | 
			
		||||
        const pdx = p.x - outsidePoint.x;
 | 
			
		||||
        const pdy = p.y - outsidePoint.y;
 | 
			
		||||
        const distp = Math.sqrt(pdx * pdx + pdy * pdy);
 | 
			
		||||
 | 
			
		||||
        const qdx = q.x - outsidePoint.x;
 | 
			
		||||
        const qdy = q.y - outsidePoint.y;
 | 
			
		||||
        const distq = Math.sqrt(qdx * qdx + qdy * qdy);
 | 
			
		||||
 | 
			
		||||
        return distp < distq ? -1 : distp === distq ? 0 : 1;
 | 
			
		||||
      });
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    return intersections[0];
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  const intersection = (
 | 
			
		||||
    node: { x: any; y: any; width: number; height: number },
 | 
			
		||||
    outsidePoint: { x: number; y: number },
 | 
			
		||||
@@ -653,7 +510,7 @@ export const render = async (
 | 
			
		||||
 | 
			
		||||
      return res;
 | 
			
		||||
    } else {
 | 
			
		||||
      // Intersection on sides of rect
 | 
			
		||||
      // Intersection onn sides of rect
 | 
			
		||||
      if (insidePoint.x < outsidePoint.x) {
 | 
			
		||||
        r = outsidePoint.x - w - x;
 | 
			
		||||
      } else {
 | 
			
		||||
@@ -696,62 +553,298 @@ export const render = async (
 | 
			
		||||
    }
 | 
			
		||||
    return false;
 | 
			
		||||
  };
 | 
			
		||||
  /**
 | 
			
		||||
   * This function will page a path and node where the last point(s) in the path is inside the node
 | 
			
		||||
   * and return an update path ending by the border of the node.
 | 
			
		||||
   */
 | 
			
		||||
  const cutPathAtIntersect = (
 | 
			
		||||
    _points: any[],
 | 
			
		||||
    bounds: { x: any; y: any; width: any; height: any; padding: any },
 | 
			
		||||
    isDiamond: boolean
 | 
			
		||||
  ) => {
 | 
			
		||||
    log.debug('APA18 cutPathAtIntersect Points:', _points, 'node:', bounds, 'isDiamond', isDiamond);
 | 
			
		||||
    const points: any[] = [];
 | 
			
		||||
    let lastPointOutside = _points[0];
 | 
			
		||||
    let isInside = false;
 | 
			
		||||
    _points.forEach((point: any) => {
 | 
			
		||||
      // check if point is inside the boundary rect
 | 
			
		||||
      if (!outsideNode(bounds, point) && !isInside) {
 | 
			
		||||
        // First point inside the rect found
 | 
			
		||||
        // Calc the intersection coord between the point and the last point outside the rect
 | 
			
		||||
        let inter;
 | 
			
		||||
 | 
			
		||||
        if (isDiamond) {
 | 
			
		||||
          const inter2 = diamondIntersection(bounds, lastPointOutside, point);
 | 
			
		||||
          const distance = Math.sqrt(
 | 
			
		||||
            (lastPointOutside.x - inter2.x) ** 2 + (lastPointOutside.y - inter2.y) ** 2
 | 
			
		||||
          );
 | 
			
		||||
          if (distance > 1) {
 | 
			
		||||
            inter = inter2;
 | 
			
		||||
          }
 | 
			
		||||
        }
 | 
			
		||||
        if (!inter) {
 | 
			
		||||
          inter = intersection(bounds, lastPointOutside, point);
 | 
			
		||||
        }
 | 
			
		||||
  const cutter2 = (startNode: any, endNode: any, _points: any[]) => {
 | 
			
		||||
    const startBounds = {
 | 
			
		||||
      x: startNode.offset.posX + startNode.width / 2,
 | 
			
		||||
      y: startNode.offset.posY + startNode.height / 2,
 | 
			
		||||
      width: startNode.width,
 | 
			
		||||
      height: startNode.height,
 | 
			
		||||
      padding: startNode.padding,
 | 
			
		||||
    };
 | 
			
		||||
    const endBounds = {
 | 
			
		||||
      x: endNode.offset.posX + endNode.width / 2,
 | 
			
		||||
      y: endNode.offset.posY + endNode.height / 2,
 | 
			
		||||
      width: endNode.width,
 | 
			
		||||
      height: endNode.height,
 | 
			
		||||
      padding: endNode.padding,
 | 
			
		||||
    };
 | 
			
		||||
 | 
			
		||||
        // Check case where the intersection is the same as the last point
 | 
			
		||||
        let pointPresent = false;
 | 
			
		||||
        points.forEach((p) => {
 | 
			
		||||
          pointPresent = pointPresent || (p.x === inter.x && p.y === inter.y);
 | 
			
		||||
        });
 | 
			
		||||
        // if (!pointPresent) {
 | 
			
		||||
        if (!points.some((e) => e.x === inter.x && e.y === inter.y)) {
 | 
			
		||||
          points.push(inter);
 | 
			
		||||
        } else {
 | 
			
		||||
          log.debug('abc88 no intersect', inter, points);
 | 
			
		||||
        }
 | 
			
		||||
        // points.push(inter);
 | 
			
		||||
        isInside = true;
 | 
			
		||||
      } else {
 | 
			
		||||
        // Outside
 | 
			
		||||
        log.debug('abc88 outside', point, lastPointOutside, points);
 | 
			
		||||
        lastPointOutside = point;
 | 
			
		||||
        // points.push(point);
 | 
			
		||||
        if (!isInside) {
 | 
			
		||||
          points.push(point);
 | 
			
		||||
    if (_points.length === 0) {
 | 
			
		||||
      return [];
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    // Copy the original points array
 | 
			
		||||
    const points = [..._points];
 | 
			
		||||
 | 
			
		||||
    // The first point is the center of sNode, the last point is the center of eNode
 | 
			
		||||
    const startCenter = points[0];
 | 
			
		||||
    const endCenter = points[points.length - 1];
 | 
			
		||||
 | 
			
		||||
    log.debug('UIO cutter2: startCenter:', startCenter);
 | 
			
		||||
    log.debug('UIO cutter2: endCenter:', endCenter);
 | 
			
		||||
 | 
			
		||||
    let firstOutsideStartIndex = -1;
 | 
			
		||||
    let lastOutsideEndIndex = -1;
 | 
			
		||||
 | 
			
		||||
    // Single iteration through the array
 | 
			
		||||
    for (const [i, point] of points.entries()) {
 | 
			
		||||
      // Check if this is the first point outside the start node
 | 
			
		||||
      if (firstOutsideStartIndex === -1 && outsideNode(startBounds, point)) {
 | 
			
		||||
        firstOutsideStartIndex = i;
 | 
			
		||||
        log.debug('UIO cutter2: First point outside start node at index', i, point);
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      // Check if this point is outside the end node (keep updating to find the last one)
 | 
			
		||||
      if (outsideNode(endBounds, point)) {
 | 
			
		||||
        lastOutsideEndIndex = i;
 | 
			
		||||
        log.debug('UIO cutter2: Point outside end node at index', i, point);
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    log.debug(
 | 
			
		||||
      'UIO cutter2: firstOutsideStartIndex:',
 | 
			
		||||
      firstOutsideStartIndex,
 | 
			
		||||
      'lastOutsideEndIndex:',
 | 
			
		||||
      lastOutsideEndIndex
 | 
			
		||||
    );
 | 
			
		||||
    log.debug('UIO cutter2: startBounds:', startBounds);
 | 
			
		||||
    log.debug('UIO cutter2: endBounds:', endBounds);
 | 
			
		||||
    log.debug('UIO cutter2: original points:', _points);
 | 
			
		||||
 | 
			
		||||
    // Calculate intersection with start node if we found a point outside it
 | 
			
		||||
    if (firstOutsideStartIndex !== -1) {
 | 
			
		||||
      const outsidePoint = points[firstOutsideStartIndex];
 | 
			
		||||
      let startIntersection;
 | 
			
		||||
 | 
			
		||||
      // Try using the node's intersect method first
 | 
			
		||||
      if (startNode.intersect) {
 | 
			
		||||
        startIntersection = startNode.intersect(outsidePoint);
 | 
			
		||||
 | 
			
		||||
        // Check if the intersection is valid (distance > 1)
 | 
			
		||||
        const distance = Math.sqrt(
 | 
			
		||||
          (startCenter.x - startIntersection.x) ** 2 + (startCenter.y - startIntersection.y) ** 2
 | 
			
		||||
        );
 | 
			
		||||
        if (distance <= 1) {
 | 
			
		||||
          startIntersection = null;
 | 
			
		||||
        }
 | 
			
		||||
      }
 | 
			
		||||
    });
 | 
			
		||||
 | 
			
		||||
      // Fallback to intersection function
 | 
			
		||||
      if (!startIntersection) {
 | 
			
		||||
        startIntersection = intersection(startBounds, startCenter, outsidePoint);
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      // Replace the first point with the intersection
 | 
			
		||||
      if (startIntersection) {
 | 
			
		||||
        // Check if the intersection is the same as any existing point
 | 
			
		||||
        const isDuplicate = points.some(
 | 
			
		||||
          (p, index) =>
 | 
			
		||||
            index > 0 &&
 | 
			
		||||
            Math.abs(p.x - startIntersection.x) < 0.1 &&
 | 
			
		||||
            Math.abs(p.y - startIntersection.y) < 0.1
 | 
			
		||||
        );
 | 
			
		||||
 | 
			
		||||
        if (isDuplicate) {
 | 
			
		||||
          log.debug(
 | 
			
		||||
            'UIO cutter2: Start intersection is duplicate of existing point, removing first point instead'
 | 
			
		||||
          );
 | 
			
		||||
          points.shift(); // Remove the first point instead of replacing it
 | 
			
		||||
        } else {
 | 
			
		||||
          log.debug(
 | 
			
		||||
            'UIO cutter2: Replacing first point',
 | 
			
		||||
            points[0],
 | 
			
		||||
            'with intersection',
 | 
			
		||||
            startIntersection
 | 
			
		||||
          );
 | 
			
		||||
          points[0] = startIntersection;
 | 
			
		||||
        }
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    // Calculate intersection with end node
 | 
			
		||||
    // Need to recalculate indices since we may have removed the first point
 | 
			
		||||
    let outsidePointForEnd = null;
 | 
			
		||||
    let outsideIndexForEnd = -1;
 | 
			
		||||
 | 
			
		||||
    // Find the last point that's outside the end node in the current points array
 | 
			
		||||
    for (let i = points.length - 1; i >= 0; i--) {
 | 
			
		||||
      if (outsideNode(endBounds, points[i])) {
 | 
			
		||||
        outsidePointForEnd = points[i];
 | 
			
		||||
        outsideIndexForEnd = i;
 | 
			
		||||
        log.debug('UIO cutter2: Found point outside end node at current index:', i, points[i]);
 | 
			
		||||
        break;
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    if (!outsidePointForEnd && points.length > 1) {
 | 
			
		||||
      // No points outside end node, try using the second-to-last point
 | 
			
		||||
      log.debug('UIO cutter2: No points outside end node, trying second-to-last point');
 | 
			
		||||
      outsidePointForEnd = points[points.length - 2];
 | 
			
		||||
      outsideIndexForEnd = points.length - 2;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    if (outsidePointForEnd) {
 | 
			
		||||
      // Check if the outside point is actually on the boundary (distance = 0 from intersection)
 | 
			
		||||
      // If so, we need to create a truly outside point
 | 
			
		||||
      let actualOutsidePoint = outsidePointForEnd;
 | 
			
		||||
 | 
			
		||||
      // Quick check: if the point is very close to the node boundary, move it further out
 | 
			
		||||
      const dx = Math.abs(outsidePointForEnd.x - endBounds.x);
 | 
			
		||||
      const dy = Math.abs(outsidePointForEnd.y - endBounds.y);
 | 
			
		||||
      const w = endBounds.width / 2;
 | 
			
		||||
      const h = endBounds.height / 2;
 | 
			
		||||
 | 
			
		||||
      log.debug('UIO cutter2: Checking if outside point is truly outside:', {
 | 
			
		||||
        outsidePoint: outsidePointForEnd,
 | 
			
		||||
        dx,
 | 
			
		||||
        dy,
 | 
			
		||||
        w,
 | 
			
		||||
        h,
 | 
			
		||||
        isOnBoundary: Math.abs(dx - w) < 1 || Math.abs(dy - h) < 1,
 | 
			
		||||
      });
 | 
			
		||||
 | 
			
		||||
      // If the point is on or very close to the boundary, move it further out
 | 
			
		||||
      if (Math.abs(dx - w) < 1 || Math.abs(dy - h) < 1) {
 | 
			
		||||
        log.debug('UIO cutter2: Outside point is on boundary, creating truly outside point');
 | 
			
		||||
        // Move the point further away from the node center
 | 
			
		||||
        const directionX = outsidePointForEnd.x - endBounds.x;
 | 
			
		||||
        const directionY = outsidePointForEnd.y - endBounds.y;
 | 
			
		||||
        const length = Math.sqrt(directionX * directionX + directionY * directionY);
 | 
			
		||||
 | 
			
		||||
        if (length > 0) {
 | 
			
		||||
          // Move the point 10 pixels further out in the same direction
 | 
			
		||||
          actualOutsidePoint = {
 | 
			
		||||
            x: endBounds.x + (directionX / length) * (length + 10),
 | 
			
		||||
            y: endBounds.y + (directionY / length) * (length + 10),
 | 
			
		||||
          };
 | 
			
		||||
          log.debug('UIO cutter2: Created truly outside point:', actualOutsidePoint);
 | 
			
		||||
        }
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      let endIntersection;
 | 
			
		||||
 | 
			
		||||
      // Try using the node's intersect method first
 | 
			
		||||
      if (endNode.intersect) {
 | 
			
		||||
        endIntersection = endNode.intersect(actualOutsidePoint);
 | 
			
		||||
        log.debug('UIO cutter2: endNode.intersect result:', endIntersection);
 | 
			
		||||
 | 
			
		||||
        // Check if the intersection is on the wrong side of the node
 | 
			
		||||
        const isWrongSide =
 | 
			
		||||
          (actualOutsidePoint.x < endBounds.x && endIntersection.x > endBounds.x) ||
 | 
			
		||||
          (actualOutsidePoint.x > endBounds.x && endIntersection.x < endBounds.x);
 | 
			
		||||
 | 
			
		||||
        if (isWrongSide) {
 | 
			
		||||
          log.debug('UIO cutter2: endNode.intersect returned wrong side, setting to null');
 | 
			
		||||
          endIntersection = null;
 | 
			
		||||
        } else {
 | 
			
		||||
          // Check if the intersection is valid (distance > 1)
 | 
			
		||||
          const distance = Math.sqrt(
 | 
			
		||||
            (actualOutsidePoint.x - endIntersection.x) ** 2 +
 | 
			
		||||
              (actualOutsidePoint.y - endIntersection.y) ** 2
 | 
			
		||||
          );
 | 
			
		||||
          log.debug('UIO cutter2: Distance from outside point to intersection:', distance);
 | 
			
		||||
          if (distance <= 1) {
 | 
			
		||||
            log.debug('UIO cutter2: endNode.intersect distance too small, setting to null');
 | 
			
		||||
            endIntersection = null;
 | 
			
		||||
          }
 | 
			
		||||
        }
 | 
			
		||||
      } else {
 | 
			
		||||
        log.debug('UIO cutter2: endNode.intersect method not available');
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      // Fallback to intersection function
 | 
			
		||||
      if (!endIntersection) {
 | 
			
		||||
        // Create a proper inside point that's on the correct side of the node
 | 
			
		||||
        // The inside point should be between the outside point and the far edge
 | 
			
		||||
        const insidePoint = {
 | 
			
		||||
          x:
 | 
			
		||||
            actualOutsidePoint.x < endBounds.x
 | 
			
		||||
              ? endBounds.x - endBounds.width / 4
 | 
			
		||||
              : endBounds.x + endBounds.width / 4,
 | 
			
		||||
          y: endCenter.y,
 | 
			
		||||
        };
 | 
			
		||||
 | 
			
		||||
        log.debug('UIO cutter2: Using fallback intersection function with:', {
 | 
			
		||||
          endBounds,
 | 
			
		||||
          actualOutsidePoint,
 | 
			
		||||
          insidePoint,
 | 
			
		||||
          endCenter,
 | 
			
		||||
        });
 | 
			
		||||
        endIntersection = intersection(endBounds, actualOutsidePoint, insidePoint);
 | 
			
		||||
        log.debug('UIO cutter2: Fallback intersection result:', endIntersection);
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      // Replace the last point with the intersection
 | 
			
		||||
      if (endIntersection) {
 | 
			
		||||
        // Check if the intersection is the same as any existing point
 | 
			
		||||
        const isDuplicate = points.some(
 | 
			
		||||
          (p, index) =>
 | 
			
		||||
            index < points.length - 1 &&
 | 
			
		||||
            Math.abs(p.x - endIntersection.x) < 0.1 &&
 | 
			
		||||
            Math.abs(p.y - endIntersection.y) < 0.1
 | 
			
		||||
        );
 | 
			
		||||
 | 
			
		||||
        if (isDuplicate) {
 | 
			
		||||
          log.debug(
 | 
			
		||||
            'UIO cutter2: End intersection is duplicate of existing point, removing last point instead'
 | 
			
		||||
          );
 | 
			
		||||
          points.pop(); // Remove the last point instead of replacing it
 | 
			
		||||
        } else {
 | 
			
		||||
          log.debug(
 | 
			
		||||
            'UIO cutter2: Replacing last point',
 | 
			
		||||
            points[points.length - 1],
 | 
			
		||||
            'with intersection',
 | 
			
		||||
            endIntersection,
 | 
			
		||||
            'using outside point at index',
 | 
			
		||||
            outsideIndexForEnd
 | 
			
		||||
          );
 | 
			
		||||
          points[points.length - 1] = endIntersection;
 | 
			
		||||
        }
 | 
			
		||||
      }
 | 
			
		||||
    } else {
 | 
			
		||||
      log.debug('UIO cutter2: No suitable outside point found for end node intersection');
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    // Final cleanup: Check if the last point is too close to the previous point
 | 
			
		||||
    if (points.length > 1) {
 | 
			
		||||
      const lastPoint = points[points.length - 1];
 | 
			
		||||
      const secondLastPoint = points[points.length - 2];
 | 
			
		||||
      const distance = Math.sqrt(
 | 
			
		||||
        (lastPoint.x - secondLastPoint.x) ** 2 + (lastPoint.y - secondLastPoint.y) ** 2
 | 
			
		||||
      );
 | 
			
		||||
 | 
			
		||||
      // If the distance is very small (less than 2 pixels), remove the last point
 | 
			
		||||
      if (distance < 2) {
 | 
			
		||||
        log.debug(
 | 
			
		||||
          'UIO cutter2: Last point too close to previous point, removing it. Distance:',
 | 
			
		||||
          distance
 | 
			
		||||
        );
 | 
			
		||||
        log.debug('UIO cutter2: Removing last point:', lastPoint, 'keeping:', secondLastPoint);
 | 
			
		||||
        points.pop();
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    log.debug('UIO cutter2: Final points:', points);
 | 
			
		||||
 | 
			
		||||
    // Debug: Check which side of the end node we're ending at
 | 
			
		||||
    if (points.length > 0) {
 | 
			
		||||
      const finalPoint = points[points.length - 1];
 | 
			
		||||
      const endNodeCenter = endBounds.x;
 | 
			
		||||
      const endNodeLeftEdge = endNodeCenter - endBounds.width / 2;
 | 
			
		||||
      const endNodeRightEdge = endNodeCenter + endBounds.width / 2;
 | 
			
		||||
 | 
			
		||||
      log.debug('UIO cutter2: End node analysis:', {
 | 
			
		||||
        finalPoint,
 | 
			
		||||
        endNodeCenter,
 | 
			
		||||
        endNodeLeftEdge,
 | 
			
		||||
        endNodeRightEdge,
 | 
			
		||||
        endingSide: finalPoint.x < endNodeCenter ? 'LEFT' : 'RIGHT',
 | 
			
		||||
        distanceFromLeftEdge: Math.abs(finalPoint.x - endNodeLeftEdge),
 | 
			
		||||
        distanceFromRightEdge: Math.abs(finalPoint.x - endNodeRightEdge),
 | 
			
		||||
      });
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    return points;
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
@@ -774,9 +867,11 @@ export const render = async (
 | 
			
		||||
      'nodePlacement.strategy': data4Layout.config.elk?.nodePlacementStrategy,
 | 
			
		||||
      'elk.layered.mergeEdges': data4Layout.config.elk?.mergeEdges,
 | 
			
		||||
      'elk.direction': 'DOWN',
 | 
			
		||||
      'spacing.baseValue': 35,
 | 
			
		||||
      'elk.layered.unnecessaryBendpoints': true,
 | 
			
		||||
      'elk.layered.cycleBreaking.strategy': data4Layout.config.elk?.cycleBreakingStrategy,
 | 
			
		||||
      'spacing.baseValue': 25,
 | 
			
		||||
      // 'elk.layered.unnecessaryBendpoints': true,
 | 
			
		||||
      // 'elk.layered.cycleBreaking.strategy': data4Layout.config.elk?.cycleBreakingStrategy,
 | 
			
		||||
      // 'elk.layered.cycleBreaking.strategy': 'GREEDY_MODEL_ORDER',
 | 
			
		||||
      // 'elk.layered.cycleBreaking.strategy': 'MODEL_ORDER',
 | 
			
		||||
      // 'spacing.nodeNode': 20,
 | 
			
		||||
      // 'spacing.nodeNodeBetweenLayers': 25,
 | 
			
		||||
      // 'spacing.edgeNode': 20,
 | 
			
		||||
@@ -784,23 +879,29 @@ export const render = async (
 | 
			
		||||
      // 'spacing.edgeEdge': 10,
 | 
			
		||||
      // 'spacing.edgeEdgeBetweenLayers': 20,
 | 
			
		||||
      // 'spacing.nodeSelfLoop': 20,
 | 
			
		||||
 | 
			
		||||
      // Tweaking options
 | 
			
		||||
      // 'elk.layered.nodePlacement.favorStraightEdges': true,
 | 
			
		||||
      // 'nodePlacement.feedbackEdges': true,
 | 
			
		||||
      // 'elk.layered.wrapping.multiEdge.improveCuts': true,
 | 
			
		||||
      // 'elk.layered.wrapping.multiEdge.improveWrappedEdges': true,
 | 
			
		||||
      // 'elk.layered.wrapping.strategy': 'MULTI_EDGE',
 | 
			
		||||
      // 'elk.layered.edgeRouting.selfLoopDistribution': 'EQUALLY',
 | 
			
		||||
      // 'elk.layered.mergeHierarchyEdges': true,
 | 
			
		||||
      // 'elk.layered.feedbackEdges': true,
 | 
			
		||||
      // 'elk.layered.crossingMinimization.semiInteractive': true,
 | 
			
		||||
      // 'elk.layered.edgeRouting.splines.sloppy.layerSpacingFactor': 1,
 | 
			
		||||
      // 'elk.layered.edgeRouting.polyline.slopedEdgeZoneWidth': 4.0,
 | 
			
		||||
      // 'elk.layered.wrapping.validify.strategy': 'LOOK_BACK',
 | 
			
		||||
      // 'elk.insideSelfLoops.activate': true,
 | 
			
		||||
      // 'elk.alg.layered.options.EdgeStraighteningStrategy': 'NONE',
 | 
			
		||||
      'nodePlacement.favorStraightEdges': true,
 | 
			
		||||
      'elk.layered.nodePlacement.favorStraightEdges': true,
 | 
			
		||||
      'nodePlacement.feedbackEdges': true,
 | 
			
		||||
      'elk.layered.wrapping.multiEdge.improveCuts': true,
 | 
			
		||||
      'elk.layered.wrapping.multiEdge.improveWrappedEdges': true,
 | 
			
		||||
      'elk.layered.wrapping.strategy': 'MULTI_EDGE',
 | 
			
		||||
      // 'elk.layered.wrapping.strategy': 'SINGLE_EDGE',
 | 
			
		||||
      'elk.layered.edgeRouting.selfLoopDistribution': 'EQUALLY',
 | 
			
		||||
      'elk.layered.mergeHierarchyEdges': true,
 | 
			
		||||
 | 
			
		||||
      'elk.layered.feedbackEdges': true,
 | 
			
		||||
      'elk.layered.crossingMinimization.semiInteractive': true,
 | 
			
		||||
      'elk.layered.edgeRouting.splines.sloppy.layerSpacingFactor': 1,
 | 
			
		||||
      'elk.layered.edgeRouting.polyline.slopedEdgeZoneWidth': 4.0,
 | 
			
		||||
      'elk.layered.wrapping.validify.strategy': 'LOOK_BACK',
 | 
			
		||||
      'elk.insideSelfLoops.activate': true,
 | 
			
		||||
      'elk.separateConnectedComponents': true,
 | 
			
		||||
      'elk.alg.layered.options.EdgeStraighteningStrategy': 'NONE',
 | 
			
		||||
      // 'elk.layered.considerModelOrder.strategy': 'NODES_AND_EDGES', // NODES_AND_EDGES
 | 
			
		||||
      // 'elk.layered.wrapping.cutting.strategy': 'ARD', // NODES_AND_EDGES
 | 
			
		||||
      'elk.layered.considerModelOrder.strategy': 'EDGES', // NODES_AND_EDGES
 | 
			
		||||
      'elk.layered.wrapping.cutting.strategy': 'ARD', // NODES_AND_EDGES
 | 
			
		||||
    },
 | 
			
		||||
    children: [],
 | 
			
		||||
    edges: [],
 | 
			
		||||
@@ -840,15 +941,16 @@ export const render = async (
 | 
			
		||||
 | 
			
		||||
    // Subgraph
 | 
			
		||||
    if (parentLookupDb.childrenById[node.id] !== undefined) {
 | 
			
		||||
      // Set label and adjust node width separately (avoid side effects in labels array)
 | 
			
		||||
      node.labels = [
 | 
			
		||||
        {
 | 
			
		||||
          text: node.label,
 | 
			
		||||
          width: node?.labelData?.width || 50,
 | 
			
		||||
          height: node?.labelData?.height || 50,
 | 
			
		||||
          width: node?.labelData?.width ?? 50,
 | 
			
		||||
          height: node?.labelData?.height ?? 50,
 | 
			
		||||
        },
 | 
			
		||||
        (node.width = node.width + 2 * node.padding),
 | 
			
		||||
        log.debug('UIO node label', node?.labelData?.width, node.padding),
 | 
			
		||||
      ];
 | 
			
		||||
      node.width = node.width + 2 * node.padding;
 | 
			
		||||
      log.debug('UIO node label', node?.labelData?.width, node.padding);
 | 
			
		||||
      node.layoutOptions = {
 | 
			
		||||
        'spacing.baseValue': 30,
 | 
			
		||||
        'nodeLabels.placement': '[H_CENTER V_TOP, INSIDE]',
 | 
			
		||||
@@ -869,11 +971,16 @@ export const render = async (
 | 
			
		||||
      delete node.height;
 | 
			
		||||
    }
 | 
			
		||||
  });
 | 
			
		||||
  elkGraph.edges.forEach((edge: any) => {
 | 
			
		||||
  log.debug('APA01 processing edges, count:', elkGraph.edges.length);
 | 
			
		||||
  elkGraph.edges.forEach((edge: any, index: number) => {
 | 
			
		||||
    log.debug('APA01 processing edge', index, ':', edge);
 | 
			
		||||
    const source = edge.sources[0];
 | 
			
		||||
    const target = edge.targets[0];
 | 
			
		||||
    log.debug('APA01 source:', source, 'target:', target);
 | 
			
		||||
    log.debug('APA01 nodeDb[source]:', nodeDb[source]);
 | 
			
		||||
    log.debug('APA01 nodeDb[target]:', nodeDb[target]);
 | 
			
		||||
 | 
			
		||||
    if (nodeDb[source].parentId !== nodeDb[target].parentId) {
 | 
			
		||||
    if (nodeDb[source] && nodeDb[target] && nodeDb[source].parentId !== nodeDb[target].parentId) {
 | 
			
		||||
      const ancestorId = findCommonAncestor(source, target, parentLookupDb);
 | 
			
		||||
      // an edge that breaks a subgraph has been identified, set configuration accordingly
 | 
			
		||||
      setIncludeChildrenPolicy(source, ancestorId);
 | 
			
		||||
@@ -881,7 +988,37 @@ export const render = async (
 | 
			
		||||
    }
 | 
			
		||||
  });
 | 
			
		||||
 | 
			
		||||
  const g = await elk.layout(elkGraph);
 | 
			
		||||
  log.debug('APA01 before');
 | 
			
		||||
  log.debug('APA01 elkGraph structure:', JSON.stringify(elkGraph, null, 2));
 | 
			
		||||
  log.debug('APA01 elkGraph.children length:', elkGraph.children?.length);
 | 
			
		||||
  log.debug('APA01 elkGraph.edges length:', elkGraph.edges?.length);
 | 
			
		||||
 | 
			
		||||
  // Validate that all edge references exist as nodes
 | 
			
		||||
  elkGraph.edges?.forEach((edge: any, index: number) => {
 | 
			
		||||
    log.debug(`APA01 validating edge ${index}:`, edge);
 | 
			
		||||
    if (edge.sources) {
 | 
			
		||||
      edge.sources.forEach((sourceId: any) => {
 | 
			
		||||
        const sourceExists = elkGraph.children?.some((child: any) => child.id === sourceId);
 | 
			
		||||
        log.debug(`APA01 source ${sourceId} exists:`, sourceExists);
 | 
			
		||||
      });
 | 
			
		||||
    }
 | 
			
		||||
    if (edge.targets) {
 | 
			
		||||
      edge.targets.forEach((targetId: any) => {
 | 
			
		||||
        const targetExists = elkGraph.children?.some((child: any) => child.id === targetId);
 | 
			
		||||
        log.debug(`APA01 target ${targetId} exists:`, targetExists);
 | 
			
		||||
      });
 | 
			
		||||
    }
 | 
			
		||||
  });
 | 
			
		||||
 | 
			
		||||
  let g;
 | 
			
		||||
  try {
 | 
			
		||||
    g = await elk.layout(elkGraph);
 | 
			
		||||
    log.debug('APA01 after - success');
 | 
			
		||||
    log.info('APA01 layout result:', JSON.stringify(g, null, 2));
 | 
			
		||||
  } catch (error) {
 | 
			
		||||
    log.error('APA01 ELK layout error:', error);
 | 
			
		||||
    throw error;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  // debugger;
 | 
			
		||||
  await drawNodes(0, 0, g.children, svg, subGraphsEl, 0);
 | 
			
		||||
@@ -941,7 +1078,7 @@ export const render = async (
 | 
			
		||||
          log.debug(
 | 
			
		||||
            'UIO width',
 | 
			
		||||
            startNode.id,
 | 
			
		||||
            startNode.with,
 | 
			
		||||
            startNode.width,
 | 
			
		||||
            'bbox.width=',
 | 
			
		||||
            bbox.width,
 | 
			
		||||
            'lw=',
 | 
			
		||||
@@ -961,7 +1098,7 @@ export const render = async (
 | 
			
		||||
          log.debug(
 | 
			
		||||
            'UIO width',
 | 
			
		||||
            startNode.id,
 | 
			
		||||
            startNode.with,
 | 
			
		||||
            startNode.width,
 | 
			
		||||
            bbox.width,
 | 
			
		||||
            'EW = ',
 | 
			
		||||
            ew,
 | 
			
		||||
@@ -969,43 +1106,26 @@ export const render = async (
 | 
			
		||||
            startNode.innerHTML
 | 
			
		||||
          );
 | 
			
		||||
        }
 | 
			
		||||
        if (startNode.shape === 'diamond' || startNode.shape === 'diam') {
 | 
			
		||||
        startNode.x = startNode.offset.posX + startNode.width / 2;
 | 
			
		||||
        startNode.y = startNode.offset.posY + startNode.height / 2;
 | 
			
		||||
        endNode.x = endNode.offset.posX + endNode.width / 2;
 | 
			
		||||
        endNode.y = endNode.offset.posY + endNode.height / 2;
 | 
			
		||||
        if (startNode.shape !== 'rect33') {
 | 
			
		||||
          edge.points.unshift({
 | 
			
		||||
            x: startNode.offset.posX + startNode.width / 2,
 | 
			
		||||
            y: startNode.offset.posY + startNode.height / 2,
 | 
			
		||||
            x: startNode.x,
 | 
			
		||||
            y: startNode.y,
 | 
			
		||||
          });
 | 
			
		||||
        }
 | 
			
		||||
        if (endNode.shape === 'diamond' || endNode.shape === 'diam') {
 | 
			
		||||
        if (endNode.shape !== 'rect33') {
 | 
			
		||||
          edge.points.push({
 | 
			
		||||
            x: endNode.offset.posX + endNode.width / 2,
 | 
			
		||||
            y: endNode.offset.posY + endNode.height / 2,
 | 
			
		||||
            x: endNode.x,
 | 
			
		||||
            y: endNode.y,
 | 
			
		||||
          });
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        edge.points = cutPathAtIntersect(
 | 
			
		||||
          edge.points.reverse(),
 | 
			
		||||
          {
 | 
			
		||||
            x: startNode.offset.posX + startNode.width / 2,
 | 
			
		||||
            y: startNode.offset.posY + startNode.height / 2,
 | 
			
		||||
            width: sw,
 | 
			
		||||
            height: startNode.height,
 | 
			
		||||
            padding: startNode.padding,
 | 
			
		||||
          },
 | 
			
		||||
          startNode.shape === 'diamond' || startNode.shape === 'diam'
 | 
			
		||||
        ).reverse();
 | 
			
		||||
 | 
			
		||||
        edge.points = cutPathAtIntersect(
 | 
			
		||||
          edge.points,
 | 
			
		||||
          {
 | 
			
		||||
            x: endNode.offset.posX + endNode.width / 2,
 | 
			
		||||
            y: endNode.offset.posY + endNode.height / 2,
 | 
			
		||||
            width: ew,
 | 
			
		||||
            height: endNode.height,
 | 
			
		||||
            padding: endNode.padding,
 | 
			
		||||
          },
 | 
			
		||||
          endNode.shape === 'diamond' || endNode.shape === 'diam'
 | 
			
		||||
        );
 | 
			
		||||
 | 
			
		||||
        log.debug('UIO cutter2: Points before cutter2:', edge.points);
 | 
			
		||||
        edge.points = cutter2(startNode, endNode, edge.points);
 | 
			
		||||
        log.debug('UIO cutter2: Points after cutter2:', edge.points);
 | 
			
		||||
        const paths = insertEdge(
 | 
			
		||||
          edgesEl,
 | 
			
		||||
          edge,
 | 
			
		||||
@@ -1013,7 +1133,8 @@ export const render = async (
 | 
			
		||||
          data4Layout.type,
 | 
			
		||||
          startNode,
 | 
			
		||||
          endNode,
 | 
			
		||||
          data4Layout.diagramId
 | 
			
		||||
          data4Layout.diagramId,
 | 
			
		||||
          true
 | 
			
		||||
        );
 | 
			
		||||
        log.info('APA12 edge points after insert', JSON.stringify(edge.points));
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -5,6 +5,6 @@
 | 
			
		||||
    "outDir": "./dist",
 | 
			
		||||
    "types": ["vitest/importMeta", "vitest/globals"]
 | 
			
		||||
  },
 | 
			
		||||
  "include": ["./src/**/*.ts"],
 | 
			
		||||
  "include": ["./src/**/*.ts", "./src/**/*.d.ts"],
 | 
			
		||||
  "typeRoots": ["./src/types"]
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										59
									
								
								packages/mermaid-layout-tidy-tree/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										59
									
								
								packages/mermaid-layout-tidy-tree/README.md
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,59 @@
 | 
			
		||||
# @mermaid-js/layout-tidy-tree
 | 
			
		||||
 | 
			
		||||
This package provides a bidirectional tidy tree layout engine for Mermaid based on the non-layered-tidy-tree-layout algorithm.
 | 
			
		||||
 | 
			
		||||
> [!NOTE]
 | 
			
		||||
> The Tidy Tree Layout engine will not be available in all providers that support mermaid by default.
 | 
			
		||||
> The websites will have to install the @mermaid-js/layout-tidy-tree package to use the Tidy Tree layout engine.
 | 
			
		||||
 | 
			
		||||
## Usage
 | 
			
		||||
 | 
			
		||||
```
 | 
			
		||||
---
 | 
			
		||||
config:
 | 
			
		||||
  layout: tidy-tree
 | 
			
		||||
---
 | 
			
		||||
mindmap
 | 
			
		||||
root((mindmap))
 | 
			
		||||
  A
 | 
			
		||||
  B
 | 
			
		||||
```
 | 
			
		||||
 | 
			
		||||
### With bundlers
 | 
			
		||||
 | 
			
		||||
```sh
 | 
			
		||||
npm install @mermaid-js/layout-tidy-tree
 | 
			
		||||
```
 | 
			
		||||
 | 
			
		||||
```ts
 | 
			
		||||
import mermaid from 'mermaid';
 | 
			
		||||
import tidyTreeLayouts from '@mermaid-js/layout-tidy-tree';
 | 
			
		||||
 | 
			
		||||
mermaid.registerLayoutLoaders(tidyTreeLayouts);
 | 
			
		||||
```
 | 
			
		||||
 | 
			
		||||
### With CDN
 | 
			
		||||
 | 
			
		||||
```html
 | 
			
		||||
<script type="module">
 | 
			
		||||
  import mermaid from 'https://cdn.jsdelivr.net/npm/mermaid@11/dist/mermaid.esm.min.mjs';
 | 
			
		||||
  import tidyTreeLayouts from 'https://cdn.jsdelivr.net/npm/@mermaid-js/layout-tidy-tree@0/dist/mermaid-layout-tidy-tree.esm.min.mjs';
 | 
			
		||||
 | 
			
		||||
  mermaid.registerLayoutLoaders(tidyTreeLayouts);
 | 
			
		||||
</script>
 | 
			
		||||
```
 | 
			
		||||
 | 
			
		||||
## Tidy Tree Layout Overview
 | 
			
		||||
 | 
			
		||||
tidy-tree: The bidirectional tidy tree layout
 | 
			
		||||
 | 
			
		||||
The bidirectional tidy tree layout algorithm creates two separate trees that grow horizontally in opposite directions from a central root node:
 | 
			
		||||
Left tree: grows horizontally to the left (children alternate: 1st, 3rd, 5th...)
 | 
			
		||||
Right tree: grows horizontally to the right (children alternate: 2nd, 4th, 6th...)
 | 
			
		||||
 | 
			
		||||
This creates a balanced, symmetric layout that is ideal for mindmaps, organizational charts, and other tree-based diagrams.
 | 
			
		||||
 | 
			
		||||
Layout Structure:
 | 
			
		||||
[Child 3] ← [Child 1] ← [Root] → [Child 2] → [Child 4]
 | 
			
		||||
↓ ↓ ↓ ↓
 | 
			
		||||
[GrandChild] [GrandChild] [GrandChild] [GrandChild]
 | 
			
		||||
							
								
								
									
										46
									
								
								packages/mermaid-layout-tidy-tree/package.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										46
									
								
								packages/mermaid-layout-tidy-tree/package.json
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,46 @@
 | 
			
		||||
{
 | 
			
		||||
  "name": "@mermaid-js/layout-tidy-tree",
 | 
			
		||||
  "version": "0.1.0",
 | 
			
		||||
  "description": "Tidy-tree layout engine for mermaid",
 | 
			
		||||
  "module": "dist/mermaid-layout-tidy-tree.core.mjs",
 | 
			
		||||
  "types": "dist/layouts.d.ts",
 | 
			
		||||
  "type": "module",
 | 
			
		||||
  "exports": {
 | 
			
		||||
    ".": {
 | 
			
		||||
      "import": "./dist/mermaid-layout-tidy-tree.core.mjs",
 | 
			
		||||
      "types": "./dist/layouts.d.ts"
 | 
			
		||||
    },
 | 
			
		||||
    "./": "./"
 | 
			
		||||
  },
 | 
			
		||||
  "keywords": [
 | 
			
		||||
    "diagram",
 | 
			
		||||
    "markdown",
 | 
			
		||||
    "tidy-tree",
 | 
			
		||||
    "mermaid",
 | 
			
		||||
    "layout"
 | 
			
		||||
  ],
 | 
			
		||||
  "scripts": {},
 | 
			
		||||
  "repository": {
 | 
			
		||||
    "type": "git",
 | 
			
		||||
    "url": "https://github.com/mermaid-js/mermaid"
 | 
			
		||||
  },
 | 
			
		||||
  "contributors": [
 | 
			
		||||
    "Knut Sveidqvist",
 | 
			
		||||
    "Sidharth Vinod"
 | 
			
		||||
  ],
 | 
			
		||||
  "license": "MIT",
 | 
			
		||||
  "dependencies": {
 | 
			
		||||
    "d3": "^7.9.0",
 | 
			
		||||
    "non-layered-tidy-tree-layout": "^2.0.2"
 | 
			
		||||
  },
 | 
			
		||||
  "devDependencies": {
 | 
			
		||||
    "@types/d3": "^7.4.3",
 | 
			
		||||
    "mermaid": "workspace:^"
 | 
			
		||||
  },
 | 
			
		||||
  "peerDependencies": {
 | 
			
		||||
    "mermaid": "^11.0.2"
 | 
			
		||||
  },
 | 
			
		||||
  "files": [
 | 
			
		||||
    "dist"
 | 
			
		||||
  ]
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										50
									
								
								packages/mermaid-layout-tidy-tree/src/index.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										50
									
								
								packages/mermaid-layout-tidy-tree/src/index.ts
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,50 @@
 | 
			
		||||
/**
 | 
			
		||||
 * Bidirectional Tidy-Tree Layout Algorithm for Generic Diagrams
 | 
			
		||||
 *
 | 
			
		||||
 * This module provides a layout algorithm implementation using the
 | 
			
		||||
 * non-layered-tidy-tree-layout algorithm for positioning nodes and edges
 | 
			
		||||
 * in tree structures with a bidirectional approach.
 | 
			
		||||
 *
 | 
			
		||||
 * The algorithm creates two separate trees that grow horizontally in opposite
 | 
			
		||||
 * directions from a central root node:
 | 
			
		||||
 * - Left tree: grows horizontally to the left (children alternate: 1st, 3rd, 5th...)
 | 
			
		||||
 * - Right tree: grows horizontally to the right (children alternate: 2nd, 4th, 6th...)
 | 
			
		||||
 *
 | 
			
		||||
 * This creates a balanced, symmetric layout that is ideal for mindmaps,
 | 
			
		||||
 * organizational charts, and other tree-based diagrams.
 | 
			
		||||
 *
 | 
			
		||||
 * The algorithm follows the unified rendering pattern and can be used
 | 
			
		||||
 * by any diagram type that provides compatible LayoutData.
 | 
			
		||||
 */
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Render function for the bidirectional tidy-tree layout algorithm
 | 
			
		||||
 *
 | 
			
		||||
 * This function follows the unified rendering pattern used by all layout algorithms.
 | 
			
		||||
 * It takes LayoutData, inserts nodes into DOM, runs the bidirectional tidy-tree layout algorithm,
 | 
			
		||||
 * and renders the positioned elements to the SVG.
 | 
			
		||||
 *
 | 
			
		||||
 * Features:
 | 
			
		||||
 * - Alternates root children between left and right trees
 | 
			
		||||
 * - Left tree grows horizontally to the left (rotated 90° counterclockwise)
 | 
			
		||||
 * - Right tree grows horizontally to the right (rotated 90° clockwise)
 | 
			
		||||
 * - Uses tidy-tree algorithm for optimal spacing within each tree
 | 
			
		||||
 * - Creates symmetric, balanced layouts
 | 
			
		||||
 * - Maintains proper edge connections between all nodes
 | 
			
		||||
 *
 | 
			
		||||
 * Layout Structure:
 | 
			
		||||
 * ```
 | 
			
		||||
 * [Child 3] ← [Child 1] ← [Root] → [Child 2] → [Child 4]
 | 
			
		||||
 *     ↓           ↓                     ↓           ↓
 | 
			
		||||
 * [GrandChild]  [GrandChild]      [GrandChild]  [GrandChild]
 | 
			
		||||
 * ```
 | 
			
		||||
 *
 | 
			
		||||
 * @param layoutData - Layout data containing nodes, edges, and configuration
 | 
			
		||||
 * @param svg - SVG element to render to
 | 
			
		||||
 * @param helpers - Internal helper functions for rendering
 | 
			
		||||
 * @param options - Rendering options
 | 
			
		||||
 */
 | 
			
		||||
export { default } from './layouts.js';
 | 
			
		||||
export * from './types.js';
 | 
			
		||||
export * from './layout.js';
 | 
			
		||||
export { render } from './render.js';
 | 
			
		||||
							
								
								
									
										410
									
								
								packages/mermaid-layout-tidy-tree/src/layout.test.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										410
									
								
								packages/mermaid-layout-tidy-tree/src/layout.test.ts
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,410 @@
 | 
			
		||||
import { describe, it, expect, beforeEach, vi } from 'vitest';
 | 
			
		||||
 | 
			
		||||
// Mock non-layered-tidy-tree-layout
 | 
			
		||||
vi.mock('non-layered-tidy-tree-layout', () => ({
 | 
			
		||||
  BoundingBox: vi.fn().mockImplementation(() => ({})),
 | 
			
		||||
  Layout: vi.fn().mockImplementation(() => ({
 | 
			
		||||
    layout: vi.fn().mockImplementation((treeData) => {
 | 
			
		||||
      const result = { ...treeData };
 | 
			
		||||
 | 
			
		||||
      if (result.id?.toString().startsWith('virtual-root')) {
 | 
			
		||||
        result.x = 0;
 | 
			
		||||
        result.y = 0;
 | 
			
		||||
      } else {
 | 
			
		||||
        result.x = 100;
 | 
			
		||||
        result.y = 50;
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      if (result.children) {
 | 
			
		||||
        result.children.forEach((child: any, index: number) => {
 | 
			
		||||
          child.x = 50 + index * 100;
 | 
			
		||||
          child.y = 100;
 | 
			
		||||
 | 
			
		||||
          if (child.children) {
 | 
			
		||||
            child.children.forEach((grandchild: any, gIndex: number) => {
 | 
			
		||||
              grandchild.x = 25 + gIndex * 50;
 | 
			
		||||
              grandchild.y = 200;
 | 
			
		||||
            });
 | 
			
		||||
          }
 | 
			
		||||
        });
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      return {
 | 
			
		||||
        result,
 | 
			
		||||
        boundingBox: {
 | 
			
		||||
          left: 0,
 | 
			
		||||
          right: 200,
 | 
			
		||||
          top: 0,
 | 
			
		||||
          bottom: 250,
 | 
			
		||||
        },
 | 
			
		||||
      };
 | 
			
		||||
    }),
 | 
			
		||||
  })),
 | 
			
		||||
}));
 | 
			
		||||
 | 
			
		||||
import { executeTidyTreeLayout, validateLayoutData } from './layout.js';
 | 
			
		||||
import type { LayoutResult } from './types.js';
 | 
			
		||||
import type { LayoutData, MermaidConfig } from 'mermaid';
 | 
			
		||||
 | 
			
		||||
describe('Tidy-Tree Layout Algorithm', () => {
 | 
			
		||||
  let mockConfig: MermaidConfig;
 | 
			
		||||
  let mockLayoutData: LayoutData;
 | 
			
		||||
 | 
			
		||||
  beforeEach(() => {
 | 
			
		||||
    mockConfig = {
 | 
			
		||||
      theme: 'default',
 | 
			
		||||
    } as MermaidConfig;
 | 
			
		||||
 | 
			
		||||
    mockLayoutData = {
 | 
			
		||||
      nodes: [
 | 
			
		||||
        {
 | 
			
		||||
          id: 'root',
 | 
			
		||||
          label: 'Root',
 | 
			
		||||
          isGroup: false,
 | 
			
		||||
          shape: 'rect',
 | 
			
		||||
          width: 100,
 | 
			
		||||
          height: 50,
 | 
			
		||||
          padding: 10,
 | 
			
		||||
          x: 0,
 | 
			
		||||
          y: 0,
 | 
			
		||||
          cssClasses: '',
 | 
			
		||||
          cssStyles: [],
 | 
			
		||||
          look: 'default',
 | 
			
		||||
        },
 | 
			
		||||
        {
 | 
			
		||||
          id: 'child1',
 | 
			
		||||
          label: 'Child 1',
 | 
			
		||||
          isGroup: false,
 | 
			
		||||
          shape: 'rect',
 | 
			
		||||
          width: 80,
 | 
			
		||||
          height: 40,
 | 
			
		||||
          padding: 10,
 | 
			
		||||
          x: 0,
 | 
			
		||||
          y: 0,
 | 
			
		||||
          cssClasses: '',
 | 
			
		||||
          cssStyles: [],
 | 
			
		||||
          look: 'default',
 | 
			
		||||
        },
 | 
			
		||||
        {
 | 
			
		||||
          id: 'child2',
 | 
			
		||||
          label: 'Child 2',
 | 
			
		||||
          isGroup: false,
 | 
			
		||||
          shape: 'rect',
 | 
			
		||||
          width: 80,
 | 
			
		||||
          height: 40,
 | 
			
		||||
          padding: 10,
 | 
			
		||||
          x: 0,
 | 
			
		||||
          y: 0,
 | 
			
		||||
          cssClasses: '',
 | 
			
		||||
          cssStyles: [],
 | 
			
		||||
          look: 'default',
 | 
			
		||||
        },
 | 
			
		||||
        {
 | 
			
		||||
          id: 'child3',
 | 
			
		||||
          label: 'Child 3',
 | 
			
		||||
          isGroup: false,
 | 
			
		||||
          shape: 'rect',
 | 
			
		||||
          width: 80,
 | 
			
		||||
          height: 40,
 | 
			
		||||
          padding: 10,
 | 
			
		||||
          x: 0,
 | 
			
		||||
          y: 0,
 | 
			
		||||
          cssClasses: '',
 | 
			
		||||
          cssStyles: [],
 | 
			
		||||
          look: 'default',
 | 
			
		||||
        },
 | 
			
		||||
        {
 | 
			
		||||
          id: 'child4',
 | 
			
		||||
          label: 'Child 4',
 | 
			
		||||
          isGroup: false,
 | 
			
		||||
          shape: 'rect',
 | 
			
		||||
          width: 80,
 | 
			
		||||
          height: 40,
 | 
			
		||||
          padding: 10,
 | 
			
		||||
          x: 0,
 | 
			
		||||
          y: 0,
 | 
			
		||||
          cssClasses: '',
 | 
			
		||||
          cssStyles: [],
 | 
			
		||||
          look: 'default',
 | 
			
		||||
        },
 | 
			
		||||
      ],
 | 
			
		||||
      edges: [
 | 
			
		||||
        {
 | 
			
		||||
          id: 'root_child1',
 | 
			
		||||
          start: 'root',
 | 
			
		||||
          end: 'child1',
 | 
			
		||||
          type: 'edge',
 | 
			
		||||
          classes: '',
 | 
			
		||||
          style: [],
 | 
			
		||||
          animate: false,
 | 
			
		||||
          arrowTypeEnd: 'arrow_point',
 | 
			
		||||
          arrowTypeStart: 'none',
 | 
			
		||||
        },
 | 
			
		||||
        {
 | 
			
		||||
          id: 'root_child2',
 | 
			
		||||
          start: 'root',
 | 
			
		||||
          end: 'child2',
 | 
			
		||||
          type: 'edge',
 | 
			
		||||
          classes: '',
 | 
			
		||||
          style: [],
 | 
			
		||||
          animate: false,
 | 
			
		||||
          arrowTypeEnd: 'arrow_point',
 | 
			
		||||
          arrowTypeStart: 'none',
 | 
			
		||||
        },
 | 
			
		||||
        {
 | 
			
		||||
          id: 'root_child3',
 | 
			
		||||
          start: 'root',
 | 
			
		||||
          end: 'child3',
 | 
			
		||||
          type: 'edge',
 | 
			
		||||
          classes: '',
 | 
			
		||||
          style: [],
 | 
			
		||||
          animate: false,
 | 
			
		||||
          arrowTypeEnd: 'arrow_point',
 | 
			
		||||
          arrowTypeStart: 'none',
 | 
			
		||||
        },
 | 
			
		||||
        {
 | 
			
		||||
          id: 'root_child4',
 | 
			
		||||
          start: 'root',
 | 
			
		||||
          end: 'child4',
 | 
			
		||||
          type: 'edge',
 | 
			
		||||
          classes: '',
 | 
			
		||||
          style: [],
 | 
			
		||||
          animate: false,
 | 
			
		||||
          arrowTypeEnd: 'arrow_point',
 | 
			
		||||
          arrowTypeStart: 'none',
 | 
			
		||||
        },
 | 
			
		||||
      ],
 | 
			
		||||
      config: mockConfig,
 | 
			
		||||
      direction: 'TB',
 | 
			
		||||
      type: 'test',
 | 
			
		||||
      diagramId: 'test-diagram',
 | 
			
		||||
      markers: [],
 | 
			
		||||
    };
 | 
			
		||||
  });
 | 
			
		||||
 | 
			
		||||
  describe('validateLayoutData', () => {
 | 
			
		||||
    it('should validate correct layout data', () => {
 | 
			
		||||
      expect(() => validateLayoutData(mockLayoutData)).not.toThrow();
 | 
			
		||||
    });
 | 
			
		||||
 | 
			
		||||
    it('should throw error for missing data', () => {
 | 
			
		||||
      expect(() => validateLayoutData(null as any)).toThrow('Layout data is required');
 | 
			
		||||
    });
 | 
			
		||||
 | 
			
		||||
    it('should throw error for missing config', () => {
 | 
			
		||||
      const invalidData = { ...mockLayoutData, config: null as any };
 | 
			
		||||
      expect(() => validateLayoutData(invalidData)).toThrow('Configuration is required');
 | 
			
		||||
    });
 | 
			
		||||
 | 
			
		||||
    it('should throw error for invalid nodes array', () => {
 | 
			
		||||
      const invalidData = { ...mockLayoutData, nodes: null as any };
 | 
			
		||||
      expect(() => validateLayoutData(invalidData)).toThrow('Nodes array is required');
 | 
			
		||||
    });
 | 
			
		||||
 | 
			
		||||
    it('should throw error for invalid edges array', () => {
 | 
			
		||||
      const invalidData = { ...mockLayoutData, edges: null as any };
 | 
			
		||||
      expect(() => validateLayoutData(invalidData)).toThrow('Edges array is required');
 | 
			
		||||
    });
 | 
			
		||||
  });
 | 
			
		||||
 | 
			
		||||
  describe('executeTidyTreeLayout function', () => {
 | 
			
		||||
    it('should execute layout algorithm successfully', async () => {
 | 
			
		||||
      const result: LayoutResult = await executeTidyTreeLayout(mockLayoutData, mockConfig);
 | 
			
		||||
 | 
			
		||||
      expect(result).toBeDefined();
 | 
			
		||||
      expect(result.nodes).toBeDefined();
 | 
			
		||||
      expect(result.edges).toBeDefined();
 | 
			
		||||
      expect(Array.isArray(result.nodes)).toBe(true);
 | 
			
		||||
      expect(Array.isArray(result.edges)).toBe(true);
 | 
			
		||||
    });
 | 
			
		||||
 | 
			
		||||
    it('should return positioned nodes with coordinates', async () => {
 | 
			
		||||
      const result: LayoutResult = await executeTidyTreeLayout(mockLayoutData, mockConfig);
 | 
			
		||||
 | 
			
		||||
      expect(result.nodes.length).toBeGreaterThan(0);
 | 
			
		||||
      result.nodes.forEach((node) => {
 | 
			
		||||
        expect(node.x).toBeDefined();
 | 
			
		||||
        expect(node.y).toBeDefined();
 | 
			
		||||
        expect(typeof node.x).toBe('number');
 | 
			
		||||
        expect(typeof node.y).toBe('number');
 | 
			
		||||
      });
 | 
			
		||||
    });
 | 
			
		||||
 | 
			
		||||
    it('should return positioned edges with coordinates', async () => {
 | 
			
		||||
      const result: LayoutResult = await executeTidyTreeLayout(mockLayoutData, mockConfig);
 | 
			
		||||
 | 
			
		||||
      expect(result.edges.length).toBeGreaterThan(0);
 | 
			
		||||
      result.edges.forEach((edge) => {
 | 
			
		||||
        expect(edge.startX).toBeDefined();
 | 
			
		||||
        expect(edge.startY).toBeDefined();
 | 
			
		||||
        expect(edge.midX).toBeDefined();
 | 
			
		||||
        expect(edge.midY).toBeDefined();
 | 
			
		||||
        expect(edge.endX).toBeDefined();
 | 
			
		||||
        expect(edge.endY).toBeDefined();
 | 
			
		||||
      });
 | 
			
		||||
    });
 | 
			
		||||
 | 
			
		||||
    it('should handle empty layout data gracefully', async () => {
 | 
			
		||||
      const emptyData: LayoutData = {
 | 
			
		||||
        ...mockLayoutData,
 | 
			
		||||
        nodes: [],
 | 
			
		||||
        edges: [],
 | 
			
		||||
      };
 | 
			
		||||
 | 
			
		||||
      await expect(executeTidyTreeLayout(emptyData, mockConfig)).rejects.toThrow(
 | 
			
		||||
        'No nodes found in layout data'
 | 
			
		||||
      );
 | 
			
		||||
    });
 | 
			
		||||
 | 
			
		||||
    it('should throw error for missing nodes', async () => {
 | 
			
		||||
      const invalidData = { ...mockLayoutData, nodes: [] };
 | 
			
		||||
 | 
			
		||||
      await expect(executeTidyTreeLayout(invalidData, mockConfig)).rejects.toThrow(
 | 
			
		||||
        'No nodes found in layout data'
 | 
			
		||||
      );
 | 
			
		||||
    });
 | 
			
		||||
 | 
			
		||||
    it('should handle empty edges (single node tree)', async () => {
 | 
			
		||||
      const singleNodeData = {
 | 
			
		||||
        ...mockLayoutData,
 | 
			
		||||
        edges: [],
 | 
			
		||||
        nodes: [mockLayoutData.nodes[0]],
 | 
			
		||||
      };
 | 
			
		||||
 | 
			
		||||
      const result = await executeTidyTreeLayout(singleNodeData, mockConfig);
 | 
			
		||||
      expect(result).toBeDefined();
 | 
			
		||||
      expect(result.nodes).toHaveLength(1);
 | 
			
		||||
      expect(result.edges).toHaveLength(0);
 | 
			
		||||
    });
 | 
			
		||||
 | 
			
		||||
    it('should create bidirectional dual-tree layout with alternating left/right children', async () => {
 | 
			
		||||
      const result = await executeTidyTreeLayout(mockLayoutData, mockConfig);
 | 
			
		||||
 | 
			
		||||
      expect(result).toBeDefined();
 | 
			
		||||
      expect(result.nodes).toHaveLength(5);
 | 
			
		||||
 | 
			
		||||
      const rootNode = result.nodes.find((node) => node.id === 'root');
 | 
			
		||||
      expect(rootNode).toBeDefined();
 | 
			
		||||
      expect(rootNode!.x).toBe(0);
 | 
			
		||||
      expect(rootNode!.y).toBe(20);
 | 
			
		||||
 | 
			
		||||
      const child1 = result.nodes.find((node) => node.id === 'child1');
 | 
			
		||||
      const child2 = result.nodes.find((node) => node.id === 'child2');
 | 
			
		||||
      const child3 = result.nodes.find((node) => node.id === 'child3');
 | 
			
		||||
      const child4 = result.nodes.find((node) => node.id === 'child4');
 | 
			
		||||
 | 
			
		||||
      expect(child1).toBeDefined();
 | 
			
		||||
      expect(child2).toBeDefined();
 | 
			
		||||
      expect(child3).toBeDefined();
 | 
			
		||||
      expect(child4).toBeDefined();
 | 
			
		||||
 | 
			
		||||
      expect(child1!.x).toBeLessThan(rootNode!.x);
 | 
			
		||||
      expect(child2!.x).toBeGreaterThan(rootNode!.x);
 | 
			
		||||
      expect(child3!.x).toBeLessThan(rootNode!.x);
 | 
			
		||||
      expect(child4!.x).toBeGreaterThan(rootNode!.x);
 | 
			
		||||
 | 
			
		||||
      expect(child1!.x).toBeLessThan(-100);
 | 
			
		||||
      expect(child3!.x).toBeLessThan(-100);
 | 
			
		||||
 | 
			
		||||
      expect(child2!.x).toBeGreaterThan(100);
 | 
			
		||||
      expect(child4!.x).toBeGreaterThan(100);
 | 
			
		||||
    });
 | 
			
		||||
 | 
			
		||||
    it('should correctly transpose coordinates to prevent high nodes from covering nodes above them', async () => {
 | 
			
		||||
      const testData = {
 | 
			
		||||
        ...mockLayoutData,
 | 
			
		||||
        nodes: [
 | 
			
		||||
          {
 | 
			
		||||
            id: 'root',
 | 
			
		||||
            label: 'Root',
 | 
			
		||||
            isGroup: false,
 | 
			
		||||
            shape: 'rect' as const,
 | 
			
		||||
            width: 100,
 | 
			
		||||
            height: 50,
 | 
			
		||||
            padding: 10,
 | 
			
		||||
            x: 0,
 | 
			
		||||
            y: 0,
 | 
			
		||||
            cssClasses: '',
 | 
			
		||||
            cssStyles: [],
 | 
			
		||||
            look: 'default',
 | 
			
		||||
          },
 | 
			
		||||
          {
 | 
			
		||||
            id: 'tall-child',
 | 
			
		||||
            label: 'Tall Child',
 | 
			
		||||
            isGroup: false,
 | 
			
		||||
            shape: 'rect' as const,
 | 
			
		||||
            width: 80,
 | 
			
		||||
            height: 120,
 | 
			
		||||
            padding: 10,
 | 
			
		||||
            x: 0,
 | 
			
		||||
            y: 0,
 | 
			
		||||
            cssClasses: '',
 | 
			
		||||
            cssStyles: [],
 | 
			
		||||
            look: 'default',
 | 
			
		||||
          },
 | 
			
		||||
          {
 | 
			
		||||
            id: 'short-child',
 | 
			
		||||
            label: 'Short Child',
 | 
			
		||||
            isGroup: false,
 | 
			
		||||
            shape: 'rect' as const,
 | 
			
		||||
            width: 80,
 | 
			
		||||
            height: 30,
 | 
			
		||||
            padding: 10,
 | 
			
		||||
            x: 0,
 | 
			
		||||
            y: 0,
 | 
			
		||||
            cssClasses: '',
 | 
			
		||||
            cssStyles: [],
 | 
			
		||||
            look: 'default',
 | 
			
		||||
          },
 | 
			
		||||
        ],
 | 
			
		||||
        edges: [
 | 
			
		||||
          {
 | 
			
		||||
            id: 'root_tall',
 | 
			
		||||
            start: 'root',
 | 
			
		||||
            end: 'tall-child',
 | 
			
		||||
            type: 'edge',
 | 
			
		||||
            classes: '',
 | 
			
		||||
            style: [],
 | 
			
		||||
            animate: false,
 | 
			
		||||
            arrowTypeEnd: 'arrow_point',
 | 
			
		||||
            arrowTypeStart: 'none',
 | 
			
		||||
          },
 | 
			
		||||
          {
 | 
			
		||||
            id: 'root_short',
 | 
			
		||||
            start: 'root',
 | 
			
		||||
            end: 'short-child',
 | 
			
		||||
            type: 'edge',
 | 
			
		||||
            classes: '',
 | 
			
		||||
            style: [],
 | 
			
		||||
            animate: false,
 | 
			
		||||
            arrowTypeEnd: 'arrow_point',
 | 
			
		||||
            arrowTypeStart: 'none',
 | 
			
		||||
          },
 | 
			
		||||
        ],
 | 
			
		||||
      };
 | 
			
		||||
 | 
			
		||||
      const result = await executeTidyTreeLayout(testData, mockConfig);
 | 
			
		||||
 | 
			
		||||
      expect(result).toBeDefined();
 | 
			
		||||
      expect(result.nodes).toHaveLength(3);
 | 
			
		||||
 | 
			
		||||
      const rootNode = result.nodes.find((node) => node.id === 'root');
 | 
			
		||||
      const tallChild = result.nodes.find((node) => node.id === 'tall-child');
 | 
			
		||||
      const shortChild = result.nodes.find((node) => node.id === 'short-child');
 | 
			
		||||
 | 
			
		||||
      expect(rootNode).toBeDefined();
 | 
			
		||||
      expect(tallChild).toBeDefined();
 | 
			
		||||
      expect(shortChild).toBeDefined();
 | 
			
		||||
 | 
			
		||||
      expect(tallChild!.x).not.toBe(shortChild!.x);
 | 
			
		||||
 | 
			
		||||
      expect(tallChild!.width).toBe(80);
 | 
			
		||||
      expect(tallChild!.height).toBe(120);
 | 
			
		||||
      expect(shortChild!.width).toBe(80);
 | 
			
		||||
      expect(shortChild!.height).toBe(30);
 | 
			
		||||
 | 
			
		||||
      const yDifference = Math.abs(tallChild!.y - shortChild!.y);
 | 
			
		||||
      expect(yDifference).toBeGreaterThanOrEqual(0);
 | 
			
		||||
    });
 | 
			
		||||
  });
 | 
			
		||||
});
 | 
			
		||||
							
								
								
									
										638
									
								
								packages/mermaid-layout-tidy-tree/src/layout.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										638
									
								
								packages/mermaid-layout-tidy-tree/src/layout.ts
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,638 @@
 | 
			
		||||
import { BoundingBox, Layout } from 'non-layered-tidy-tree-layout';
 | 
			
		||||
import type { MermaidConfig, LayoutData } from 'mermaid';
 | 
			
		||||
import type {
 | 
			
		||||
  LayoutResult,
 | 
			
		||||
  TidyTreeNode,
 | 
			
		||||
  PositionedNode,
 | 
			
		||||
  PositionedEdge,
 | 
			
		||||
  Node,
 | 
			
		||||
  Edge,
 | 
			
		||||
} from './types.js';
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Execute the tidy-tree layout algorithm on generic layout data
 | 
			
		||||
 *
 | 
			
		||||
 * This function takes layout data and uses the non-layered-tidy-tree-layout
 | 
			
		||||
 * algorithm to calculate optimal node positions for tree structures.
 | 
			
		||||
 *
 | 
			
		||||
 * @param data - The layout data containing nodes, edges, and configuration
 | 
			
		||||
 * @param config - Mermaid configuration object
 | 
			
		||||
 * @returns Promise resolving to layout result with positioned nodes and edges
 | 
			
		||||
 */
 | 
			
		||||
export function executeTidyTreeLayout(
 | 
			
		||||
  data: LayoutData,
 | 
			
		||||
  _config: MermaidConfig
 | 
			
		||||
): Promise<LayoutResult> {
 | 
			
		||||
  let intersectionShift = 50;
 | 
			
		||||
 | 
			
		||||
  return new Promise((resolve, reject) => {
 | 
			
		||||
    try {
 | 
			
		||||
      if (!data.nodes || !Array.isArray(data.nodes) || data.nodes.length === 0) {
 | 
			
		||||
        throw new Error('No nodes found in layout data');
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      if (!data.edges || !Array.isArray(data.edges)) {
 | 
			
		||||
        data.edges = [];
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      const { leftTree, rightTree, rootNode } = convertToDualTreeFormat(data);
 | 
			
		||||
 | 
			
		||||
      const gap = 20;
 | 
			
		||||
      const bottomPadding = 40;
 | 
			
		||||
      intersectionShift = 30;
 | 
			
		||||
 | 
			
		||||
      const bb = new BoundingBox(gap, bottomPadding);
 | 
			
		||||
      const layout = new Layout(bb);
 | 
			
		||||
 | 
			
		||||
      let leftResult = null;
 | 
			
		||||
      let rightResult = null;
 | 
			
		||||
      let leftBoundingBox = null;
 | 
			
		||||
      let rightBoundingBox = null;
 | 
			
		||||
 | 
			
		||||
      if (leftTree) {
 | 
			
		||||
        const leftLayoutResult = layout.layout(leftTree);
 | 
			
		||||
        leftResult = leftLayoutResult.result;
 | 
			
		||||
        leftBoundingBox = leftLayoutResult.boundingBox;
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      if (rightTree) {
 | 
			
		||||
        const rightLayoutResult = layout.layout(rightTree);
 | 
			
		||||
        rightResult = rightLayoutResult.result;
 | 
			
		||||
        rightBoundingBox = rightLayoutResult.boundingBox;
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      const positionedNodes = combineAndPositionTrees(
 | 
			
		||||
        rootNode,
 | 
			
		||||
        leftResult,
 | 
			
		||||
        rightResult,
 | 
			
		||||
        leftBoundingBox,
 | 
			
		||||
        rightBoundingBox,
 | 
			
		||||
        data
 | 
			
		||||
      );
 | 
			
		||||
      const positionedEdges = calculateEdgePositions(
 | 
			
		||||
        data.edges,
 | 
			
		||||
        positionedNodes,
 | 
			
		||||
        intersectionShift
 | 
			
		||||
      );
 | 
			
		||||
 | 
			
		||||
      resolve({
 | 
			
		||||
        nodes: positionedNodes,
 | 
			
		||||
        edges: positionedEdges,
 | 
			
		||||
      });
 | 
			
		||||
    } catch (error) {
 | 
			
		||||
      reject(error);
 | 
			
		||||
    }
 | 
			
		||||
  });
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Convert LayoutData to dual-tree format (left and right trees)
 | 
			
		||||
 *
 | 
			
		||||
 * This function builds two separate tree structures from the nodes and edges,
 | 
			
		||||
 * alternating children between left and right trees.
 | 
			
		||||
 */
 | 
			
		||||
function convertToDualTreeFormat(data: LayoutData): {
 | 
			
		||||
  leftTree: TidyTreeNode | null;
 | 
			
		||||
  rightTree: TidyTreeNode | null;
 | 
			
		||||
  rootNode: TidyTreeNode;
 | 
			
		||||
} {
 | 
			
		||||
  const { nodes, edges } = data;
 | 
			
		||||
 | 
			
		||||
  const nodeMap = new Map<string, Node>();
 | 
			
		||||
  nodes.forEach((node) => nodeMap.set(node.id, node));
 | 
			
		||||
 | 
			
		||||
  const children = new Map<string, string[]>();
 | 
			
		||||
  const parents = new Map<string, string>();
 | 
			
		||||
 | 
			
		||||
  edges.forEach((edge) => {
 | 
			
		||||
    const parentId = edge.start;
 | 
			
		||||
    const childId = edge.end;
 | 
			
		||||
 | 
			
		||||
    if (parentId && childId) {
 | 
			
		||||
      if (!children.has(parentId)) {
 | 
			
		||||
        children.set(parentId, []);
 | 
			
		||||
      }
 | 
			
		||||
      children.get(parentId)!.push(childId);
 | 
			
		||||
      parents.set(childId, parentId);
 | 
			
		||||
    }
 | 
			
		||||
  });
 | 
			
		||||
 | 
			
		||||
  const rootNodeData = nodes.find((node) => !parents.has(node.id));
 | 
			
		||||
  if (!rootNodeData && nodes.length === 0) {
 | 
			
		||||
    throw new Error('No nodes available to create tree');
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  const actualRoot = rootNodeData ?? nodes[0];
 | 
			
		||||
 | 
			
		||||
  const rootNode: TidyTreeNode = {
 | 
			
		||||
    id: actualRoot.id,
 | 
			
		||||
    width: actualRoot.width ?? 100,
 | 
			
		||||
    height: actualRoot.height ?? 50,
 | 
			
		||||
    _originalNode: actualRoot,
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  const rootChildren = children.get(actualRoot.id) ?? [];
 | 
			
		||||
  const leftChildren: string[] = [];
 | 
			
		||||
  const rightChildren: string[] = [];
 | 
			
		||||
 | 
			
		||||
  rootChildren.forEach((childId, index) => {
 | 
			
		||||
    if (index % 2 === 0) {
 | 
			
		||||
      leftChildren.push(childId);
 | 
			
		||||
    } else {
 | 
			
		||||
      rightChildren.push(childId);
 | 
			
		||||
    }
 | 
			
		||||
  });
 | 
			
		||||
 | 
			
		||||
  const leftTree = leftChildren.length > 0 ? buildSubTree(leftChildren, children, nodeMap) : null;
 | 
			
		||||
 | 
			
		||||
  const rightTree =
 | 
			
		||||
    rightChildren.length > 0 ? buildSubTree(rightChildren, children, nodeMap) : null;
 | 
			
		||||
 | 
			
		||||
  return { leftTree, rightTree, rootNode };
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Build a subtree from a list of root children
 | 
			
		||||
 * For horizontal trees, we need to transpose width/height since the tree will be rotated 90°
 | 
			
		||||
 */
 | 
			
		||||
function buildSubTree(
 | 
			
		||||
  rootChildren: string[],
 | 
			
		||||
  children: Map<string, string[]>,
 | 
			
		||||
  nodeMap: Map<string, Node>
 | 
			
		||||
): TidyTreeNode {
 | 
			
		||||
  const virtualRoot: TidyTreeNode = {
 | 
			
		||||
    id: `virtual-root-${Math.random()}`,
 | 
			
		||||
    width: 1,
 | 
			
		||||
    height: 1,
 | 
			
		||||
    children: rootChildren
 | 
			
		||||
      .map((childId) => nodeMap.get(childId))
 | 
			
		||||
      .filter((child): child is Node => child !== undefined)
 | 
			
		||||
      .map((child) => convertNodeToTidyTreeTransposed(child, children, nodeMap)),
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  return virtualRoot;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Recursively convert a node and its children to tidy-tree format
 | 
			
		||||
 * This version transposes width/height for horizontal tree layout
 | 
			
		||||
 */
 | 
			
		||||
function convertNodeToTidyTreeTransposed(
 | 
			
		||||
  node: Node,
 | 
			
		||||
  children: Map<string, string[]>,
 | 
			
		||||
  nodeMap: Map<string, Node>
 | 
			
		||||
): TidyTreeNode {
 | 
			
		||||
  const childIds = children.get(node.id) ?? [];
 | 
			
		||||
  const childNodes = childIds
 | 
			
		||||
    .map((childId) => nodeMap.get(childId))
 | 
			
		||||
    .filter((child): child is Node => child !== undefined)
 | 
			
		||||
    .map((child) => convertNodeToTidyTreeTransposed(child, children, nodeMap));
 | 
			
		||||
 | 
			
		||||
  return {
 | 
			
		||||
    id: node.id,
 | 
			
		||||
    width: node.height ?? 50,
 | 
			
		||||
    height: node.width ?? 100,
 | 
			
		||||
    children: childNodes.length > 0 ? childNodes : undefined,
 | 
			
		||||
    _originalNode: node,
 | 
			
		||||
  };
 | 
			
		||||
}
 | 
			
		||||
/**
 | 
			
		||||
 * Combine and position the left and right trees around the root node
 | 
			
		||||
 * Creates a bidirectional layout where left tree grows left and right tree grows right
 | 
			
		||||
 */
 | 
			
		||||
function combineAndPositionTrees(
 | 
			
		||||
  rootNode: TidyTreeNode,
 | 
			
		||||
  leftResult: TidyTreeNode | null,
 | 
			
		||||
  rightResult: TidyTreeNode | null,
 | 
			
		||||
  _leftBoundingBox: any,
 | 
			
		||||
  _rightBoundingBox: any,
 | 
			
		||||
  _data: LayoutData
 | 
			
		||||
): PositionedNode[] {
 | 
			
		||||
  const positionedNodes: PositionedNode[] = [];
 | 
			
		||||
 | 
			
		||||
  const rootX = 0;
 | 
			
		||||
  const rootY = 0;
 | 
			
		||||
 | 
			
		||||
  const treeSpacing = rootNode.width / 2 + 30;
 | 
			
		||||
  const leftTreeNodes: PositionedNode[] = [];
 | 
			
		||||
  const rightTreeNodes: PositionedNode[] = [];
 | 
			
		||||
 | 
			
		||||
  if (leftResult?.children) {
 | 
			
		||||
    positionLeftTreeBidirectional(leftResult.children, leftTreeNodes, rootX - treeSpacing, rootY);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  if (rightResult?.children) {
 | 
			
		||||
    positionRightTreeBidirectional(
 | 
			
		||||
      rightResult.children,
 | 
			
		||||
      rightTreeNodes,
 | 
			
		||||
      rootX + treeSpacing,
 | 
			
		||||
      rootY
 | 
			
		||||
    );
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  let leftTreeCenterY = 0;
 | 
			
		||||
  let rightTreeCenterY = 0;
 | 
			
		||||
 | 
			
		||||
  if (leftTreeNodes.length > 0) {
 | 
			
		||||
    const leftTreeXPositions = [...new Set(leftTreeNodes.map((node) => node.x))].sort(
 | 
			
		||||
      (a, b) => b - a
 | 
			
		||||
    );
 | 
			
		||||
    const firstLevelLeftX = leftTreeXPositions[0];
 | 
			
		||||
    const firstLevelLeftNodes = leftTreeNodes.filter((node) => node.x === firstLevelLeftX);
 | 
			
		||||
 | 
			
		||||
    if (firstLevelLeftNodes.length > 0) {
 | 
			
		||||
      const leftMinY = Math.min(
 | 
			
		||||
        ...firstLevelLeftNodes.map((node) => node.y - (node.height ?? 50) / 2)
 | 
			
		||||
      );
 | 
			
		||||
      const leftMaxY = Math.max(
 | 
			
		||||
        ...firstLevelLeftNodes.map((node) => node.y + (node.height ?? 50) / 2)
 | 
			
		||||
      );
 | 
			
		||||
      leftTreeCenterY = (leftMinY + leftMaxY) / 2;
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  if (rightTreeNodes.length > 0) {
 | 
			
		||||
    const rightTreeXPositions = [...new Set(rightTreeNodes.map((node) => node.x))].sort(
 | 
			
		||||
      (a, b) => a - b
 | 
			
		||||
    );
 | 
			
		||||
    const firstLevelRightX = rightTreeXPositions[0];
 | 
			
		||||
    const firstLevelRightNodes = rightTreeNodes.filter((node) => node.x === firstLevelRightX);
 | 
			
		||||
 | 
			
		||||
    if (firstLevelRightNodes.length > 0) {
 | 
			
		||||
      const rightMinY = Math.min(
 | 
			
		||||
        ...firstLevelRightNodes.map((node) => node.y - (node.height ?? 50) / 2)
 | 
			
		||||
      );
 | 
			
		||||
      const rightMaxY = Math.max(
 | 
			
		||||
        ...firstLevelRightNodes.map((node) => node.y + (node.height ?? 50) / 2)
 | 
			
		||||
      );
 | 
			
		||||
      rightTreeCenterY = (rightMinY + rightMaxY) / 2;
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  const leftTreeOffset = -leftTreeCenterY;
 | 
			
		||||
  const rightTreeOffset = -rightTreeCenterY;
 | 
			
		||||
 | 
			
		||||
  positionedNodes.push({
 | 
			
		||||
    id: String(rootNode.id),
 | 
			
		||||
    x: rootX,
 | 
			
		||||
    y: rootY + 20,
 | 
			
		||||
    section: 'root',
 | 
			
		||||
    width: rootNode._originalNode?.width ?? rootNode.width,
 | 
			
		||||
    height: rootNode._originalNode?.height ?? rootNode.height,
 | 
			
		||||
    originalNode: rootNode._originalNode,
 | 
			
		||||
  });
 | 
			
		||||
 | 
			
		||||
  const leftTreeNodesWithOffset = leftTreeNodes.map((node) => ({
 | 
			
		||||
    id: node.id,
 | 
			
		||||
    x: node.x - (node.width ?? 0) / 2,
 | 
			
		||||
    y: node.y + leftTreeOffset + (node.height ?? 0) / 2,
 | 
			
		||||
    section: 'left' as const,
 | 
			
		||||
    width: node.width,
 | 
			
		||||
    height: node.height,
 | 
			
		||||
    originalNode: node.originalNode,
 | 
			
		||||
  }));
 | 
			
		||||
 | 
			
		||||
  const rightTreeNodesWithOffset = rightTreeNodes.map((node) => ({
 | 
			
		||||
    id: node.id,
 | 
			
		||||
    x: node.x + (node.width ?? 0) / 2,
 | 
			
		||||
    y: node.y + rightTreeOffset + (node.height ?? 0) / 2,
 | 
			
		||||
    section: 'right' as const,
 | 
			
		||||
    width: node.width,
 | 
			
		||||
    height: node.height,
 | 
			
		||||
    originalNode: node.originalNode,
 | 
			
		||||
  }));
 | 
			
		||||
 | 
			
		||||
  positionedNodes.push(...leftTreeNodesWithOffset);
 | 
			
		||||
  positionedNodes.push(...rightTreeNodesWithOffset);
 | 
			
		||||
 | 
			
		||||
  return positionedNodes;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Position nodes from the left tree in a bidirectional layout (grows to the left)
 | 
			
		||||
 * Rotates the tree 90 degrees counterclockwise so it grows horizontally to the left
 | 
			
		||||
 */
 | 
			
		||||
function positionLeftTreeBidirectional(
 | 
			
		||||
  nodes: TidyTreeNode[],
 | 
			
		||||
  positionedNodes: PositionedNode[],
 | 
			
		||||
  offsetX: number,
 | 
			
		||||
  offsetY: number
 | 
			
		||||
): void {
 | 
			
		||||
  nodes.forEach((node) => {
 | 
			
		||||
    const distanceFromRoot = node.y ?? 0;
 | 
			
		||||
    const verticalPosition = node.x ?? 0;
 | 
			
		||||
 | 
			
		||||
    const originalWidth = node._originalNode?.width ?? 100;
 | 
			
		||||
    const originalHeight = node._originalNode?.height ?? 50;
 | 
			
		||||
 | 
			
		||||
    const adjustedY = offsetY + verticalPosition;
 | 
			
		||||
 | 
			
		||||
    positionedNodes.push({
 | 
			
		||||
      id: String(node.id),
 | 
			
		||||
      x: offsetX - distanceFromRoot,
 | 
			
		||||
      y: adjustedY,
 | 
			
		||||
      width: originalWidth,
 | 
			
		||||
      height: originalHeight,
 | 
			
		||||
      originalNode: node._originalNode,
 | 
			
		||||
    });
 | 
			
		||||
 | 
			
		||||
    if (node.children) {
 | 
			
		||||
      positionLeftTreeBidirectional(node.children, positionedNodes, offsetX, offsetY);
 | 
			
		||||
    }
 | 
			
		||||
  });
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Position nodes from the right tree in a bidirectional layout (grows to the right)
 | 
			
		||||
 * Rotates the tree 90 degrees clockwise so it grows horizontally to the right
 | 
			
		||||
 */
 | 
			
		||||
function positionRightTreeBidirectional(
 | 
			
		||||
  nodes: TidyTreeNode[],
 | 
			
		||||
  positionedNodes: PositionedNode[],
 | 
			
		||||
  offsetX: number,
 | 
			
		||||
  offsetY: number
 | 
			
		||||
): void {
 | 
			
		||||
  nodes.forEach((node) => {
 | 
			
		||||
    const distanceFromRoot = node.y ?? 0;
 | 
			
		||||
    const verticalPosition = node.x ?? 0;
 | 
			
		||||
 | 
			
		||||
    const originalWidth = node._originalNode?.width ?? 100;
 | 
			
		||||
    const originalHeight = node._originalNode?.height ?? 50;
 | 
			
		||||
 | 
			
		||||
    const adjustedY = offsetY + verticalPosition;
 | 
			
		||||
 | 
			
		||||
    positionedNodes.push({
 | 
			
		||||
      id: String(node.id),
 | 
			
		||||
      x: offsetX + distanceFromRoot,
 | 
			
		||||
      y: adjustedY,
 | 
			
		||||
      width: originalWidth,
 | 
			
		||||
      height: originalHeight,
 | 
			
		||||
      originalNode: node._originalNode,
 | 
			
		||||
    });
 | 
			
		||||
 | 
			
		||||
    if (node.children) {
 | 
			
		||||
      positionRightTreeBidirectional(node.children, positionedNodes, offsetX, offsetY);
 | 
			
		||||
    }
 | 
			
		||||
  });
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Calculate the intersection point of a line with a circle
 | 
			
		||||
 * @param circle - Circle coordinates and radius
 | 
			
		||||
 * @param lineStart - Starting point of the line
 | 
			
		||||
 * @param lineEnd - Ending point of the line
 | 
			
		||||
 * @returns The intersection point
 | 
			
		||||
 */
 | 
			
		||||
function computeCircleEdgeIntersection(
 | 
			
		||||
  circle: { x: number; y: number; width: number; height: number },
 | 
			
		||||
  lineStart: { x: number; y: number },
 | 
			
		||||
  lineEnd: { x: number; y: number }
 | 
			
		||||
): { x: number; y: number } {
 | 
			
		||||
  const radius = Math.min(circle.width, circle.height) / 2;
 | 
			
		||||
 | 
			
		||||
  const dx = lineEnd.x - lineStart.x;
 | 
			
		||||
  const dy = lineEnd.y - lineStart.y;
 | 
			
		||||
  const length = Math.sqrt(dx * dx + dy * dy);
 | 
			
		||||
 | 
			
		||||
  if (length === 0) {
 | 
			
		||||
    return lineStart;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  const nx = dx / length;
 | 
			
		||||
  const ny = dy / length;
 | 
			
		||||
 | 
			
		||||
  return {
 | 
			
		||||
    x: circle.x - nx * radius,
 | 
			
		||||
    y: circle.y - ny * radius,
 | 
			
		||||
  };
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Calculate intersection point of a line with a rectangle
 | 
			
		||||
 * This is a simplified version that we'll use instead of importing from mermaid
 | 
			
		||||
 */
 | 
			
		||||
function intersection(
 | 
			
		||||
  node: { x: number; y: number; width?: number; height?: number },
 | 
			
		||||
  point1: { x: number; y: number },
 | 
			
		||||
  point2: { x: number; y: number }
 | 
			
		||||
): { x: number; y: number } {
 | 
			
		||||
  const nodeWidth = node.width ?? 100;
 | 
			
		||||
  const nodeHeight = node.height ?? 50;
 | 
			
		||||
 | 
			
		||||
  const centerX = node.x;
 | 
			
		||||
  const centerY = node.y;
 | 
			
		||||
 | 
			
		||||
  const dx = point2.x - point1.x;
 | 
			
		||||
  const dy = point2.y - point1.y;
 | 
			
		||||
 | 
			
		||||
  if (dx === 0 && dy === 0) {
 | 
			
		||||
    return { x: centerX, y: centerY };
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  const halfWidth = nodeWidth / 2;
 | 
			
		||||
  const halfHeight = nodeHeight / 2;
 | 
			
		||||
 | 
			
		||||
  let intersectionX = centerX;
 | 
			
		||||
  let intersectionY = centerY;
 | 
			
		||||
 | 
			
		||||
  if (Math.abs(dx) > Math.abs(dy)) {
 | 
			
		||||
    if (dx > 0) {
 | 
			
		||||
      intersectionX = centerX + halfWidth;
 | 
			
		||||
      intersectionY = centerY + (halfWidth * dy) / dx;
 | 
			
		||||
    } else {
 | 
			
		||||
      intersectionX = centerX - halfWidth;
 | 
			
		||||
      intersectionY = centerY - (halfWidth * dy) / dx;
 | 
			
		||||
    }
 | 
			
		||||
  } else {
 | 
			
		||||
    if (dy > 0) {
 | 
			
		||||
      intersectionY = centerY + halfHeight;
 | 
			
		||||
      intersectionX = centerX + (halfHeight * dx) / dy;
 | 
			
		||||
    } else {
 | 
			
		||||
      intersectionY = centerY - halfHeight;
 | 
			
		||||
      intersectionX = centerX - (halfHeight * dx) / dy;
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  return { x: intersectionX, y: intersectionY };
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Calculate edge positions based on positioned nodes
 | 
			
		||||
 * Now includes tree membership and node dimensions for precise edge calculations
 | 
			
		||||
 * Edges now stop at shape boundaries instead of extending to centers
 | 
			
		||||
 */
 | 
			
		||||
function calculateEdgePositions(
 | 
			
		||||
  edges: Edge[],
 | 
			
		||||
  positionedNodes: PositionedNode[],
 | 
			
		||||
  intersectionShift: number
 | 
			
		||||
): PositionedEdge[] {
 | 
			
		||||
  const nodeInfo = new Map<string, PositionedNode>();
 | 
			
		||||
  positionedNodes.forEach((node) => {
 | 
			
		||||
    nodeInfo.set(node.id, node);
 | 
			
		||||
  });
 | 
			
		||||
 | 
			
		||||
  return edges.map((edge) => {
 | 
			
		||||
    const sourceNode = nodeInfo.get(edge.start ?? '');
 | 
			
		||||
    const targetNode = nodeInfo.get(edge.end ?? '');
 | 
			
		||||
 | 
			
		||||
    if (!sourceNode || !targetNode) {
 | 
			
		||||
      return {
 | 
			
		||||
        id: edge.id,
 | 
			
		||||
        source: edge.start ?? '',
 | 
			
		||||
        target: edge.end ?? '',
 | 
			
		||||
        startX: 0,
 | 
			
		||||
        startY: 0,
 | 
			
		||||
        midX: 0,
 | 
			
		||||
        midY: 0,
 | 
			
		||||
        endX: 0,
 | 
			
		||||
        endY: 0,
 | 
			
		||||
        points: [{ x: 0, y: 0 }],
 | 
			
		||||
        sourceSection: undefined,
 | 
			
		||||
        targetSection: undefined,
 | 
			
		||||
        sourceWidth: undefined,
 | 
			
		||||
        sourceHeight: undefined,
 | 
			
		||||
        targetWidth: undefined,
 | 
			
		||||
        targetHeight: undefined,
 | 
			
		||||
      };
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    const sourceCenter = { x: sourceNode.x, y: sourceNode.y };
 | 
			
		||||
    const targetCenter = { x: targetNode.x, y: targetNode.y };
 | 
			
		||||
 | 
			
		||||
    const isSourceRound = ['circle', 'cloud', 'bang'].includes(
 | 
			
		||||
      sourceNode.originalNode?.shape ?? ''
 | 
			
		||||
    );
 | 
			
		||||
    const isTargetRound = ['circle', 'cloud', 'bang'].includes(
 | 
			
		||||
      targetNode.originalNode?.shape ?? ''
 | 
			
		||||
    );
 | 
			
		||||
 | 
			
		||||
    let startPos = isSourceRound
 | 
			
		||||
      ? computeCircleEdgeIntersection(
 | 
			
		||||
          {
 | 
			
		||||
            x: sourceNode.x,
 | 
			
		||||
            y: sourceNode.y,
 | 
			
		||||
            width: sourceNode.width ?? 100,
 | 
			
		||||
            height: sourceNode.height ?? 100,
 | 
			
		||||
          },
 | 
			
		||||
          targetCenter,
 | 
			
		||||
          sourceCenter
 | 
			
		||||
        )
 | 
			
		||||
      : intersection(sourceNode, sourceCenter, targetCenter);
 | 
			
		||||
 | 
			
		||||
    let endPos = isTargetRound
 | 
			
		||||
      ? computeCircleEdgeIntersection(
 | 
			
		||||
          {
 | 
			
		||||
            x: targetNode.x,
 | 
			
		||||
            y: targetNode.y,
 | 
			
		||||
            width: targetNode.width ?? 100,
 | 
			
		||||
            height: targetNode.height ?? 100,
 | 
			
		||||
          },
 | 
			
		||||
          sourceCenter,
 | 
			
		||||
          targetCenter
 | 
			
		||||
        )
 | 
			
		||||
      : intersection(targetNode, targetCenter, sourceCenter);
 | 
			
		||||
 | 
			
		||||
    const midX = (startPos.x + endPos.x) / 2;
 | 
			
		||||
    const midY = (startPos.y + endPos.y) / 2;
 | 
			
		||||
 | 
			
		||||
    const points = [startPos];
 | 
			
		||||
    if (sourceNode.section === 'left') {
 | 
			
		||||
      points.push({
 | 
			
		||||
        x: sourceNode.x - (sourceNode.width ?? 0) / 2 - intersectionShift,
 | 
			
		||||
        y: sourceNode.y,
 | 
			
		||||
      });
 | 
			
		||||
    } else if (sourceNode.section === 'right') {
 | 
			
		||||
      points.push({
 | 
			
		||||
        x: sourceNode.x + (sourceNode.width ?? 0) / 2 + intersectionShift,
 | 
			
		||||
        y: sourceNode.y,
 | 
			
		||||
      });
 | 
			
		||||
    }
 | 
			
		||||
    if (targetNode.section === 'left') {
 | 
			
		||||
      points.push({
 | 
			
		||||
        x: targetNode.x + (targetNode.width ?? 0) / 2 + intersectionShift,
 | 
			
		||||
        y: targetNode.y,
 | 
			
		||||
      });
 | 
			
		||||
    } else if (targetNode.section === 'right') {
 | 
			
		||||
      points.push({
 | 
			
		||||
        x: targetNode.x - (targetNode.width ?? 0) / 2 - intersectionShift,
 | 
			
		||||
        y: targetNode.y,
 | 
			
		||||
      });
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    points.push(endPos);
 | 
			
		||||
 | 
			
		||||
    const secondPoint = points.length > 1 ? points[1] : targetCenter;
 | 
			
		||||
    startPos = isSourceRound
 | 
			
		||||
      ? computeCircleEdgeIntersection(
 | 
			
		||||
          {
 | 
			
		||||
            x: sourceNode.x,
 | 
			
		||||
            y: sourceNode.y,
 | 
			
		||||
            width: sourceNode.width ?? 100,
 | 
			
		||||
            height: sourceNode.height ?? 100,
 | 
			
		||||
          },
 | 
			
		||||
          secondPoint,
 | 
			
		||||
          sourceCenter
 | 
			
		||||
        )
 | 
			
		||||
      : intersection(sourceNode, secondPoint, sourceCenter);
 | 
			
		||||
    points[0] = startPos;
 | 
			
		||||
 | 
			
		||||
    const secondLastPoint = points.length > 1 ? points[points.length - 2] : sourceCenter;
 | 
			
		||||
    endPos = isTargetRound
 | 
			
		||||
      ? computeCircleEdgeIntersection(
 | 
			
		||||
          {
 | 
			
		||||
            x: targetNode.x,
 | 
			
		||||
            y: targetNode.y,
 | 
			
		||||
            width: targetNode.width ?? 100,
 | 
			
		||||
            height: targetNode.height ?? 100,
 | 
			
		||||
          },
 | 
			
		||||
          secondLastPoint,
 | 
			
		||||
          targetCenter
 | 
			
		||||
        )
 | 
			
		||||
      : intersection(targetNode, secondLastPoint, targetCenter);
 | 
			
		||||
    points[points.length - 1] = endPos;
 | 
			
		||||
 | 
			
		||||
    return {
 | 
			
		||||
      id: edge.id,
 | 
			
		||||
      source: edge.start ?? '',
 | 
			
		||||
      target: edge.end ?? '',
 | 
			
		||||
      startX: startPos.x,
 | 
			
		||||
      startY: startPos.y,
 | 
			
		||||
      midX,
 | 
			
		||||
      midY,
 | 
			
		||||
      endX: endPos.x,
 | 
			
		||||
      endY: endPos.y,
 | 
			
		||||
      points,
 | 
			
		||||
      sourceSection: sourceNode?.section,
 | 
			
		||||
      targetSection: targetNode?.section,
 | 
			
		||||
      sourceWidth: sourceNode?.width,
 | 
			
		||||
      sourceHeight: sourceNode?.height,
 | 
			
		||||
      targetWidth: targetNode?.width,
 | 
			
		||||
      targetHeight: targetNode?.height,
 | 
			
		||||
    };
 | 
			
		||||
  });
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Validate layout data structure
 | 
			
		||||
 * @param data - The data to validate
 | 
			
		||||
 * @returns True if data is valid, throws error otherwise
 | 
			
		||||
 */
 | 
			
		||||
export function validateLayoutData(data: LayoutData): boolean {
 | 
			
		||||
  if (!data) {
 | 
			
		||||
    throw new Error('Layout data is required');
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  if (!data.config) {
 | 
			
		||||
    throw new Error('Configuration is required in layout data');
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  if (!Array.isArray(data.nodes)) {
 | 
			
		||||
    throw new Error('Nodes array is required in layout data');
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  if (!Array.isArray(data.edges)) {
 | 
			
		||||
    throw new Error('Edges array is required in layout data');
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  return true;
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										13
									
								
								packages/mermaid-layout-tidy-tree/src/layouts.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										13
									
								
								packages/mermaid-layout-tidy-tree/src/layouts.ts
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,13 @@
 | 
			
		||||
import type { LayoutLoaderDefinition } from 'mermaid';
 | 
			
		||||
 | 
			
		||||
const loader = async () => await import(`./render.js`);
 | 
			
		||||
 | 
			
		||||
const tidyTreeLayout: LayoutLoaderDefinition[] = [
 | 
			
		||||
  {
 | 
			
		||||
    name: 'tidy-tree',
 | 
			
		||||
    loader,
 | 
			
		||||
    algorithm: 'tidy-tree',
 | 
			
		||||
  },
 | 
			
		||||
];
 | 
			
		||||
 | 
			
		||||
export default tidyTreeLayout;
 | 
			
		||||
							
								
								
									
										18
									
								
								packages/mermaid-layout-tidy-tree/src/non-layered-tidy-tree-layout.d.ts
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								packages/mermaid-layout-tidy-tree/src/non-layered-tidy-tree-layout.d.ts
									
									
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,18 @@
 | 
			
		||||
declare module 'non-layered-tidy-tree-layout' {
 | 
			
		||||
  export class BoundingBox {
 | 
			
		||||
    constructor(gap: number, bottomPadding: number);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  export class Layout {
 | 
			
		||||
    constructor(boundingBox: BoundingBox);
 | 
			
		||||
    layout(data: any): {
 | 
			
		||||
      result: any;
 | 
			
		||||
      boundingBox: {
 | 
			
		||||
        left: number;
 | 
			
		||||
        right: number;
 | 
			
		||||
        top: number;
 | 
			
		||||
        bottom: number;
 | 
			
		||||
      };
 | 
			
		||||
    };
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										180
									
								
								packages/mermaid-layout-tidy-tree/src/render.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										180
									
								
								packages/mermaid-layout-tidy-tree/src/render.ts
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,180 @@
 | 
			
		||||
import type { InternalHelpers, LayoutData, RenderOptions, SVG } from 'mermaid';
 | 
			
		||||
import { executeTidyTreeLayout } from './layout.js';
 | 
			
		||||
 | 
			
		||||
interface NodeWithPosition {
 | 
			
		||||
  id: string;
 | 
			
		||||
  x?: number;
 | 
			
		||||
  y?: number;
 | 
			
		||||
  width?: number;
 | 
			
		||||
  height?: number;
 | 
			
		||||
  domId?: any;
 | 
			
		||||
  [key: string]: any;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Render function for bidirectional tidy-tree layout algorithm
 | 
			
		||||
 *
 | 
			
		||||
 * This follows the same pattern as ELK and dagre renderers:
 | 
			
		||||
 * 1. Insert nodes into DOM to get their actual dimensions
 | 
			
		||||
 * 2. Run the bidirectional tidy-tree layout algorithm to calculate positions
 | 
			
		||||
 * 3. Position the nodes and edges based on layout results
 | 
			
		||||
 *
 | 
			
		||||
 * The bidirectional layout creates two trees that grow horizontally in opposite
 | 
			
		||||
 * directions from a central root node:
 | 
			
		||||
 * - Left tree: grows horizontally to the left (children: 1st, 3rd, 5th...)
 | 
			
		||||
 * - Right tree: grows horizontally to the right (children: 2nd, 4th, 6th...)
 | 
			
		||||
 */
 | 
			
		||||
export const render = async (
 | 
			
		||||
  data4Layout: LayoutData,
 | 
			
		||||
  svg: SVG,
 | 
			
		||||
  {
 | 
			
		||||
    insertCluster,
 | 
			
		||||
    insertEdge,
 | 
			
		||||
    insertEdgeLabel,
 | 
			
		||||
    insertMarkers,
 | 
			
		||||
    insertNode,
 | 
			
		||||
    log,
 | 
			
		||||
    positionEdgeLabel,
 | 
			
		||||
  }: InternalHelpers,
 | 
			
		||||
  { algorithm: _algorithm }: RenderOptions
 | 
			
		||||
) => {
 | 
			
		||||
  const nodeDb: Record<string, NodeWithPosition> = {};
 | 
			
		||||
  const clusterDb: Record<string, any> = {};
 | 
			
		||||
 | 
			
		||||
  const element = svg.select('g');
 | 
			
		||||
  insertMarkers(element, data4Layout.markers, data4Layout.type, data4Layout.diagramId);
 | 
			
		||||
 | 
			
		||||
  const subGraphsEl = element.insert('g').attr('class', 'subgraphs');
 | 
			
		||||
  const edgePaths = element.insert('g').attr('class', 'edgePaths');
 | 
			
		||||
  const edgeLabels = element.insert('g').attr('class', 'edgeLabels');
 | 
			
		||||
  const nodes = element.insert('g').attr('class', 'nodes');
 | 
			
		||||
  // Step 1: Insert nodes into DOM to get their actual dimensions
 | 
			
		||||
  log.debug('Inserting nodes into DOM for dimension calculation');
 | 
			
		||||
 | 
			
		||||
  await Promise.all(
 | 
			
		||||
    data4Layout.nodes.map(async (node) => {
 | 
			
		||||
      if (node.isGroup) {
 | 
			
		||||
        const clusterNode: NodeWithPosition = {
 | 
			
		||||
          ...node,
 | 
			
		||||
          id: node.id,
 | 
			
		||||
          width: node.width,
 | 
			
		||||
          height: node.height,
 | 
			
		||||
        };
 | 
			
		||||
        clusterDb[node.id] = clusterNode;
 | 
			
		||||
        nodeDb[node.id] = clusterNode;
 | 
			
		||||
 | 
			
		||||
        await insertCluster(subGraphsEl, node);
 | 
			
		||||
      } else {
 | 
			
		||||
        const nodeWithPosition: NodeWithPosition = {
 | 
			
		||||
          ...node,
 | 
			
		||||
          id: node.id,
 | 
			
		||||
          width: node.width,
 | 
			
		||||
          height: node.height,
 | 
			
		||||
        };
 | 
			
		||||
        nodeDb[node.id] = nodeWithPosition;
 | 
			
		||||
 | 
			
		||||
        const nodeEl = await insertNode(nodes, node, {
 | 
			
		||||
          config: data4Layout.config,
 | 
			
		||||
          dir: data4Layout.direction || 'TB',
 | 
			
		||||
        });
 | 
			
		||||
 | 
			
		||||
        const boundingBox = nodeEl.node()!.getBBox();
 | 
			
		||||
        nodeWithPosition.width = boundingBox.width;
 | 
			
		||||
        nodeWithPosition.height = boundingBox.height;
 | 
			
		||||
        nodeWithPosition.domId = nodeEl;
 | 
			
		||||
 | 
			
		||||
        log.debug(`Node ${node.id} dimensions: ${boundingBox.width}x${boundingBox.height}`);
 | 
			
		||||
      }
 | 
			
		||||
    })
 | 
			
		||||
  );
 | 
			
		||||
  // Step 2: Run the bidirectional tidy-tree layout algorithm
 | 
			
		||||
  log.debug('Running bidirectional tidy-tree layout algorithm');
 | 
			
		||||
 | 
			
		||||
  const updatedLayoutData = {
 | 
			
		||||
    ...data4Layout,
 | 
			
		||||
    nodes: data4Layout.nodes.map((node) => {
 | 
			
		||||
      const nodeWithDimensions = nodeDb[node.id];
 | 
			
		||||
      return {
 | 
			
		||||
        ...node,
 | 
			
		||||
        width: nodeWithDimensions.width ?? node.width ?? 100,
 | 
			
		||||
        height: nodeWithDimensions.height ?? node.height ?? 50,
 | 
			
		||||
      };
 | 
			
		||||
    }),
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  const layoutResult = await executeTidyTreeLayout(updatedLayoutData, data4Layout.config);
 | 
			
		||||
  // Step 3: Position the nodes based on bidirectional layout results
 | 
			
		||||
  log.debug('Positioning nodes based on bidirectional layout results');
 | 
			
		||||
 | 
			
		||||
  layoutResult.nodes.forEach((positionedNode) => {
 | 
			
		||||
    const node = nodeDb[positionedNode.id];
 | 
			
		||||
    if (node?.domId) {
 | 
			
		||||
      // Position the node at the calculated coordinates from bidirectional layout
 | 
			
		||||
      // The layout algorithm has already calculated positions for:
 | 
			
		||||
      // - Root node at center (0, 0)
 | 
			
		||||
      // - Left tree nodes with negative x coordinates (growing left)
 | 
			
		||||
      // - Right tree nodes with positive x coordinates (growing right)
 | 
			
		||||
      node.domId.attr('transform', `translate(${positionedNode.x}, ${positionedNode.y})`);
 | 
			
		||||
      // Store the final position
 | 
			
		||||
      node.x = positionedNode.x;
 | 
			
		||||
      node.y = positionedNode.y;
 | 
			
		||||
      // Step 3: Position the nodes based on bidirectional layout results
 | 
			
		||||
      log.debug(`Positioned node ${node.id} at (${positionedNode.x}, ${positionedNode.y})`);
 | 
			
		||||
    }
 | 
			
		||||
  });
 | 
			
		||||
 | 
			
		||||
  log.debug('Inserting and positioning edges');
 | 
			
		||||
 | 
			
		||||
  await Promise.all(
 | 
			
		||||
    data4Layout.edges.map(async (edge) => {
 | 
			
		||||
      await insertEdgeLabel(edgeLabels, edge);
 | 
			
		||||
 | 
			
		||||
      const startNode = nodeDb[edge.start ?? ''];
 | 
			
		||||
      const endNode = nodeDb[edge.end ?? ''];
 | 
			
		||||
 | 
			
		||||
      if (startNode && endNode) {
 | 
			
		||||
        const positionedEdge = layoutResult.edges.find((e) => e.id === edge.id);
 | 
			
		||||
 | 
			
		||||
        if (positionedEdge) {
 | 
			
		||||
          log.debug('APA01 positionedEdge', positionedEdge);
 | 
			
		||||
          const edgeWithPath = {
 | 
			
		||||
            ...edge,
 | 
			
		||||
            points: positionedEdge.points,
 | 
			
		||||
          };
 | 
			
		||||
          const paths = insertEdge(
 | 
			
		||||
            edgePaths,
 | 
			
		||||
            edgeWithPath,
 | 
			
		||||
            clusterDb,
 | 
			
		||||
            data4Layout.type,
 | 
			
		||||
            startNode,
 | 
			
		||||
            endNode,
 | 
			
		||||
            data4Layout.diagramId
 | 
			
		||||
          );
 | 
			
		||||
 | 
			
		||||
          positionEdgeLabel(edgeWithPath, paths);
 | 
			
		||||
        } else {
 | 
			
		||||
          const edgeWithPath = {
 | 
			
		||||
            ...edge,
 | 
			
		||||
            points: [
 | 
			
		||||
              { x: startNode.x ?? 0, y: startNode.y ?? 0 },
 | 
			
		||||
              { x: endNode.x ?? 0, y: endNode.y ?? 0 },
 | 
			
		||||
            ],
 | 
			
		||||
          };
 | 
			
		||||
 | 
			
		||||
          const paths = insertEdge(
 | 
			
		||||
            edgePaths,
 | 
			
		||||
            edgeWithPath,
 | 
			
		||||
            clusterDb,
 | 
			
		||||
            data4Layout.type,
 | 
			
		||||
            startNode,
 | 
			
		||||
            endNode,
 | 
			
		||||
            data4Layout.diagramId
 | 
			
		||||
          );
 | 
			
		||||
          positionEdgeLabel(edgeWithPath, paths);
 | 
			
		||||
        }
 | 
			
		||||
      }
 | 
			
		||||
    })
 | 
			
		||||
  );
 | 
			
		||||
 | 
			
		||||
  log.debug('Bidirectional tidy-tree rendering completed');
 | 
			
		||||
};
 | 
			
		||||
							
								
								
									
										69
									
								
								packages/mermaid-layout-tidy-tree/src/types.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										69
									
								
								packages/mermaid-layout-tidy-tree/src/types.ts
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,69 @@
 | 
			
		||||
import type { LayoutData } from 'mermaid';
 | 
			
		||||
 | 
			
		||||
export type Node = LayoutData['nodes'][number];
 | 
			
		||||
export type Edge = LayoutData['edges'][number];
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Positioned node after layout calculation
 | 
			
		||||
 */
 | 
			
		||||
export interface PositionedNode {
 | 
			
		||||
  id: string;
 | 
			
		||||
  x: number;
 | 
			
		||||
  y: number;
 | 
			
		||||
  section?: 'root' | 'left' | 'right';
 | 
			
		||||
  width?: number;
 | 
			
		||||
  height?: number;
 | 
			
		||||
  originalNode?: Node;
 | 
			
		||||
  [key: string]: unknown;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Positioned edge after layout calculation
 | 
			
		||||
 */
 | 
			
		||||
export interface PositionedEdge {
 | 
			
		||||
  id: string;
 | 
			
		||||
  source: string;
 | 
			
		||||
  target: string;
 | 
			
		||||
  startX: number;
 | 
			
		||||
  startY: number;
 | 
			
		||||
  midX: number;
 | 
			
		||||
  midY: number;
 | 
			
		||||
  endX: number;
 | 
			
		||||
  endY: number;
 | 
			
		||||
  sourceSection?: 'root' | 'left' | 'right';
 | 
			
		||||
  targetSection?: 'root' | 'left' | 'right';
 | 
			
		||||
  sourceWidth?: number;
 | 
			
		||||
  sourceHeight?: number;
 | 
			
		||||
  targetWidth?: number;
 | 
			
		||||
  targetHeight?: number;
 | 
			
		||||
  [key: string]: unknown;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Result of layout algorithm execution
 | 
			
		||||
 */
 | 
			
		||||
export interface LayoutResult {
 | 
			
		||||
  nodes: PositionedNode[];
 | 
			
		||||
  edges: PositionedEdge[];
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Tidy-tree node structure compatible with non-layered-tidy-tree-layout
 | 
			
		||||
 */
 | 
			
		||||
export interface TidyTreeNode {
 | 
			
		||||
  id: string | number;
 | 
			
		||||
  width: number;
 | 
			
		||||
  height: number;
 | 
			
		||||
  x?: number;
 | 
			
		||||
  y?: number;
 | 
			
		||||
  children?: TidyTreeNode[];
 | 
			
		||||
  _originalNode?: Node;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Tidy-tree layout configuration
 | 
			
		||||
 */
 | 
			
		||||
export interface TidyTreeLayoutConfig {
 | 
			
		||||
  gap: number;
 | 
			
		||||
  bottomPadding: number;
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										10
									
								
								packages/mermaid-layout-tidy-tree/tsconfig.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								packages/mermaid-layout-tidy-tree/tsconfig.json
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,10 @@
 | 
			
		||||
{
 | 
			
		||||
  "extends": "../../tsconfig.json",
 | 
			
		||||
  "compilerOptions": {
 | 
			
		||||
    "rootDir": "./src",
 | 
			
		||||
    "outDir": "./dist",
 | 
			
		||||
    "types": ["vitest/importMeta", "vitest/globals"]
 | 
			
		||||
  },
 | 
			
		||||
  "include": ["./src/**/*.ts", "./src/**/*.d.ts"],
 | 
			
		||||
  "typeRoots": ["./src/types"]
 | 
			
		||||
}
 | 
			
		||||
@@ -87,7 +87,6 @@
 | 
			
		||||
### Minor Changes
 | 
			
		||||
 | 
			
		||||
- [#6408](https://github.com/mermaid-js/mermaid/pull/6408) [`ad65313`](https://github.com/mermaid-js/mermaid/commit/ad653138e16765d095613a6e5de86dc5e52ac8f0) Thanks [@ashishjain0512](https://github.com/ashishjain0512)! - fix: restore curve type configuration functionality for flowcharts. This fixes the issue where curve type settings were not being applied when configured through any of the following methods:
 | 
			
		||||
 | 
			
		||||
  - Config
 | 
			
		||||
  - Init directive (%%{ init: { 'flowchart': { 'curve': '...' } } }%%)
 | 
			
		||||
  - LinkStyle command (linkStyle default interpolate ...)
 | 
			
		||||
@@ -106,14 +105,12 @@
 | 
			
		||||
### Minor Changes
 | 
			
		||||
 | 
			
		||||
- [#6187](https://github.com/mermaid-js/mermaid/pull/6187) [`7809b5a`](https://github.com/mermaid-js/mermaid/commit/7809b5a93fae127f45727071f5ff14325222c518) Thanks [@ashishjain0512](https://github.com/ashishjain0512)! - Flowchart new syntax for node metadata bugs
 | 
			
		||||
 | 
			
		||||
  - Incorrect label mapping for nodes when using `&`
 | 
			
		||||
  - Syntax error when `}` with trailing spaces before new line
 | 
			
		||||
 | 
			
		||||
- [#6136](https://github.com/mermaid-js/mermaid/pull/6136) [`ec0d9c3`](https://github.com/mermaid-js/mermaid/commit/ec0d9c389aa6018043187654044c1e0b5aa4f600) Thanks [@knsv](https://github.com/knsv)! - Adding support for animation of flowchart edges
 | 
			
		||||
 | 
			
		||||
- [#6373](https://github.com/mermaid-js/mermaid/pull/6373) [`05bdf0e`](https://github.com/mermaid-js/mermaid/commit/05bdf0e20e2629fe77513218fbd4e28e65f75882) Thanks [@ashishjain0512](https://github.com/ashishjain0512)! - Upgrade Requirement and ER diagram to use the common renderer flow
 | 
			
		||||
 | 
			
		||||
  - Added support for directions
 | 
			
		||||
  - Added support for hand drawn look
 | 
			
		||||
 | 
			
		||||
@@ -162,7 +159,6 @@
 | 
			
		||||
- [#5999](https://github.com/mermaid-js/mermaid/pull/5999) [`742ad7c`](https://github.com/mermaid-js/mermaid/commit/742ad7c130964df1fb5544e909d9556081285f68) Thanks [@knsv](https://github.com/knsv)! - Adding Kanban board, a new diagram type
 | 
			
		||||
 | 
			
		||||
- [#5880](https://github.com/mermaid-js/mermaid/pull/5880) [`bdf145f`](https://github.com/mermaid-js/mermaid/commit/bdf145ffe362462176d9c1e68d5f3ff5c9d962b0) Thanks [@yari-dewalt](https://github.com/yari-dewalt)! - Class diagram changes:
 | 
			
		||||
 | 
			
		||||
  - Updates the class diagram to the new unified way of rendering.
 | 
			
		||||
  - Includes a new "classBox" shape to be used in diagrams
 | 
			
		||||
  - Other updates such as:
 | 
			
		||||
 
 | 
			
		||||
@@ -82,7 +82,7 @@
 | 
			
		||||
    "katex": "^0.16.22",
 | 
			
		||||
    "khroma": "^2.1.0",
 | 
			
		||||
    "lodash-es": "^4.17.21",
 | 
			
		||||
    "marked": "^16.0.0",
 | 
			
		||||
    "marked": "^15.0.7",
 | 
			
		||||
    "roughjs": "^4.6.6",
 | 
			
		||||
    "stylis": "^4.3.6",
 | 
			
		||||
    "ts-dedent": "^2.2.0",
 | 
			
		||||
@@ -123,8 +123,8 @@
 | 
			
		||||
    "rimraf": "^6.0.1",
 | 
			
		||||
    "start-server-and-test": "^2.0.10",
 | 
			
		||||
    "type-fest": "^4.35.0",
 | 
			
		||||
    "typedoc": "^0.27.8",
 | 
			
		||||
    "typedoc-plugin-markdown": "^4.4.2",
 | 
			
		||||
    "typedoc": "^0.28.9",
 | 
			
		||||
    "typedoc-plugin-markdown": "^4.8.0",
 | 
			
		||||
    "typescript": "~5.7.3",
 | 
			
		||||
    "unist-util-flatmap": "^1.0.0",
 | 
			
		||||
    "unist-util-visit": "^5.0.0",
 | 
			
		||||
 
 | 
			
		||||
@@ -171,7 +171,9 @@ This Markdown should be kept.
 | 
			
		||||
      expect(buildShapeDoc()).toMatchInlineSnapshot(`
 | 
			
		||||
        "| **Semantic Name**                 | **Shape Name**         | **Short Name** | **Description**                | **Alias Supported**                                              |
 | 
			
		||||
        | --------------------------------- | ---------------------- | -------------- | ------------------------------ | ---------------------------------------------------------------- |
 | 
			
		||||
        | Bang                              | Bang                   | \`bang\`         | Bang                           | \`bang\`                                                           |
 | 
			
		||||
        | Card                              | Notched Rectangle      | \`notch-rect\`   | Represents a card              | \`card\`, \`notched-rectangle\`                                      |
 | 
			
		||||
        | Cloud                             | Cloud                  | \`cloud\`        | cloud                          | \`cloud\`                                                          |
 | 
			
		||||
        | Collate                           | Hourglass              | \`hourglass\`    | Represents a collate operation | \`collate\`, \`hourglass\`                                           |
 | 
			
		||||
        | Com Link                          | Lightning Bolt         | \`bolt\`         | Communication link             | \`com-link\`, \`lightning-bolt\`                                     |
 | 
			
		||||
        | Comment                           | Curly Brace            | \`brace\`        | Adds a comment                 | \`brace-l\`, \`comment\`                                             |
 | 
			
		||||
 
 | 
			
		||||
@@ -1075,6 +1075,10 @@ export interface ArchitectureDiagramConfig extends BaseDiagramConfig {
 | 
			
		||||
export interface MindmapDiagramConfig extends BaseDiagramConfig {
 | 
			
		||||
  padding?: number;
 | 
			
		||||
  maxNodeWidth?: number;
 | 
			
		||||
  /**
 | 
			
		||||
   * Layout algorithm to use for positioning mindmap nodes
 | 
			
		||||
   */
 | 
			
		||||
  layoutAlgorithm?: string;
 | 
			
		||||
}
 | 
			
		||||
/**
 | 
			
		||||
 * The object containing configurations specific for kanban diagrams
 | 
			
		||||
 
 | 
			
		||||
@@ -1,5 +1,3 @@
 | 
			
		||||
// tests to check that comments are removed
 | 
			
		||||
 | 
			
		||||
import { cleanupComments } from './comments.js';
 | 
			
		||||
import { describe, it, expect } from 'vitest';
 | 
			
		||||
 | 
			
		||||
@@ -10,12 +8,12 @@ describe('comments', () => {
 | 
			
		||||
%% This is a comment
 | 
			
		||||
%% This is another comment
 | 
			
		||||
graph TD
 | 
			
		||||
	A-->B
 | 
			
		||||
    A-->B
 | 
			
		||||
%% This is a comment
 | 
			
		||||
`;
 | 
			
		||||
    expect(cleanupComments(text)).toMatchInlineSnapshot(`
 | 
			
		||||
      "graph TD
 | 
			
		||||
      	A-->B
 | 
			
		||||
          A-->B
 | 
			
		||||
      "
 | 
			
		||||
    `);
 | 
			
		||||
  });
 | 
			
		||||
@@ -29,9 +27,9 @@ graph TD
 | 
			
		||||
%%{ init: {'theme': 'space before init'}}%%
 | 
			
		||||
%%{init: {'theme': 'space after ending'}}%%
 | 
			
		||||
graph TD
 | 
			
		||||
	A-->B
 | 
			
		||||
    A-->B
 | 
			
		||||
 | 
			
		||||
	B-->C
 | 
			
		||||
    B-->C
 | 
			
		||||
%% This is a comment
 | 
			
		||||
`;
 | 
			
		||||
    expect(cleanupComments(text)).toMatchInlineSnapshot(`
 | 
			
		||||
@@ -39,9 +37,9 @@ graph TD
 | 
			
		||||
      %%{ init: {'theme': 'space before init'}}%%
 | 
			
		||||
      %%{init: {'theme': 'space after ending'}}%%
 | 
			
		||||
      graph TD
 | 
			
		||||
      	A-->B
 | 
			
		||||
          A-->B
 | 
			
		||||
 | 
			
		||||
      	B-->C
 | 
			
		||||
          B-->C
 | 
			
		||||
      "
 | 
			
		||||
    `);
 | 
			
		||||
  });
 | 
			
		||||
@@ -50,14 +48,14 @@ graph TD
 | 
			
		||||
    const text = `
 | 
			
		||||
%% This is a comment
 | 
			
		||||
graph TD
 | 
			
		||||
	A-->B
 | 
			
		||||
	%% This is a comment
 | 
			
		||||
	C-->D
 | 
			
		||||
    A-->B
 | 
			
		||||
    %% This is a comment
 | 
			
		||||
    C-->D
 | 
			
		||||
`;
 | 
			
		||||
    expect(cleanupComments(text)).toMatchInlineSnapshot(`
 | 
			
		||||
      "graph TD
 | 
			
		||||
	A-->B
 | 
			
		||||
	C-->D
 | 
			
		||||
          A-->B
 | 
			
		||||
          C-->D
 | 
			
		||||
      "
 | 
			
		||||
    `);
 | 
			
		||||
  });
 | 
			
		||||
@@ -70,11 +68,11 @@ graph TD
 | 
			
		||||
 | 
			
		||||
%% This is a comment
 | 
			
		||||
graph TD
 | 
			
		||||
	A-->B
 | 
			
		||||
    A-->B
 | 
			
		||||
`;
 | 
			
		||||
    expect(cleanupComments(text)).toMatchInlineSnapshot(`
 | 
			
		||||
      "graph TD
 | 
			
		||||
      	A-->B
 | 
			
		||||
          A-->B
 | 
			
		||||
      "
 | 
			
		||||
    `);
 | 
			
		||||
  });
 | 
			
		||||
@@ -82,12 +80,12 @@ graph TD
 | 
			
		||||
  it('should remove comments at end of text with no newline', () => {
 | 
			
		||||
    const text = `
 | 
			
		||||
graph TD
 | 
			
		||||
	A-->B
 | 
			
		||||
    A-->B
 | 
			
		||||
%% This is a comment`;
 | 
			
		||||
 | 
			
		||||
    expect(cleanupComments(text)).toMatchInlineSnapshot(`
 | 
			
		||||
      "graph TD
 | 
			
		||||
	A-->B
 | 
			
		||||
          A-->B
 | 
			
		||||
      "
 | 
			
		||||
    `);
 | 
			
		||||
  });
 | 
			
		||||
 
 | 
			
		||||
@@ -4,5 +4,6 @@
 | 
			
		||||
 * @returns cleaned text
 | 
			
		||||
 */
 | 
			
		||||
export const cleanupComments = (text: string): string => {
 | 
			
		||||
  return text.replace(/^\s*%%(?!{)[^\n]+\n?/gm, '').trimStart();
 | 
			
		||||
  const cleaned = text.replace(/^\s*%%(?!{)[^\n]+\n?/gm, '');
 | 
			
		||||
  return cleaned.trimStart();
 | 
			
		||||
};
 | 
			
		||||
 
 | 
			
		||||
@@ -1,6 +1,5 @@
 | 
			
		||||
import type { Position } from 'cytoscape';
 | 
			
		||||
import type { LayoutOptions, Position } from 'cytoscape';
 | 
			
		||||
import cytoscape from 'cytoscape';
 | 
			
		||||
import type { FcoseLayoutOptions } from 'cytoscape-fcose';
 | 
			
		||||
import fcose from 'cytoscape-fcose';
 | 
			
		||||
import { select } from 'd3';
 | 
			
		||||
import type { DrawDefinition, SVG } from '../../diagram-api/types.js';
 | 
			
		||||
@@ -41,7 +40,7 @@ registerIconPacks([
 | 
			
		||||
    icons: architectureIcons,
 | 
			
		||||
  },
 | 
			
		||||
]);
 | 
			
		||||
cytoscape.use(fcose);
 | 
			
		||||
cytoscape.use(fcose as any);
 | 
			
		||||
 | 
			
		||||
function addServices(services: ArchitectureService[], cy: cytoscape.Core, db: ArchitectureDB) {
 | 
			
		||||
  services.forEach((service) => {
 | 
			
		||||
@@ -429,7 +428,7 @@ function layoutArchitecture(
 | 
			
		||||
      },
 | 
			
		||||
      alignmentConstraint,
 | 
			
		||||
      relativePlacementConstraint,
 | 
			
		||||
    } as FcoseLayoutOptions);
 | 
			
		||||
    } as LayoutOptions);
 | 
			
		||||
 | 
			
		||||
    // Once the diagram has been generated and the service's position cords are set, adjust the XY edges to have a 90deg bend
 | 
			
		||||
    layout.one('layoutstop', () => {
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										297
									
								
								packages/mermaid/src/diagrams/mindmap/mindmapDb.getData.test.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										297
									
								
								packages/mermaid/src/diagrams/mindmap/mindmapDb.getData.test.ts
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,297 @@
 | 
			
		||||
import { describe, it, expect, beforeEach, vi } from 'vitest';
 | 
			
		||||
import { MindmapDB } from './mindmapDb.js';
 | 
			
		||||
import type { MindmapLayoutNode, MindmapLayoutEdge } from './mindmapDb.js';
 | 
			
		||||
import type { Edge } from '../../rendering-util/types.js';
 | 
			
		||||
 | 
			
		||||
// Mock the getConfig function
 | 
			
		||||
vi.mock('../../diagram-api/diagramAPI.js', () => ({
 | 
			
		||||
  getConfig: vi.fn(() => ({
 | 
			
		||||
    mindmap: {
 | 
			
		||||
      layoutAlgorithm: 'cose-bilkent',
 | 
			
		||||
      padding: 10,
 | 
			
		||||
      maxNodeWidth: 200,
 | 
			
		||||
      useMaxWidth: true,
 | 
			
		||||
    },
 | 
			
		||||
  })),
 | 
			
		||||
}));
 | 
			
		||||
 | 
			
		||||
describe('MindmapDb getData function', () => {
 | 
			
		||||
  let db: MindmapDB;
 | 
			
		||||
 | 
			
		||||
  beforeEach(() => {
 | 
			
		||||
    db = new MindmapDB();
 | 
			
		||||
    // Clear the database before each test
 | 
			
		||||
    db.clear();
 | 
			
		||||
  });
 | 
			
		||||
 | 
			
		||||
  describe('getData', () => {
 | 
			
		||||
    it('should return empty data when no mindmap is set', () => {
 | 
			
		||||
      const result = db.getData();
 | 
			
		||||
 | 
			
		||||
      expect(result.nodes).toEqual([]);
 | 
			
		||||
      expect(result.edges).toEqual([]);
 | 
			
		||||
      expect(result.config).toBeDefined();
 | 
			
		||||
      expect(result.rootNode).toBeUndefined();
 | 
			
		||||
    });
 | 
			
		||||
 | 
			
		||||
    it('should return structured data for simple mindmap', () => {
 | 
			
		||||
      // Create a simple mindmap structure
 | 
			
		||||
      db.addNode(0, 'root', 'Root Node', 0);
 | 
			
		||||
      db.addNode(1, 'child1', 'Child 1', 0);
 | 
			
		||||
      db.addNode(1, 'child2', 'Child 2', 0);
 | 
			
		||||
 | 
			
		||||
      const result = db.getData();
 | 
			
		||||
 | 
			
		||||
      expect(result.nodes).toHaveLength(3);
 | 
			
		||||
      expect(result.edges).toHaveLength(2);
 | 
			
		||||
      expect(result.config).toBeDefined();
 | 
			
		||||
      expect(result.rootNode).toBeDefined();
 | 
			
		||||
 | 
			
		||||
      // Check root node
 | 
			
		||||
      const rootNode = (result.nodes as MindmapLayoutNode[]).find((n) => n.id === '0');
 | 
			
		||||
      expect(rootNode).toBeDefined();
 | 
			
		||||
      expect(rootNode?.label).toBe('Root Node');
 | 
			
		||||
      expect(rootNode?.level).toBe(0);
 | 
			
		||||
 | 
			
		||||
      // Check child nodes
 | 
			
		||||
      const child1 = (result.nodes as MindmapLayoutNode[]).find((n) => n.id === '1');
 | 
			
		||||
      expect(child1).toBeDefined();
 | 
			
		||||
      expect(child1?.label).toBe('Child 1');
 | 
			
		||||
      expect(child1?.level).toBe(1);
 | 
			
		||||
 | 
			
		||||
      // Check edges
 | 
			
		||||
      expect(result.edges).toContainEqual(
 | 
			
		||||
        expect.objectContaining({
 | 
			
		||||
          start: '0',
 | 
			
		||||
          end: '1',
 | 
			
		||||
          depth: 0,
 | 
			
		||||
        })
 | 
			
		||||
      );
 | 
			
		||||
    });
 | 
			
		||||
 | 
			
		||||
    it('should return structured data for hierarchical mindmap', () => {
 | 
			
		||||
      // Create a hierarchical mindmap structure
 | 
			
		||||
      db.addNode(0, 'root', 'Root Node', 0);
 | 
			
		||||
      db.addNode(1, 'child1', 'Child 1', 0);
 | 
			
		||||
      db.addNode(2, 'grandchild1', 'Grandchild 1', 0);
 | 
			
		||||
      db.addNode(2, 'grandchild2', 'Grandchild 2', 0);
 | 
			
		||||
      db.addNode(1, 'child2', 'Child 2', 0);
 | 
			
		||||
 | 
			
		||||
      const result = db.getData();
 | 
			
		||||
 | 
			
		||||
      expect(result.nodes).toHaveLength(5);
 | 
			
		||||
      expect(result.edges).toHaveLength(4);
 | 
			
		||||
 | 
			
		||||
      // Check that all levels are represented
 | 
			
		||||
      const levels = result.nodes.map((n) => (n as MindmapLayoutNode).level);
 | 
			
		||||
      expect(levels).toContain(0); // root
 | 
			
		||||
      expect(levels).toContain(1); // children
 | 
			
		||||
      expect(levels).toContain(2); // grandchildren
 | 
			
		||||
 | 
			
		||||
      // Check edge relationships
 | 
			
		||||
      const edgeRelations = result.edges.map(
 | 
			
		||||
        (e) => `${(e as MindmapLayoutEdge).start}->${(e as MindmapLayoutEdge).end}`
 | 
			
		||||
      );
 | 
			
		||||
      expect(edgeRelations).toContain('0->1'); // root to child1
 | 
			
		||||
      expect(edgeRelations).toContain('1->2'); // child1 to grandchild1
 | 
			
		||||
      expect(edgeRelations).toContain('1->3'); // child1 to grandchild2
 | 
			
		||||
      expect(edgeRelations).toContain('0->4'); // root to child2
 | 
			
		||||
    });
 | 
			
		||||
 | 
			
		||||
    it('should preserve node properties in processed data', () => {
 | 
			
		||||
      // Add a node with specific properties
 | 
			
		||||
      db.addNode(0, 'root', 'Root Node', 2); // type 2 = rectangle
 | 
			
		||||
 | 
			
		||||
      // Set additional properties
 | 
			
		||||
      const mindmap = db.getMindmap();
 | 
			
		||||
      if (mindmap) {
 | 
			
		||||
        mindmap.width = 150;
 | 
			
		||||
        mindmap.height = 75;
 | 
			
		||||
        mindmap.padding = 15;
 | 
			
		||||
        mindmap.section = 1;
 | 
			
		||||
        mindmap.class = 'custom-class';
 | 
			
		||||
        mindmap.icon = 'star';
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      const result = db.getData();
 | 
			
		||||
 | 
			
		||||
      expect(result.nodes).toHaveLength(1);
 | 
			
		||||
      const node = result.nodes[0] as MindmapLayoutNode;
 | 
			
		||||
 | 
			
		||||
      expect(node.type).toBe(2);
 | 
			
		||||
      expect(node.width).toBe(150);
 | 
			
		||||
      expect(node.height).toBe(75);
 | 
			
		||||
      expect(node.padding).toBe(15);
 | 
			
		||||
      expect(node.section).toBeUndefined(); // Root node has undefined section
 | 
			
		||||
      expect(node.cssClasses).toBe('mindmap-node section-root section--1 custom-class');
 | 
			
		||||
      expect(node.icon).toBe('star');
 | 
			
		||||
    });
 | 
			
		||||
 | 
			
		||||
    it('should generate unique edge IDs', () => {
 | 
			
		||||
      db.addNode(0, 'root', 'Root Node', 0);
 | 
			
		||||
      db.addNode(1, 'child1', 'Child 1', 0);
 | 
			
		||||
      db.addNode(1, 'child2', 'Child 2', 0);
 | 
			
		||||
      db.addNode(1, 'child3', 'Child 3', 0);
 | 
			
		||||
 | 
			
		||||
      const result = db.getData();
 | 
			
		||||
 | 
			
		||||
      const edgeIds = result.edges.map((e: Edge) => e.id);
 | 
			
		||||
      const uniqueIds = new Set(edgeIds);
 | 
			
		||||
 | 
			
		||||
      expect(edgeIds).toHaveLength(3);
 | 
			
		||||
      expect(uniqueIds.size).toBe(3); // All IDs should be unique
 | 
			
		||||
    });
 | 
			
		||||
 | 
			
		||||
    it('should handle nodes with missing optional properties', () => {
 | 
			
		||||
      db.addNode(0, 'root', 'Root Node', 0);
 | 
			
		||||
 | 
			
		||||
      const result = db.getData();
 | 
			
		||||
      const node = result.nodes[0] as MindmapLayoutNode;
 | 
			
		||||
 | 
			
		||||
      // Should handle undefined/missing properties gracefully
 | 
			
		||||
      expect(node.section).toBeUndefined(); // Root node has undefined section
 | 
			
		||||
      expect(node.cssClasses).toBe('mindmap-node section-root section--1'); // Root node gets special classes
 | 
			
		||||
      expect(node.icon).toBeUndefined();
 | 
			
		||||
      expect(node.x).toBeUndefined();
 | 
			
		||||
      expect(node.y).toBeUndefined();
 | 
			
		||||
    });
 | 
			
		||||
 | 
			
		||||
    it('should assign correct section classes based on sibling position', () => {
 | 
			
		||||
      // Create the example mindmap structure:
 | 
			
		||||
      // A
 | 
			
		||||
      //   a0
 | 
			
		||||
      //     aa0
 | 
			
		||||
      //   a1
 | 
			
		||||
      //     aaa
 | 
			
		||||
      //   a2
 | 
			
		||||
      db.addNode(0, 'A', 'A', 0); // Root
 | 
			
		||||
      db.addNode(1, 'a0', 'a0', 0); // First child of root
 | 
			
		||||
      db.addNode(2, 'aa0', 'aa0', 0); // Child of a0
 | 
			
		||||
      db.addNode(1, 'a1', 'a1', 0); // Second child of root
 | 
			
		||||
      db.addNode(2, 'aaa', 'aaa', 0); // Child of a1
 | 
			
		||||
      db.addNode(1, 'a2', 'a2', 0); // Third child of root
 | 
			
		||||
 | 
			
		||||
      const result = db.getData();
 | 
			
		||||
 | 
			
		||||
      // Find nodes by their labels
 | 
			
		||||
      const nodeA = result.nodes.find((n) => n.label === 'A') as MindmapLayoutNode;
 | 
			
		||||
      const nodeA0 = result.nodes.find((n) => n.label === 'a0') as MindmapLayoutNode;
 | 
			
		||||
      const nodeAa0 = result.nodes.find((n) => n.label === 'aa0') as MindmapLayoutNode;
 | 
			
		||||
      const nodeA1 = result.nodes.find((n) => n.label === 'a1') as MindmapLayoutNode;
 | 
			
		||||
      const nodeAaa = result.nodes.find((n) => n.label === 'aaa') as MindmapLayoutNode;
 | 
			
		||||
      const nodeA2 = result.nodes.find((n) => n.label === 'a2') as MindmapLayoutNode;
 | 
			
		||||
 | 
			
		||||
      // Check section assignments
 | 
			
		||||
      expect(nodeA.section).toBeUndefined(); // Root has undefined section
 | 
			
		||||
      expect(nodeA0.section).toBe(0); // First child of root
 | 
			
		||||
      expect(nodeAa0.section).toBe(0); // Inherits from parent a0
 | 
			
		||||
      expect(nodeA1.section).toBe(1); // Second child of root
 | 
			
		||||
      expect(nodeAaa.section).toBe(1); // Inherits from parent a1
 | 
			
		||||
      expect(nodeA2.section).toBe(2); // Third child of root
 | 
			
		||||
 | 
			
		||||
      // Check CSS classes
 | 
			
		||||
      expect(nodeA.cssClasses).toBe('mindmap-node section-root section--1');
 | 
			
		||||
      expect(nodeA0.cssClasses).toBe('mindmap-node section-0');
 | 
			
		||||
      expect(nodeAa0.cssClasses).toBe('mindmap-node section-0');
 | 
			
		||||
      expect(nodeA1.cssClasses).toBe('mindmap-node section-1');
 | 
			
		||||
      expect(nodeAaa.cssClasses).toBe('mindmap-node section-1');
 | 
			
		||||
      expect(nodeA2.cssClasses).toBe('mindmap-node section-2');
 | 
			
		||||
    });
 | 
			
		||||
 | 
			
		||||
    it('should preserve custom classes while adding section classes', () => {
 | 
			
		||||
      db.addNode(0, 'root', 'Root Node', 0);
 | 
			
		||||
      db.addNode(1, 'child', 'Child Node', 0);
 | 
			
		||||
 | 
			
		||||
      // Add custom classes to nodes
 | 
			
		||||
      const mindmap = db.getMindmap();
 | 
			
		||||
      if (mindmap) {
 | 
			
		||||
        mindmap.class = 'custom-root-class';
 | 
			
		||||
        if (mindmap.children?.[0]) {
 | 
			
		||||
          mindmap.children[0].class = 'custom-child-class';
 | 
			
		||||
        }
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      const result = db.getData();
 | 
			
		||||
      const rootNode = result.nodes.find((n) => n.label === 'Root Node') as MindmapLayoutNode;
 | 
			
		||||
      const childNode = result.nodes.find((n) => n.label === 'Child Node') as MindmapLayoutNode;
 | 
			
		||||
 | 
			
		||||
      // Should include both section classes and custom classes
 | 
			
		||||
      expect(rootNode.cssClasses).toBe('mindmap-node section-root section--1 custom-root-class');
 | 
			
		||||
      expect(childNode.cssClasses).toBe('mindmap-node section-0 custom-child-class');
 | 
			
		||||
    });
 | 
			
		||||
 | 
			
		||||
    it('should not create any fake root nodes', () => {
 | 
			
		||||
      // Create a simple mindmap
 | 
			
		||||
      db.addNode(0, 'A', 'A', 0);
 | 
			
		||||
      db.addNode(1, 'a0', 'a0', 0);
 | 
			
		||||
      db.addNode(1, 'a1', 'a1', 0);
 | 
			
		||||
 | 
			
		||||
      const result = db.getData();
 | 
			
		||||
 | 
			
		||||
      // Check that we only have the expected nodes
 | 
			
		||||
      expect(result.nodes).toHaveLength(3);
 | 
			
		||||
      expect(result.nodes.map((n) => n.label)).toEqual(['A', 'a0', 'a1']);
 | 
			
		||||
 | 
			
		||||
      // Check that there's no node with label "mindmap" or any other fake root
 | 
			
		||||
      const mindmapNode = result.nodes.find((n) => n.label === 'mindmap');
 | 
			
		||||
      expect(mindmapNode).toBeUndefined();
 | 
			
		||||
 | 
			
		||||
      // Verify the root node has the correct classes
 | 
			
		||||
      const rootNode = result.nodes.find((n) => n.label === 'A') as MindmapLayoutNode;
 | 
			
		||||
      expect(rootNode.cssClasses).toBe('mindmap-node section-root section--1');
 | 
			
		||||
      expect(rootNode.level).toBe(0);
 | 
			
		||||
    });
 | 
			
		||||
 | 
			
		||||
    it('should assign correct section classes to edges', () => {
 | 
			
		||||
      // Create the example mindmap structure:
 | 
			
		||||
      // A
 | 
			
		||||
      //   a0
 | 
			
		||||
      //     aa0
 | 
			
		||||
      //   a1
 | 
			
		||||
      //     aaa
 | 
			
		||||
      //   a2
 | 
			
		||||
      db.addNode(0, 'A', 'A', 0); // Root
 | 
			
		||||
      db.addNode(1, 'a0', 'a0', 0); // First child of root
 | 
			
		||||
      db.addNode(2, 'aa0', 'aa0', 0); // Child of a0
 | 
			
		||||
      db.addNode(1, 'a1', 'a1', 0); // Second child of root
 | 
			
		||||
      db.addNode(2, 'aaa', 'aaa', 0); // Child of a1
 | 
			
		||||
      db.addNode(1, 'a2', 'a2', 0); // Third child of root
 | 
			
		||||
 | 
			
		||||
      const result = db.getData();
 | 
			
		||||
 | 
			
		||||
      // Should have 5 edges: A->a0, a0->aa0, A->a1, a1->aaa, A->a2
 | 
			
		||||
      expect(result.edges).toHaveLength(5);
 | 
			
		||||
 | 
			
		||||
      // Find edges by their start and end nodes
 | 
			
		||||
      const edgeA_a0 = result.edges.find(
 | 
			
		||||
        (e) => e.start === '0' && e.end === '1'
 | 
			
		||||
      ) as MindmapLayoutEdge;
 | 
			
		||||
      const edgeA0_aa0 = result.edges.find(
 | 
			
		||||
        (e) => e.start === '1' && e.end === '2'
 | 
			
		||||
      ) as MindmapLayoutEdge;
 | 
			
		||||
      const edgeA_a1 = result.edges.find(
 | 
			
		||||
        (e) => e.start === '0' && e.end === '3'
 | 
			
		||||
      ) as MindmapLayoutEdge;
 | 
			
		||||
      const edgeA1_aaa = result.edges.find(
 | 
			
		||||
        (e) => e.start === '3' && e.end === '4'
 | 
			
		||||
      ) as MindmapLayoutEdge;
 | 
			
		||||
      const edgeA_a2 = result.edges.find(
 | 
			
		||||
        (e) => e.start === '0' && e.end === '5'
 | 
			
		||||
      ) as MindmapLayoutEdge;
 | 
			
		||||
 | 
			
		||||
      // Check edge classes
 | 
			
		||||
      expect(edgeA_a0.classes).toBe('edge section-edge-0 edge-depth-1'); // A->a0: section-0, depth-1
 | 
			
		||||
      expect(edgeA0_aa0.classes).toBe('edge section-edge-0 edge-depth-2'); // a0->aa0: section-0, depth-2
 | 
			
		||||
      expect(edgeA_a1.classes).toBe('edge section-edge-1 edge-depth-1'); // A->a1: section-1, depth-1
 | 
			
		||||
      expect(edgeA1_aaa.classes).toBe('edge section-edge-1 edge-depth-2'); // a1->aaa: section-1, depth-2
 | 
			
		||||
      expect(edgeA_a2.classes).toBe('edge section-edge-2 edge-depth-1'); // A->a2: section-2, depth-1
 | 
			
		||||
 | 
			
		||||
      // Check section assignments match the child nodes
 | 
			
		||||
      expect(edgeA_a0.section).toBe(0);
 | 
			
		||||
      expect(edgeA0_aa0.section).toBe(0);
 | 
			
		||||
      expect(edgeA_a1.section).toBe(1);
 | 
			
		||||
      expect(edgeA1_aaa.section).toBe(1);
 | 
			
		||||
      expect(edgeA_a2.section).toBe(2);
 | 
			
		||||
    });
 | 
			
		||||
  });
 | 
			
		||||
});
 | 
			
		||||
@@ -5,6 +5,22 @@ import { log } from '../../logger.js';
 | 
			
		||||
import type { MindmapNode } from './mindmapTypes.js';
 | 
			
		||||
import defaultConfig from '../../defaultConfig.js';
 | 
			
		||||
 | 
			
		||||
import type { LayoutData, Node, Edge } from '../../rendering-util/types.js';
 | 
			
		||||
 | 
			
		||||
// Extend Node type for mindmap-specific properties
 | 
			
		||||
export type MindmapLayoutNode = Node & {
 | 
			
		||||
  level: number;
 | 
			
		||||
  nodeId: string;
 | 
			
		||||
  type: number;
 | 
			
		||||
  section?: number;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
// Extend Edge type for mindmap-specific properties
 | 
			
		||||
export type MindmapLayoutEdge = Edge & {
 | 
			
		||||
  depth: number;
 | 
			
		||||
  section?: number;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
const nodeType = {
 | 
			
		||||
  DEFAULT: 0,
 | 
			
		||||
  NO_BORDER: 0,
 | 
			
		||||
@@ -27,7 +43,6 @@ export class MindmapDB {
 | 
			
		||||
    this.nodeType = nodeType;
 | 
			
		||||
    this.clear();
 | 
			
		||||
    this.getType = this.getType.bind(this);
 | 
			
		||||
    this.getMindmap = this.getMindmap.bind(this);
 | 
			
		||||
    this.getElementById = this.getElementById.bind(this);
 | 
			
		||||
    this.getParent = this.getParent.bind(this);
 | 
			
		||||
    this.getMindmap = this.getMindmap.bind(this);
 | 
			
		||||
@@ -156,6 +171,214 @@ export class MindmapDB {
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  /**
 | 
			
		||||
   * Assign section numbers to nodes based on their position relative to root
 | 
			
		||||
   * @param node - The mindmap node to process
 | 
			
		||||
   * @param sectionNumber - The section number to assign (undefined for root)
 | 
			
		||||
   */
 | 
			
		||||
  public assignSections(node: MindmapNode, sectionNumber?: number): void {
 | 
			
		||||
    // For root node, section should be undefined (not -1)
 | 
			
		||||
    if (node.level === 0) {
 | 
			
		||||
      node.section = undefined;
 | 
			
		||||
    } else {
 | 
			
		||||
      // For non-root nodes, assign the section number
 | 
			
		||||
      node.section = sectionNumber;
 | 
			
		||||
    }
 | 
			
		||||
    // For root node's children, assign section numbers based on their index
 | 
			
		||||
    // For other nodes, inherit parent's section number
 | 
			
		||||
    if (node.children) {
 | 
			
		||||
      for (const [index, child] of node.children.entries()) {
 | 
			
		||||
        const childSectionNumber = node.level === 0 ? index : sectionNumber;
 | 
			
		||||
        this.assignSections(child, childSectionNumber);
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  /**
 | 
			
		||||
   * Convert mindmap tree structure to flat array of nodes
 | 
			
		||||
   * @param node - The mindmap node to process
 | 
			
		||||
   * @param processedNodes - Array to collect processed nodes
 | 
			
		||||
   */
 | 
			
		||||
  public flattenNodes(node: MindmapNode, processedNodes: MindmapLayoutNode[]): void {
 | 
			
		||||
    // Build CSS classes for the node
 | 
			
		||||
    const cssClasses = ['mindmap-node'];
 | 
			
		||||
 | 
			
		||||
    // Add section-specific classes
 | 
			
		||||
    if (node.level === 0) {
 | 
			
		||||
      // Root node gets special classes
 | 
			
		||||
      cssClasses.push('section-root', 'section--1');
 | 
			
		||||
    } else if (node.section !== undefined) {
 | 
			
		||||
      // Child nodes get section class based on their section number
 | 
			
		||||
      cssClasses.push(`section-${node.section}`);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    // Add any custom classes from the node
 | 
			
		||||
    if (node.class) {
 | 
			
		||||
      cssClasses.push(node.class);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    const classes = cssClasses.join(' ');
 | 
			
		||||
 | 
			
		||||
    // Map mindmap node type to valid shape name
 | 
			
		||||
    const getShapeFromType = (type: number) => {
 | 
			
		||||
      switch (type) {
 | 
			
		||||
        case nodeType.CIRCLE:
 | 
			
		||||
          return 'circle';
 | 
			
		||||
        case nodeType.RECT:
 | 
			
		||||
          return 'rect';
 | 
			
		||||
        case nodeType.ROUNDED_RECT:
 | 
			
		||||
          return 'rounded';
 | 
			
		||||
        case nodeType.CLOUD:
 | 
			
		||||
          return 'cloud';
 | 
			
		||||
        case nodeType.BANG:
 | 
			
		||||
          return 'bang';
 | 
			
		||||
        case nodeType.HEXAGON:
 | 
			
		||||
          return 'hexagon';
 | 
			
		||||
        case nodeType.DEFAULT:
 | 
			
		||||
        case nodeType.NO_BORDER:
 | 
			
		||||
        default:
 | 
			
		||||
          return 'rect';
 | 
			
		||||
      }
 | 
			
		||||
    };
 | 
			
		||||
 | 
			
		||||
    const processedNode: MindmapLayoutNode = {
 | 
			
		||||
      id: node.id.toString(),
 | 
			
		||||
      domId: 'node_' + node.id.toString(),
 | 
			
		||||
      label: node.descr,
 | 
			
		||||
      isGroup: false,
 | 
			
		||||
      shape: getShapeFromType(node.type),
 | 
			
		||||
      width: node.width,
 | 
			
		||||
      height: node.height ?? 0,
 | 
			
		||||
      padding: node.padding,
 | 
			
		||||
      cssClasses: classes,
 | 
			
		||||
      cssStyles: [],
 | 
			
		||||
      look: 'default',
 | 
			
		||||
      icon: node.icon,
 | 
			
		||||
      x: node.x,
 | 
			
		||||
      y: node.y,
 | 
			
		||||
      // Mindmap-specific properties
 | 
			
		||||
      level: node.level,
 | 
			
		||||
      nodeId: node.nodeId,
 | 
			
		||||
      type: node.type,
 | 
			
		||||
      section: node.section,
 | 
			
		||||
    };
 | 
			
		||||
 | 
			
		||||
    processedNodes.push(processedNode);
 | 
			
		||||
 | 
			
		||||
    // Recursively process children
 | 
			
		||||
    if (node.children) {
 | 
			
		||||
      for (const child of node.children) {
 | 
			
		||||
        this.flattenNodes(child, processedNodes);
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  /**
 | 
			
		||||
   * Generate edges from parent-child relationships in mindmap tree
 | 
			
		||||
   * @param node - The mindmap node to process
 | 
			
		||||
   * @param edges - Array to collect edges
 | 
			
		||||
   */
 | 
			
		||||
  public generateEdges(node: MindmapNode, edges: MindmapLayoutEdge[]): void {
 | 
			
		||||
    if (!node.children) {
 | 
			
		||||
      return;
 | 
			
		||||
    }
 | 
			
		||||
    for (const child of node.children) {
 | 
			
		||||
      // Build CSS classes for the edge
 | 
			
		||||
      let edgeClasses = 'edge';
 | 
			
		||||
 | 
			
		||||
      // Add section-specific classes based on the child's section
 | 
			
		||||
      if (child.section !== undefined) {
 | 
			
		||||
        edgeClasses += ` section-edge-${child.section}`;
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      // Add depth class based on the parent's level + 1 (depth of the edge)
 | 
			
		||||
      const edgeDepth = node.level + 1;
 | 
			
		||||
      edgeClasses += ` edge-depth-${edgeDepth}`;
 | 
			
		||||
 | 
			
		||||
      const edge: MindmapLayoutEdge = {
 | 
			
		||||
        id: `edge_${node.id}_${child.id}`,
 | 
			
		||||
        start: node.id.toString(),
 | 
			
		||||
        end: child.id.toString(),
 | 
			
		||||
        type: 'normal',
 | 
			
		||||
        curve: 'basis',
 | 
			
		||||
        thickness: 'normal',
 | 
			
		||||
        look: 'default',
 | 
			
		||||
        classes: edgeClasses,
 | 
			
		||||
        // Store mindmap-specific data
 | 
			
		||||
        depth: node.level,
 | 
			
		||||
        section: child.section,
 | 
			
		||||
      };
 | 
			
		||||
 | 
			
		||||
      edges.push(edge);
 | 
			
		||||
 | 
			
		||||
      // Recursively process child edges
 | 
			
		||||
      this.generateEdges(child, edges);
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  /**
 | 
			
		||||
   * Get structured data for layout algorithms
 | 
			
		||||
   * Following the pattern established by ER diagrams
 | 
			
		||||
   * @returns Structured data containing nodes, edges, and config
 | 
			
		||||
   */
 | 
			
		||||
  public getData(): LayoutData {
 | 
			
		||||
    const mindmapRoot = this.getMindmap();
 | 
			
		||||
    const config = getConfig();
 | 
			
		||||
 | 
			
		||||
    if (!mindmapRoot) {
 | 
			
		||||
      return {
 | 
			
		||||
        nodes: [],
 | 
			
		||||
        edges: [],
 | 
			
		||||
        config,
 | 
			
		||||
      };
 | 
			
		||||
    }
 | 
			
		||||
    log.debug('getData: mindmapRoot', mindmapRoot, config);
 | 
			
		||||
 | 
			
		||||
    // Assign section numbers to all nodes based on their position relative to root
 | 
			
		||||
    this.assignSections(mindmapRoot);
 | 
			
		||||
 | 
			
		||||
    // Convert tree structure to flat arrays
 | 
			
		||||
    const processedNodes: MindmapLayoutNode[] = [];
 | 
			
		||||
    const processedEdges: MindmapLayoutEdge[] = [];
 | 
			
		||||
 | 
			
		||||
    this.flattenNodes(mindmapRoot, processedNodes);
 | 
			
		||||
    this.generateEdges(mindmapRoot, processedEdges);
 | 
			
		||||
 | 
			
		||||
    log.debug(
 | 
			
		||||
      `getData: processed ${processedNodes.length} nodes and ${processedEdges.length} edges`
 | 
			
		||||
    );
 | 
			
		||||
 | 
			
		||||
    // Create shapes map for ELK compatibility
 | 
			
		||||
    const shapes = new Map<string, any>();
 | 
			
		||||
    for (const node of processedNodes) {
 | 
			
		||||
      shapes.set(node.id, {
 | 
			
		||||
        shape: node.shape,
 | 
			
		||||
        width: node.width,
 | 
			
		||||
        height: node.height,
 | 
			
		||||
        padding: node.padding,
 | 
			
		||||
      });
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    return {
 | 
			
		||||
      nodes: processedNodes,
 | 
			
		||||
      edges: processedEdges,
 | 
			
		||||
      config,
 | 
			
		||||
      // Store the root node for mindmap-specific layout algorithms
 | 
			
		||||
      rootNode: mindmapRoot,
 | 
			
		||||
      // Properties required by dagre layout algorithm
 | 
			
		||||
      markers: [], // Mindmaps don't use markers
 | 
			
		||||
      direction: 'TB', // Top-to-bottom direction for mindmaps
 | 
			
		||||
      nodeSpacing: 50, // Default spacing between nodes
 | 
			
		||||
      rankSpacing: 50, // Default spacing between ranks
 | 
			
		||||
      // Add shapes for ELK compatibility
 | 
			
		||||
      shapes: Object.fromEntries(shapes),
 | 
			
		||||
      // Additional properties that layout algorithms might expect
 | 
			
		||||
      type: 'mindmap',
 | 
			
		||||
      diagramId: 'mindmap-' + Date.now(),
 | 
			
		||||
    };
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  // Expose logger to grammar
 | 
			
		||||
  public getLogger() {
 | 
			
		||||
    return log;
 | 
			
		||||
  }
 | 
			
		||||
 
 | 
			
		||||
@@ -1,200 +1,94 @@
 | 
			
		||||
import cytoscape from 'cytoscape';
 | 
			
		||||
// @ts-expect-error No types available
 | 
			
		||||
import coseBilkent from 'cytoscape-cose-bilkent';
 | 
			
		||||
import { select } from 'd3';
 | 
			
		||||
import type { MermaidConfig } from '../../config.type.js';
 | 
			
		||||
import { getConfig } from '../../diagram-api/diagramAPI.js';
 | 
			
		||||
import type { DrawDefinition } from '../../diagram-api/types.js';
 | 
			
		||||
import { log } from '../../logger.js';
 | 
			
		||||
import type { D3Element } from '../../types.js';
 | 
			
		||||
import { selectSvgElement } from '../../rendering-util/selectSvgElement.js';
 | 
			
		||||
import { setupGraphViewbox } from '../../setupGraphViewbox.js';
 | 
			
		||||
import type { FilledMindMapNode, MindmapNode } from './mindmapTypes.js';
 | 
			
		||||
import { drawNode, positionNode } from './svgDraw.js';
 | 
			
		||||
import { getDiagramElement } from '../../rendering-util/insertElementsForSize.js';
 | 
			
		||||
import { getRegisteredLayoutAlgorithm, render } from '../../rendering-util/render.js';
 | 
			
		||||
import { setupViewPortForSVG } from '../../rendering-util/setupViewPortForSVG.js';
 | 
			
		||||
import type { LayoutData } from '../../rendering-util/types.js';
 | 
			
		||||
import type { FilledMindMapNode } from './mindmapTypes.js';
 | 
			
		||||
import { drawNode } from './svgDraw.js';
 | 
			
		||||
import defaultConfig from '../../defaultConfig.js';
 | 
			
		||||
import type { MindmapDB } from './mindmapDb.js';
 | 
			
		||||
// Inject the layout algorithm into cytoscape
 | 
			
		||||
cytoscape.use(coseBilkent);
 | 
			
		||||
 | 
			
		||||
async function drawNodes(
 | 
			
		||||
async function _drawNodes(
 | 
			
		||||
  db: MindmapDB,
 | 
			
		||||
  svg: D3Element,
 | 
			
		||||
  svg: any,
 | 
			
		||||
  mindmap: FilledMindMapNode,
 | 
			
		||||
  section: number,
 | 
			
		||||
  conf: MermaidConfig
 | 
			
		||||
  conf: any
 | 
			
		||||
) {
 | 
			
		||||
  await drawNode(db, svg, mindmap, section, conf);
 | 
			
		||||
  if (mindmap.children) {
 | 
			
		||||
    await Promise.all(
 | 
			
		||||
      mindmap.children.map((child, index) =>
 | 
			
		||||
        drawNodes(db, svg, child, section < 0 ? index : section, conf)
 | 
			
		||||
        _drawNodes(db, svg, child, section < 0 ? index : section, conf)
 | 
			
		||||
      )
 | 
			
		||||
    );
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
declare module 'cytoscape' {
 | 
			
		||||
  interface EdgeSingular {
 | 
			
		||||
    _private: {
 | 
			
		||||
      bodyBounds: unknown;
 | 
			
		||||
      rscratch: {
 | 
			
		||||
        startX: number;
 | 
			
		||||
        startY: number;
 | 
			
		||||
        midX: number;
 | 
			
		||||
        midY: number;
 | 
			
		||||
        endX: number;
 | 
			
		||||
        endY: number;
 | 
			
		||||
      };
 | 
			
		||||
    };
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function drawEdges(edgesEl: D3Element, cy: cytoscape.Core) {
 | 
			
		||||
  cy.edges().map((edge, id) => {
 | 
			
		||||
    const data = edge.data();
 | 
			
		||||
    if (edge[0]._private.bodyBounds) {
 | 
			
		||||
      const bounds = edge[0]._private.rscratch;
 | 
			
		||||
      log.trace('Edge: ', id, data);
 | 
			
		||||
      edgesEl
 | 
			
		||||
        .insert('path')
 | 
			
		||||
        .attr(
 | 
			
		||||
          'd',
 | 
			
		||||
          `M ${bounds.startX},${bounds.startY} L ${bounds.midX},${bounds.midY} L${bounds.endX},${bounds.endY} `
 | 
			
		||||
        )
 | 
			
		||||
        .attr('class', 'edge section-edge-' + data.section + ' edge-depth-' + data.depth);
 | 
			
		||||
/**
 | 
			
		||||
 * Update the layout data with actual node dimensions after drawing
 | 
			
		||||
 */
 | 
			
		||||
function _updateNodeDimensions(data4Layout: LayoutData, mindmapRoot: FilledMindMapNode) {
 | 
			
		||||
  const updateNode = (node: FilledMindMapNode) => {
 | 
			
		||||
    // Find the corresponding node in the layout data
 | 
			
		||||
    const layoutNode = data4Layout.nodes.find((n) => n.id === node.id.toString());
 | 
			
		||||
    if (layoutNode) {
 | 
			
		||||
      // Update with the actual dimensions calculated by drawNode
 | 
			
		||||
      layoutNode.width = node.width;
 | 
			
		||||
      layoutNode.height = node.height;
 | 
			
		||||
      log.debug('Updated node dimensions:', node.id, 'width:', node.width, 'height:', node.height);
 | 
			
		||||
    }
 | 
			
		||||
  });
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function addNodes(mindmap: MindmapNode, cy: cytoscape.Core, conf: MermaidConfig, level: number) {
 | 
			
		||||
  cy.add({
 | 
			
		||||
    group: 'nodes',
 | 
			
		||||
    data: {
 | 
			
		||||
      id: mindmap.id.toString(),
 | 
			
		||||
      labelText: mindmap.descr,
 | 
			
		||||
      height: mindmap.height,
 | 
			
		||||
      width: mindmap.width,
 | 
			
		||||
      level: level,
 | 
			
		||||
      nodeId: mindmap.id,
 | 
			
		||||
      padding: mindmap.padding,
 | 
			
		||||
      type: mindmap.type,
 | 
			
		||||
    },
 | 
			
		||||
    position: {
 | 
			
		||||
      x: mindmap.x!,
 | 
			
		||||
      y: mindmap.y!,
 | 
			
		||||
    },
 | 
			
		||||
  });
 | 
			
		||||
  if (mindmap.children) {
 | 
			
		||||
    mindmap.children.forEach((child) => {
 | 
			
		||||
      addNodes(child, cy, conf, level + 1);
 | 
			
		||||
      cy.add({
 | 
			
		||||
        group: 'edges',
 | 
			
		||||
        data: {
 | 
			
		||||
          id: `${mindmap.id}_${child.id}`,
 | 
			
		||||
          source: mindmap.id,
 | 
			
		||||
          target: child.id,
 | 
			
		||||
          depth: level,
 | 
			
		||||
          section: child.section,
 | 
			
		||||
        },
 | 
			
		||||
      });
 | 
			
		||||
    });
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
    // Recursively update children
 | 
			
		||||
    if (node.children) {
 | 
			
		||||
      node.children.forEach(updateNode);
 | 
			
		||||
    }
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
function layoutMindmap(node: MindmapNode, conf: MermaidConfig): Promise<cytoscape.Core> {
 | 
			
		||||
  return new Promise((resolve) => {
 | 
			
		||||
    // Add temporary render element
 | 
			
		||||
    const renderEl = select('body').append('div').attr('id', 'cy').attr('style', 'display:none');
 | 
			
		||||
    const cy = cytoscape({
 | 
			
		||||
      container: document.getElementById('cy'), // container to render in
 | 
			
		||||
      style: [
 | 
			
		||||
        {
 | 
			
		||||
          selector: 'edge',
 | 
			
		||||
          style: {
 | 
			
		||||
            'curve-style': 'bezier',
 | 
			
		||||
          },
 | 
			
		||||
        },
 | 
			
		||||
      ],
 | 
			
		||||
    });
 | 
			
		||||
    // Remove element after layout
 | 
			
		||||
    renderEl.remove();
 | 
			
		||||
    addNodes(node, cy, conf, 0);
 | 
			
		||||
 | 
			
		||||
    // Make cytoscape care about the dimensions of the nodes
 | 
			
		||||
    cy.nodes().forEach(function (n) {
 | 
			
		||||
      n.layoutDimensions = () => {
 | 
			
		||||
        const data = n.data();
 | 
			
		||||
        return { w: data.width, h: data.height };
 | 
			
		||||
      };
 | 
			
		||||
    });
 | 
			
		||||
 | 
			
		||||
    cy.layout({
 | 
			
		||||
      name: 'cose-bilkent',
 | 
			
		||||
      // @ts-ignore Types for cose-bilkent are not correct?
 | 
			
		||||
      quality: 'proof',
 | 
			
		||||
      styleEnabled: false,
 | 
			
		||||
      animate: false,
 | 
			
		||||
    }).run();
 | 
			
		||||
    cy.ready((e) => {
 | 
			
		||||
      log.info('Ready', e);
 | 
			
		||||
      resolve(cy);
 | 
			
		||||
    });
 | 
			
		||||
  });
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function positionNodes(db: MindmapDB, cy: cytoscape.Core) {
 | 
			
		||||
  cy.nodes().map((node, id) => {
 | 
			
		||||
    const data = node.data();
 | 
			
		||||
    data.x = node.position().x;
 | 
			
		||||
    data.y = node.position().y;
 | 
			
		||||
    positionNode(db, data);
 | 
			
		||||
    const el = db.getElementById(data.nodeId);
 | 
			
		||||
    log.info('id:', id, 'Position: (', node.position().x, ', ', node.position().y, ')', data);
 | 
			
		||||
    el.attr(
 | 
			
		||||
      'transform',
 | 
			
		||||
      `translate(${node.position().x - data.width / 2}, ${node.position().y - data.height / 2})`
 | 
			
		||||
    );
 | 
			
		||||
    el.attr('attr', `apa-${id})`);
 | 
			
		||||
  });
 | 
			
		||||
  updateNode(mindmapRoot);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export const draw: DrawDefinition = async (text, id, _version, diagObj) => {
 | 
			
		||||
  log.debug('Rendering mindmap diagram\n' + text);
 | 
			
		||||
  const { securityLevel, mindmap: conf, layout } = getConfig();
 | 
			
		||||
 | 
			
		||||
  // Draw the nodes first to get their dimensions, then update the layout data
 | 
			
		||||
  const db = diagObj.db as MindmapDB;
 | 
			
		||||
 | 
			
		||||
  // The getData method provided in all supported diagrams is used to extract the data from the parsed structure
 | 
			
		||||
  // into the Layout data format
 | 
			
		||||
  const data4Layout = db.getData();
 | 
			
		||||
 | 
			
		||||
  // Create the root SVG - the element is the div containing the SVG element
 | 
			
		||||
  const svg = getDiagramElement(id, securityLevel);
 | 
			
		||||
 | 
			
		||||
  data4Layout.type = diagObj.type;
 | 
			
		||||
  data4Layout.layoutAlgorithm = getRegisteredLayoutAlgorithm(layout, {
 | 
			
		||||
    fallback: 'cose-bilkent',
 | 
			
		||||
  });
 | 
			
		||||
  // For mindmap diagrams, prioritize mindmap-specific layout algorithm configuration
 | 
			
		||||
 | 
			
		||||
  data4Layout.diagramId = id;
 | 
			
		||||
 | 
			
		||||
  // Ensure required properties are set for compatibility with different layout algorithms
 | 
			
		||||
  data4Layout.markers = ['point'];
 | 
			
		||||
  data4Layout.direction = 'TB';
 | 
			
		||||
 | 
			
		||||
  const mm = db.getMindmap();
 | 
			
		||||
  if (!mm) {
 | 
			
		||||
    return;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  const conf = getConfig();
 | 
			
		||||
  conf.htmlLabels = false;
 | 
			
		||||
 | 
			
		||||
  const svg = selectSvgElement(id);
 | 
			
		||||
 | 
			
		||||
  // Draw the graph and start with drawing the nodes without proper position
 | 
			
		||||
  // this gives us the size of the nodes and we can set the positions later
 | 
			
		||||
 | 
			
		||||
  const edgesElem = svg.append('g');
 | 
			
		||||
  edgesElem.attr('class', 'mindmap-edges');
 | 
			
		||||
  const nodesElem = svg.append('g');
 | 
			
		||||
  nodesElem.attr('class', 'mindmap-nodes');
 | 
			
		||||
  await drawNodes(db, nodesElem, mm as FilledMindMapNode, -1, conf);
 | 
			
		||||
 | 
			
		||||
  // Next step is to layout the mindmap, giving each node a position
 | 
			
		||||
 | 
			
		||||
  const cy = await layoutMindmap(mm, conf);
 | 
			
		||||
 | 
			
		||||
  // After this we can draw, first the edges and the then nodes with the correct position
 | 
			
		||||
  drawEdges(edgesElem, cy);
 | 
			
		||||
  positionNodes(db, cy);
 | 
			
		||||
 | 
			
		||||
  // Use the unified rendering system
 | 
			
		||||
  await render(data4Layout, svg);
 | 
			
		||||
  // Setup the view box and size of the svg element
 | 
			
		||||
  setupGraphViewbox(
 | 
			
		||||
    undefined,
 | 
			
		||||
  setupViewPortForSVG(
 | 
			
		||||
    svg,
 | 
			
		||||
    conf.mindmap?.padding ?? defaultConfig.mindmap.padding,
 | 
			
		||||
    conf.mindmap?.useMaxWidth ?? defaultConfig.mindmap.useMaxWidth
 | 
			
		||||
    conf?.padding ?? defaultConfig.mindmap.padding,
 | 
			
		||||
    'mindmapDiagram',
 | 
			
		||||
    conf?.useMaxWidth ?? defaultConfig.mindmap.useMaxWidth
 | 
			
		||||
  );
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -64,6 +64,9 @@ const getStyles: DiagramStylesProvider = (options) =>
 | 
			
		||||
  .section-root text {
 | 
			
		||||
    fill: ${options.gitBranchLabel0};
 | 
			
		||||
  }
 | 
			
		||||
  .section-root span {
 | 
			
		||||
    color: ${options.gitBranchLabel0};
 | 
			
		||||
  }
 | 
			
		||||
  .icon-container {
 | 
			
		||||
    height:100%;
 | 
			
		||||
    display: flex;
 | 
			
		||||
 
 | 
			
		||||
@@ -1368,7 +1368,7 @@ link a: Tests @ https://tests.contoso.com/?svc=alice@contoso.com
 | 
			
		||||
  it('should handle box without description', async () => {
 | 
			
		||||
    const diagram = await Diagram.fromText(`
 | 
			
		||||
  sequenceDiagram
 | 
			
		||||
  box Aqua
 | 
			
		||||
  box aqua
 | 
			
		||||
  participant a as Alice
 | 
			
		||||
  participant b as Bob
 | 
			
		||||
  end
 | 
			
		||||
@@ -1384,7 +1384,7 @@ link a: Tests @ https://tests.contoso.com/?svc=alice@contoso.com
 | 
			
		||||
    const boxes = diagram.db.getBoxes();
 | 
			
		||||
    expect(boxes[0].name).toBeFalsy();
 | 
			
		||||
    expect(boxes[0].actorKeys).toEqual(['a', 'b']);
 | 
			
		||||
    expect(boxes[0].fill).toEqual('Aqua');
 | 
			
		||||
    expect(boxes[0].fill).toEqual('aqua');
 | 
			
		||||
  });
 | 
			
		||||
 | 
			
		||||
  it('should handle simple actor creation', async () => {
 | 
			
		||||
 
 | 
			
		||||
@@ -17,7 +17,6 @@ While directives allow you to change most of the default configuration settings,
 | 
			
		||||
Mermaid basically supports two types of configuration options to be overridden by directives.
 | 
			
		||||
 | 
			
		||||
1. _General/Top Level configurations_ : These are the configurations that are available and applied to all the diagram. **Some of the most important top-level** configurations are:
 | 
			
		||||
 | 
			
		||||
   - theme
 | 
			
		||||
   - fontFamily
 | 
			
		||||
   - logLevel
 | 
			
		||||
 
 | 
			
		||||
@@ -23,7 +23,6 @@ Try the Ultimate AI, Mermaid, and Visual Diagramming Suite by creating an accoun
 | 
			
		||||
- **Plugins** - A plugin system for extending the functionality of Mermaid.
 | 
			
		||||
 | 
			
		||||
  Official Mermaid Chart plugins:
 | 
			
		||||
 | 
			
		||||
  - [Mermaid Chart GPT](https://chatgpt.com/g/g-684cc36f30208191b21383b88650a45d-mermaid-chart-diagrams-and-charts)
 | 
			
		||||
  - [Confluence](https://marketplace.atlassian.com/apps/1234056/mermaid-chart-for-confluence?hosting=cloud&tab=overview)
 | 
			
		||||
  - [Jira](https://marketplace.atlassian.com/apps/1234810/mermaid-chart-for-jira?tab=overview&hosting=cloud)
 | 
			
		||||
 
 | 
			
		||||
@@ -33,13 +33,11 @@ The Mermaid Chart team is excited to introduce a new Visual Editor for Flowchart
 | 
			
		||||
Learn more:
 | 
			
		||||
 | 
			
		||||
- Visual Editor For Flowcharts
 | 
			
		||||
 | 
			
		||||
  - [Blog post](https://www.mermaidchart.com/blog/posts/mermaid-chart-releases-new-visual-editor-for-flowcharts)
 | 
			
		||||
 | 
			
		||||
  - [Demo video](https://www.youtube.com/watch?v=5aja0gijoO0)
 | 
			
		||||
 | 
			
		||||
- Visual Editor For Sequence diagrams
 | 
			
		||||
 | 
			
		||||
  - [Blog post](https://www.mermaidchart.com/blog/posts/mermaid-chart-unveils-visual-editor-for-sequence-diagrams)
 | 
			
		||||
 | 
			
		||||
  - [Demo video](https://youtu.be/imc2u5_N6Dc)
 | 
			
		||||
 
 | 
			
		||||
@@ -83,7 +83,6 @@ The following unfinished features are not supported in the short term.
 | 
			
		||||
- [ ] Legend
 | 
			
		||||
 | 
			
		||||
- [x] System Context
 | 
			
		||||
 | 
			
		||||
  - [x] Person(alias, label, ?descr, ?sprite, ?tags, $link)
 | 
			
		||||
  - [x] Person_Ext
 | 
			
		||||
  - [x] System(alias, label, ?descr, ?sprite, ?tags, $link)
 | 
			
		||||
@@ -97,7 +96,6 @@ The following unfinished features are not supported in the short term.
 | 
			
		||||
  - [x] System_Boundary
 | 
			
		||||
 | 
			
		||||
- [x] Container diagram
 | 
			
		||||
 | 
			
		||||
  - [x] Container(alias, label, ?techn, ?descr, ?sprite, ?tags, $link)
 | 
			
		||||
  - [x] ContainerDb
 | 
			
		||||
  - [x] ContainerQueue
 | 
			
		||||
@@ -107,7 +105,6 @@ The following unfinished features are not supported in the short term.
 | 
			
		||||
  - [x] Container_Boundary(alias, label, ?tags, $link)
 | 
			
		||||
 | 
			
		||||
- [x] Component diagram
 | 
			
		||||
 | 
			
		||||
  - [x] Component(alias, label, ?techn, ?descr, ?sprite, ?tags, $link)
 | 
			
		||||
  - [x] ComponentDb
 | 
			
		||||
  - [x] ComponentQueue
 | 
			
		||||
@@ -116,18 +113,15 @@ The following unfinished features are not supported in the short term.
 | 
			
		||||
  - [x] ComponentQueue_Ext
 | 
			
		||||
 | 
			
		||||
- [x] Dynamic diagram
 | 
			
		||||
 | 
			
		||||
  - [x] RelIndex(index, from, to, label, ?tags, $link)
 | 
			
		||||
 | 
			
		||||
- [x] Deployment diagram
 | 
			
		||||
 | 
			
		||||
  - [x] Deployment_Node(alias, label, ?type, ?descr, ?sprite, ?tags, $link)
 | 
			
		||||
  - [x] Node(alias, label, ?type, ?descr, ?sprite, ?tags, $link): short name of Deployment_Node()
 | 
			
		||||
  - [x] Node_L(alias, label, ?type, ?descr, ?sprite, ?tags, $link): left aligned Node()
 | 
			
		||||
  - [x] Node_R(alias, label, ?type, ?descr, ?sprite, ?tags, $link): right aligned Node()
 | 
			
		||||
 | 
			
		||||
- [x] Relationship Types
 | 
			
		||||
 | 
			
		||||
  - [x] Rel(from, to, label, ?techn, ?descr, ?sprite, ?tags, $link)
 | 
			
		||||
  - [x] BiRel (bidirectional relationship)
 | 
			
		||||
  - [x] Rel_U, Rel_Up
 | 
			
		||||
 
 | 
			
		||||
@@ -1,4 +1,5 @@
 | 
			
		||||
import { cleanupComments } from './diagram-api/comments.js';
 | 
			
		||||
import { detectType } from './diagram-api/detectType.js';
 | 
			
		||||
import { extractFrontMatter } from './diagram-api/frontmatter.js';
 | 
			
		||||
import type { DiagramMetadata } from './diagram-api/types.js';
 | 
			
		||||
import utils, { cleanAndMerge, removeDirectives } from './utils.js';
 | 
			
		||||
@@ -18,6 +19,7 @@ const cleanupText = (code: string) => {
 | 
			
		||||
 | 
			
		||||
const processFrontmatter = (code: string) => {
 | 
			
		||||
  const { text, metadata } = extractFrontMatter(code);
 | 
			
		||||
  const diagramType = detectType(text);
 | 
			
		||||
  const { displayMode, title, config = {} } = metadata;
 | 
			
		||||
  if (displayMode) {
 | 
			
		||||
    // Needs to be supported for legacy reasons
 | 
			
		||||
@@ -26,6 +28,9 @@ const processFrontmatter = (code: string) => {
 | 
			
		||||
    }
 | 
			
		||||
    config.gantt.displayMode = displayMode;
 | 
			
		||||
  }
 | 
			
		||||
  if (diagramType === 'mindmap' && !config.layout) {
 | 
			
		||||
    config.layout = 'tidy-tree';
 | 
			
		||||
  }
 | 
			
		||||
  return { title, config, text };
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										148
									
								
								packages/mermaid/src/rendering-util/createGraph.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										148
									
								
								packages/mermaid/src/rendering-util/createGraph.ts
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,148 @@
 | 
			
		||||
import { insertNode } from './rendering-elements/nodes.js';
 | 
			
		||||
import type { LayoutData, NonClusterNode } from './types.ts';
 | 
			
		||||
import type { Selection } from 'd3';
 | 
			
		||||
import { getConfig } from '../diagram-api/diagramAPI.js';
 | 
			
		||||
import * as graphlib from 'dagre-d3-es/src/graphlib/index.js';
 | 
			
		||||
 | 
			
		||||
// Update type:
 | 
			
		||||
type D3Selection<T extends SVGElement = SVGElement> = Selection<
 | 
			
		||||
  T,
 | 
			
		||||
  unknown,
 | 
			
		||||
  Element | null,
 | 
			
		||||
  unknown
 | 
			
		||||
>;
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Creates a  graph by merging the graph construction and DOM element insertion.
 | 
			
		||||
 *
 | 
			
		||||
 * This function creates the graph, inserts the SVG groups (clusters, edgePaths, edgeLabels, nodes)
 | 
			
		||||
 * into the provided element, and uses `insertNode` to add nodes to the diagram. Node dimensions
 | 
			
		||||
 * are computed using each node's bounding box.
 | 
			
		||||
 *
 | 
			
		||||
 * @param element - The D3 selection in which the SVG groups are inserted.
 | 
			
		||||
 * @param data4Layout - The layout data containing nodes and edges.
 | 
			
		||||
 * @returns A promise resolving to an object containing the Graphology graph and the inserted groups.
 | 
			
		||||
 */
 | 
			
		||||
export async function createGraphWithElements(
 | 
			
		||||
  element: D3Selection,
 | 
			
		||||
  data4Layout: LayoutData
 | 
			
		||||
): Promise<{
 | 
			
		||||
  graph: graphlib.Graph;
 | 
			
		||||
  groups: {
 | 
			
		||||
    clusters: D3Selection<SVGGElement>;
 | 
			
		||||
    edgePaths: D3Selection<SVGGElement>;
 | 
			
		||||
    edgeLabels: D3Selection<SVGGElement>;
 | 
			
		||||
    nodes: D3Selection<SVGGElement>;
 | 
			
		||||
    rootGroups: D3Selection<SVGGElement>;
 | 
			
		||||
  };
 | 
			
		||||
  nodeElements: Map<string, D3Selection<SVGElement | SVGGElement>>;
 | 
			
		||||
}> {
 | 
			
		||||
  // Create a directed, multi graph.
 | 
			
		||||
  const graph = new graphlib.Graph({
 | 
			
		||||
    multigraph: true,
 | 
			
		||||
    compound: true,
 | 
			
		||||
  });
 | 
			
		||||
  const edgesToProcess = [...data4Layout.edges];
 | 
			
		||||
  const config = getConfig();
 | 
			
		||||
  // Create groups for clusters, edge paths, edge labels, and nodes.
 | 
			
		||||
  const clusters = element.insert('g').attr('class', 'clusters');
 | 
			
		||||
  const edgePaths = element.insert('g').attr('class', 'edges edgePath');
 | 
			
		||||
  const edgeLabels = element.insert('g').attr('class', 'edgeLabels');
 | 
			
		||||
  const nodesGroup = element.insert('g').attr('class', 'nodes');
 | 
			
		||||
  const rootGroups = element.insert('g').attr('class', 'root');
 | 
			
		||||
 | 
			
		||||
  const nodeElements = new Map<string, D3Selection<SVGElement | SVGGElement>>();
 | 
			
		||||
 | 
			
		||||
  // Insert nodes into the DOM and add them to the graph.
 | 
			
		||||
  await Promise.all(
 | 
			
		||||
    data4Layout.nodes.map(async (node) => {
 | 
			
		||||
      if (node.isGroup) {
 | 
			
		||||
        graph.setNode(node.id, { ...node });
 | 
			
		||||
      } else {
 | 
			
		||||
        const childNodeEl = await insertNode(nodesGroup, node, { config, dir: node.dir });
 | 
			
		||||
        const boundingBox = childNodeEl.node()?.getBBox() ?? { width: 0, height: 0 };
 | 
			
		||||
        nodeElements.set(node.id, childNodeEl as D3Selection<SVGElement | SVGGElement>);
 | 
			
		||||
        node.width = boundingBox.width;
 | 
			
		||||
        node.height = boundingBox.height;
 | 
			
		||||
        graph.setNode(node.id, { ...node });
 | 
			
		||||
      }
 | 
			
		||||
    })
 | 
			
		||||
  );
 | 
			
		||||
  // Add edges to the graph.
 | 
			
		||||
  for (const edge of edgesToProcess) {
 | 
			
		||||
    if (edge.label && edge.label?.length > 0) {
 | 
			
		||||
      // Create a label node for the edge
 | 
			
		||||
      const labelNodeId = `edge-label-${edge.start}-${edge.end}-${edge.id}`;
 | 
			
		||||
      const labelNode = {
 | 
			
		||||
        id: labelNodeId,
 | 
			
		||||
        label: edge.label,
 | 
			
		||||
        edgeStart: edge.start,
 | 
			
		||||
        edgeEnd: edge.end,
 | 
			
		||||
        shape: 'labelRect',
 | 
			
		||||
        width: 0, // Will be updated after insertion
 | 
			
		||||
        height: 0, // Will be updated after insertion
 | 
			
		||||
        isEdgeLabel: true,
 | 
			
		||||
        isDummy: true,
 | 
			
		||||
        isGroup: false,
 | 
			
		||||
        parentId: edge.parentId,
 | 
			
		||||
        ...(edge.dir ? { dir: edge.dir } : {}),
 | 
			
		||||
      } as NonClusterNode;
 | 
			
		||||
 | 
			
		||||
      // Insert the label node into the DOM
 | 
			
		||||
      const labelNodeEl = await insertNode(nodesGroup, labelNode, { config, dir: edge.dir });
 | 
			
		||||
      const boundingBox = labelNodeEl.node()?.getBBox() ?? { width: 0, height: 0 };
 | 
			
		||||
 | 
			
		||||
      // Update node dimensions
 | 
			
		||||
      labelNode.width = boundingBox.width;
 | 
			
		||||
      labelNode.height = boundingBox.height;
 | 
			
		||||
 | 
			
		||||
      // Add to graph and tracking maps
 | 
			
		||||
      graph.setNode(labelNodeId, { ...labelNode });
 | 
			
		||||
      nodeElements.set(labelNodeId, labelNodeEl as D3Selection<SVGElement | SVGGElement>);
 | 
			
		||||
      data4Layout.nodes.push(labelNode);
 | 
			
		||||
 | 
			
		||||
      // Create two edges to replace the original one
 | 
			
		||||
      const edgeToLabel = {
 | 
			
		||||
        ...edge,
 | 
			
		||||
        id: `${edge.id}-to-label`,
 | 
			
		||||
        end: labelNodeId,
 | 
			
		||||
        label: undefined,
 | 
			
		||||
        isLabelEdge: true,
 | 
			
		||||
        arrowTypeEnd: 'none',
 | 
			
		||||
        arrowTypeStart: 'none',
 | 
			
		||||
      };
 | 
			
		||||
      const edgeFromLabel = {
 | 
			
		||||
        ...edge,
 | 
			
		||||
        id: `${edge.id}-from-label`,
 | 
			
		||||
        start: labelNodeId,
 | 
			
		||||
        end: edge.end,
 | 
			
		||||
        label: undefined,
 | 
			
		||||
        isLabelEdge: true,
 | 
			
		||||
        arrowTypeStart: 'none',
 | 
			
		||||
        arrowTypeEnd: 'arrow_point',
 | 
			
		||||
      };
 | 
			
		||||
      graph.setEdge(edgeToLabel.id, edgeToLabel.start, edgeToLabel.end, { ...edgeToLabel });
 | 
			
		||||
      graph.setEdge(edgeFromLabel.id, edgeFromLabel.start, edgeFromLabel.end, { ...edgeFromLabel });
 | 
			
		||||
      data4Layout.edges.push(edgeToLabel, edgeFromLabel);
 | 
			
		||||
      const edgeIdToRemove = edge.id;
 | 
			
		||||
      data4Layout.edges = data4Layout.edges.filter((edge) => edge.id !== edgeIdToRemove);
 | 
			
		||||
      const indexInOriginal = data4Layout.edges.findIndex((e) => e.id === edge.id);
 | 
			
		||||
      if (indexInOriginal !== -1) {
 | 
			
		||||
        data4Layout.edges.splice(indexInOriginal, 1);
 | 
			
		||||
      }
 | 
			
		||||
    } else {
 | 
			
		||||
      // Regular edge without label
 | 
			
		||||
      graph.setEdge(edge.id, edge.start, edge.end, { ...edge });
 | 
			
		||||
      const edgeExists = data4Layout.edges.some((existingEdge) => existingEdge.id === edge.id);
 | 
			
		||||
      if (!edgeExists) {
 | 
			
		||||
        data4Layout.edges.push(edge);
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  return {
 | 
			
		||||
    graph,
 | 
			
		||||
    groups: { clusters, edgePaths, edgeLabels, nodes: nodesGroup, rootGroups },
 | 
			
		||||
    nodeElements,
 | 
			
		||||
  };
 | 
			
		||||
}
 | 
			
		||||
@@ -0,0 +1,269 @@
 | 
			
		||||
import { describe, it, expect, beforeEach, vi } from 'vitest';
 | 
			
		||||
import {
 | 
			
		||||
  addNodes,
 | 
			
		||||
  addEdges,
 | 
			
		||||
  extractPositionedNodes,
 | 
			
		||||
  extractPositionedEdges,
 | 
			
		||||
} from './cytoscape-setup.js';
 | 
			
		||||
import type { Node, Edge } from '../../types.js';
 | 
			
		||||
 | 
			
		||||
// Mock cytoscape
 | 
			
		||||
const mockCy = {
 | 
			
		||||
  add: vi.fn(),
 | 
			
		||||
  nodes: vi.fn(),
 | 
			
		||||
  edges: vi.fn(),
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
vi.mock('cytoscape', () => {
 | 
			
		||||
  const mockCytoscape = vi.fn(() => mockCy) as any;
 | 
			
		||||
  mockCytoscape.use = vi.fn();
 | 
			
		||||
  return {
 | 
			
		||||
    default: mockCytoscape,
 | 
			
		||||
  };
 | 
			
		||||
});
 | 
			
		||||
 | 
			
		||||
vi.mock('cytoscape-cose-bilkent', () => ({
 | 
			
		||||
  default: vi.fn(),
 | 
			
		||||
}));
 | 
			
		||||
 | 
			
		||||
describe('Cytoscape Setup', () => {
 | 
			
		||||
  let mockNodes: Node[];
 | 
			
		||||
  let mockEdges: Edge[];
 | 
			
		||||
 | 
			
		||||
  beforeEach(() => {
 | 
			
		||||
    vi.clearAllMocks();
 | 
			
		||||
 | 
			
		||||
    mockNodes = [
 | 
			
		||||
      {
 | 
			
		||||
        id: '1',
 | 
			
		||||
        label: 'Root',
 | 
			
		||||
        isGroup: false,
 | 
			
		||||
        shape: 'rect',
 | 
			
		||||
        width: 100,
 | 
			
		||||
        height: 50,
 | 
			
		||||
        padding: 10,
 | 
			
		||||
        x: 100,
 | 
			
		||||
        y: 100,
 | 
			
		||||
        cssClasses: '',
 | 
			
		||||
        cssStyles: [],
 | 
			
		||||
        look: 'default',
 | 
			
		||||
      },
 | 
			
		||||
      {
 | 
			
		||||
        id: '2',
 | 
			
		||||
        label: 'Child 1',
 | 
			
		||||
        isGroup: false,
 | 
			
		||||
        shape: 'rect',
 | 
			
		||||
        width: 80,
 | 
			
		||||
        height: 40,
 | 
			
		||||
        padding: 10,
 | 
			
		||||
        x: 150,
 | 
			
		||||
        y: 150,
 | 
			
		||||
        cssClasses: '',
 | 
			
		||||
        cssStyles: [],
 | 
			
		||||
        look: 'default',
 | 
			
		||||
      },
 | 
			
		||||
    ];
 | 
			
		||||
 | 
			
		||||
    mockEdges = [
 | 
			
		||||
      {
 | 
			
		||||
        id: '1_2',
 | 
			
		||||
        start: '1',
 | 
			
		||||
        end: '2',
 | 
			
		||||
        type: 'edge',
 | 
			
		||||
        classes: '',
 | 
			
		||||
        style: [],
 | 
			
		||||
        animate: false,
 | 
			
		||||
        arrowTypeEnd: 'arrow_point',
 | 
			
		||||
        arrowTypeStart: 'none',
 | 
			
		||||
      },
 | 
			
		||||
    ];
 | 
			
		||||
  });
 | 
			
		||||
 | 
			
		||||
  describe('addNodes', () => {
 | 
			
		||||
    it('should add nodes to cytoscape', () => {
 | 
			
		||||
      addNodes([mockNodes[0]], mockCy as unknown as any);
 | 
			
		||||
 | 
			
		||||
      expect(mockCy.add).toHaveBeenCalledWith({
 | 
			
		||||
        group: 'nodes',
 | 
			
		||||
        data: {
 | 
			
		||||
          id: '1',
 | 
			
		||||
          labelText: 'Root',
 | 
			
		||||
          height: 50,
 | 
			
		||||
          width: 100,
 | 
			
		||||
          padding: 10,
 | 
			
		||||
          isGroup: false,
 | 
			
		||||
          shape: 'rect',
 | 
			
		||||
          cssClasses: '',
 | 
			
		||||
          cssStyles: [],
 | 
			
		||||
          look: 'default',
 | 
			
		||||
        },
 | 
			
		||||
        position: {
 | 
			
		||||
          x: 100,
 | 
			
		||||
          y: 100,
 | 
			
		||||
        },
 | 
			
		||||
      });
 | 
			
		||||
    });
 | 
			
		||||
 | 
			
		||||
    it('should add multiple nodes to cytoscape', () => {
 | 
			
		||||
      addNodes(mockNodes, mockCy as unknown as any);
 | 
			
		||||
 | 
			
		||||
      expect(mockCy.add).toHaveBeenCalledTimes(2);
 | 
			
		||||
 | 
			
		||||
      expect(mockCy.add).toHaveBeenCalledWith({
 | 
			
		||||
        group: 'nodes',
 | 
			
		||||
        data: {
 | 
			
		||||
          id: '1',
 | 
			
		||||
          labelText: 'Root',
 | 
			
		||||
          height: 50,
 | 
			
		||||
          width: 100,
 | 
			
		||||
          padding: 10,
 | 
			
		||||
          isGroup: false,
 | 
			
		||||
          shape: 'rect',
 | 
			
		||||
          cssClasses: '',
 | 
			
		||||
          cssStyles: [],
 | 
			
		||||
          look: 'default',
 | 
			
		||||
        },
 | 
			
		||||
        position: {
 | 
			
		||||
          x: 100,
 | 
			
		||||
          y: 100,
 | 
			
		||||
        },
 | 
			
		||||
      });
 | 
			
		||||
 | 
			
		||||
      expect(mockCy.add).toHaveBeenCalledWith({
 | 
			
		||||
        group: 'nodes',
 | 
			
		||||
        data: {
 | 
			
		||||
          id: '2',
 | 
			
		||||
          labelText: 'Child 1',
 | 
			
		||||
          height: 40,
 | 
			
		||||
          width: 80,
 | 
			
		||||
          padding: 10,
 | 
			
		||||
          isGroup: false,
 | 
			
		||||
          shape: 'rect',
 | 
			
		||||
          cssClasses: '',
 | 
			
		||||
          cssStyles: [],
 | 
			
		||||
          look: 'default',
 | 
			
		||||
        },
 | 
			
		||||
        position: {
 | 
			
		||||
          x: 150,
 | 
			
		||||
          y: 150,
 | 
			
		||||
        },
 | 
			
		||||
      });
 | 
			
		||||
    });
 | 
			
		||||
  });
 | 
			
		||||
 | 
			
		||||
  describe('addEdges', () => {
 | 
			
		||||
    it('should add edges to cytoscape', () => {
 | 
			
		||||
      addEdges(mockEdges, mockCy as unknown as any);
 | 
			
		||||
 | 
			
		||||
      expect(mockCy.add).toHaveBeenCalledWith({
 | 
			
		||||
        group: 'edges',
 | 
			
		||||
        data: {
 | 
			
		||||
          id: '1_2',
 | 
			
		||||
          source: '1',
 | 
			
		||||
          target: '2',
 | 
			
		||||
          type: 'edge',
 | 
			
		||||
          classes: '',
 | 
			
		||||
          style: [],
 | 
			
		||||
          animate: false,
 | 
			
		||||
          arrowTypeEnd: 'arrow_point',
 | 
			
		||||
          arrowTypeStart: 'none',
 | 
			
		||||
        },
 | 
			
		||||
      });
 | 
			
		||||
    });
 | 
			
		||||
  });
 | 
			
		||||
 | 
			
		||||
  describe('extractPositionedNodes', () => {
 | 
			
		||||
    it('should extract positioned nodes from cytoscape', () => {
 | 
			
		||||
      const mockCytoscapeNodes = [
 | 
			
		||||
        {
 | 
			
		||||
          data: () => ({
 | 
			
		||||
            id: '1',
 | 
			
		||||
            labelText: 'Root',
 | 
			
		||||
            width: 100,
 | 
			
		||||
            height: 50,
 | 
			
		||||
            padding: 10,
 | 
			
		||||
            isGroup: false,
 | 
			
		||||
            shape: 'rect',
 | 
			
		||||
          }),
 | 
			
		||||
          position: () => ({ x: 100, y: 100 }),
 | 
			
		||||
        },
 | 
			
		||||
        {
 | 
			
		||||
          data: () => ({
 | 
			
		||||
            id: '2',
 | 
			
		||||
            labelText: 'Child 1',
 | 
			
		||||
            width: 80,
 | 
			
		||||
            height: 40,
 | 
			
		||||
            padding: 10,
 | 
			
		||||
            isGroup: false,
 | 
			
		||||
            shape: 'rect',
 | 
			
		||||
          }),
 | 
			
		||||
          position: () => ({ x: 150, y: 150 }),
 | 
			
		||||
        },
 | 
			
		||||
      ];
 | 
			
		||||
 | 
			
		||||
      mockCy.nodes.mockReturnValue({
 | 
			
		||||
        map: (fn: unknown) => mockCytoscapeNodes.map(fn as any),
 | 
			
		||||
      });
 | 
			
		||||
 | 
			
		||||
      const result = extractPositionedNodes(mockCy as unknown as any);
 | 
			
		||||
 | 
			
		||||
      expect(result).toHaveLength(2);
 | 
			
		||||
      expect(result[0]).toEqual({
 | 
			
		||||
        id: '1',
 | 
			
		||||
        x: 100,
 | 
			
		||||
        y: 100,
 | 
			
		||||
        labelText: 'Root',
 | 
			
		||||
        width: 100,
 | 
			
		||||
        height: 50,
 | 
			
		||||
        padding: 10,
 | 
			
		||||
        isGroup: false,
 | 
			
		||||
        shape: 'rect',
 | 
			
		||||
      });
 | 
			
		||||
    });
 | 
			
		||||
  });
 | 
			
		||||
 | 
			
		||||
  describe('extractPositionedEdges', () => {
 | 
			
		||||
    it('should extract positioned edges from cytoscape', () => {
 | 
			
		||||
      const mockCytoscapeEdges = [
 | 
			
		||||
        {
 | 
			
		||||
          data: () => ({
 | 
			
		||||
            id: '1_2',
 | 
			
		||||
            source: '1',
 | 
			
		||||
            target: '2',
 | 
			
		||||
            type: 'edge',
 | 
			
		||||
          }),
 | 
			
		||||
          _private: {
 | 
			
		||||
            rscratch: {
 | 
			
		||||
              startX: 100,
 | 
			
		||||
              startY: 100,
 | 
			
		||||
              midX: 125,
 | 
			
		||||
              midY: 125,
 | 
			
		||||
              endX: 150,
 | 
			
		||||
              endY: 150,
 | 
			
		||||
            },
 | 
			
		||||
          },
 | 
			
		||||
        },
 | 
			
		||||
      ];
 | 
			
		||||
 | 
			
		||||
      mockCy.edges.mockReturnValue({
 | 
			
		||||
        map: (fn: unknown) => mockCytoscapeEdges.map(fn as any),
 | 
			
		||||
      });
 | 
			
		||||
 | 
			
		||||
      const result = extractPositionedEdges(mockCy as unknown as any);
 | 
			
		||||
 | 
			
		||||
      expect(result).toHaveLength(1);
 | 
			
		||||
      expect(result[0]).toEqual({
 | 
			
		||||
        id: '1_2',
 | 
			
		||||
        source: '1',
 | 
			
		||||
        target: '2',
 | 
			
		||||
        type: 'edge',
 | 
			
		||||
        startX: 100,
 | 
			
		||||
        startY: 100,
 | 
			
		||||
        midX: 125,
 | 
			
		||||
        midY: 125,
 | 
			
		||||
        endX: 150,
 | 
			
		||||
        endY: 150,
 | 
			
		||||
      });
 | 
			
		||||
    });
 | 
			
		||||
  });
 | 
			
		||||
});
 | 
			
		||||
@@ -0,0 +1,207 @@
 | 
			
		||||
import cytoscape from 'cytoscape';
 | 
			
		||||
import coseBilkent from 'cytoscape-cose-bilkent';
 | 
			
		||||
import { select } from 'd3';
 | 
			
		||||
import { log } from '../../../logger.js';
 | 
			
		||||
import type { LayoutData, Node, Edge } from '../../types.js';
 | 
			
		||||
import type { CytoscapeLayoutConfig, PositionedNode, PositionedEdge } from './types.js';
 | 
			
		||||
 | 
			
		||||
// Inject the layout algorithm into cytoscape
 | 
			
		||||
cytoscape.use(coseBilkent);
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Declare module augmentation for cytoscape edge types
 | 
			
		||||
 */
 | 
			
		||||
declare module 'cytoscape' {
 | 
			
		||||
  interface EdgeSingular {
 | 
			
		||||
    _private: {
 | 
			
		||||
      bodyBounds: unknown;
 | 
			
		||||
      rscratch: {
 | 
			
		||||
        startX: number;
 | 
			
		||||
        startY: number;
 | 
			
		||||
        midX: number;
 | 
			
		||||
        midY: number;
 | 
			
		||||
        endX: number;
 | 
			
		||||
        endY: number;
 | 
			
		||||
      };
 | 
			
		||||
    };
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Add nodes to cytoscape instance from provided node array
 | 
			
		||||
 * This function processes only the nodes provided in the data structure
 | 
			
		||||
 * @param nodes - Array of nodes to add
 | 
			
		||||
 * @param cy - The cytoscape instance
 | 
			
		||||
 */
 | 
			
		||||
export function addNodes(nodes: Node[], cy: cytoscape.Core): void {
 | 
			
		||||
  nodes.forEach((node) => {
 | 
			
		||||
    const nodeData: Record<string, unknown> = {
 | 
			
		||||
      id: node.id,
 | 
			
		||||
      labelText: node.label,
 | 
			
		||||
      height: node.height,
 | 
			
		||||
      width: node.width,
 | 
			
		||||
      padding: node.padding ?? 0,
 | 
			
		||||
    };
 | 
			
		||||
 | 
			
		||||
    // Add any additional properties from the node
 | 
			
		||||
    Object.keys(node).forEach((key) => {
 | 
			
		||||
      if (!['id', 'label', 'height', 'width', 'padding', 'x', 'y'].includes(key)) {
 | 
			
		||||
        nodeData[key] = (node as unknown as Record<string, unknown>)[key];
 | 
			
		||||
      }
 | 
			
		||||
    });
 | 
			
		||||
 | 
			
		||||
    cy.add({
 | 
			
		||||
      group: 'nodes',
 | 
			
		||||
      data: nodeData,
 | 
			
		||||
      position: {
 | 
			
		||||
        x: node.x ?? 0,
 | 
			
		||||
        y: node.y ?? 0,
 | 
			
		||||
      },
 | 
			
		||||
    });
 | 
			
		||||
  });
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Add edges to cytoscape instance from provided edge array
 | 
			
		||||
 * This function processes only the edges provided in the data structure
 | 
			
		||||
 * @param edges - Array of edges to add
 | 
			
		||||
 * @param cy - The cytoscape instance
 | 
			
		||||
 */
 | 
			
		||||
export function addEdges(edges: Edge[], cy: cytoscape.Core): void {
 | 
			
		||||
  edges.forEach((edge) => {
 | 
			
		||||
    const edgeData: Record<string, unknown> = {
 | 
			
		||||
      id: edge.id,
 | 
			
		||||
      source: edge.start,
 | 
			
		||||
      target: edge.end,
 | 
			
		||||
    };
 | 
			
		||||
 | 
			
		||||
    // Add any additional properties from the edge
 | 
			
		||||
    Object.keys(edge).forEach((key) => {
 | 
			
		||||
      if (!['id', 'start', 'end'].includes(key)) {
 | 
			
		||||
        edgeData[key] = (edge as unknown as Record<string, unknown>)[key];
 | 
			
		||||
      }
 | 
			
		||||
    });
 | 
			
		||||
 | 
			
		||||
    cy.add({
 | 
			
		||||
      group: 'edges',
 | 
			
		||||
      data: edgeData,
 | 
			
		||||
    });
 | 
			
		||||
  });
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Create and configure cytoscape instance
 | 
			
		||||
 * @param data - Layout data containing nodes and edges
 | 
			
		||||
 * @returns Promise resolving to configured cytoscape instance
 | 
			
		||||
 */
 | 
			
		||||
export function createCytoscapeInstance(data: LayoutData): Promise<cytoscape.Core> {
 | 
			
		||||
  return new Promise((resolve) => {
 | 
			
		||||
    // Add temporary render element
 | 
			
		||||
    const renderEl = select('body').append('div').attr('id', 'cy').attr('style', 'display:none');
 | 
			
		||||
 | 
			
		||||
    const cy = cytoscape({
 | 
			
		||||
      container: document.getElementById('cy'), // container to render in
 | 
			
		||||
      style: [
 | 
			
		||||
        {
 | 
			
		||||
          selector: 'edge',
 | 
			
		||||
          style: {
 | 
			
		||||
            'curve-style': 'bezier',
 | 
			
		||||
          },
 | 
			
		||||
        },
 | 
			
		||||
      ],
 | 
			
		||||
    });
 | 
			
		||||
 | 
			
		||||
    // Remove element after layout
 | 
			
		||||
    renderEl.remove();
 | 
			
		||||
 | 
			
		||||
    // Add all nodes and edges to cytoscape using the generic functions
 | 
			
		||||
    addNodes(data.nodes, cy);
 | 
			
		||||
    addEdges(data.edges, cy);
 | 
			
		||||
 | 
			
		||||
    // Make cytoscape care about the dimensions of the nodes
 | 
			
		||||
    cy.nodes().forEach(function (n) {
 | 
			
		||||
      n.layoutDimensions = () => {
 | 
			
		||||
        const nodeData = n.data();
 | 
			
		||||
        return { w: nodeData.width, h: nodeData.height };
 | 
			
		||||
      };
 | 
			
		||||
    });
 | 
			
		||||
 | 
			
		||||
    // Configure and run the cose-bilkent layout
 | 
			
		||||
    const layoutConfig: CytoscapeLayoutConfig = {
 | 
			
		||||
      name: 'cose-bilkent',
 | 
			
		||||
      // @ts-ignore Types for cose-bilkent are not correct?
 | 
			
		||||
      quality: 'proof',
 | 
			
		||||
      styleEnabled: false,
 | 
			
		||||
      animate: false,
 | 
			
		||||
    };
 | 
			
		||||
 | 
			
		||||
    cy.layout(layoutConfig).run();
 | 
			
		||||
 | 
			
		||||
    cy.ready((e) => {
 | 
			
		||||
      log.info('Cytoscape ready', e);
 | 
			
		||||
      resolve(cy);
 | 
			
		||||
    });
 | 
			
		||||
  });
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Extract positioned nodes from cytoscape instance
 | 
			
		||||
 * @param cy - The cytoscape instance after layout
 | 
			
		||||
 * @returns Array of positioned nodes
 | 
			
		||||
 */
 | 
			
		||||
export function extractPositionedNodes(cy: cytoscape.Core): PositionedNode[] {
 | 
			
		||||
  return cy.nodes().map((node) => {
 | 
			
		||||
    const data = node.data();
 | 
			
		||||
    const position = node.position();
 | 
			
		||||
 | 
			
		||||
    // Create a positioned node with all original data plus position
 | 
			
		||||
    const positionedNode: PositionedNode = {
 | 
			
		||||
      id: data.id,
 | 
			
		||||
      x: position.x,
 | 
			
		||||
      y: position.y,
 | 
			
		||||
    };
 | 
			
		||||
 | 
			
		||||
    // Add all other properties from the original data
 | 
			
		||||
    Object.keys(data).forEach((key) => {
 | 
			
		||||
      if (key !== 'id') {
 | 
			
		||||
        positionedNode[key] = data[key];
 | 
			
		||||
      }
 | 
			
		||||
    });
 | 
			
		||||
 | 
			
		||||
    return positionedNode;
 | 
			
		||||
  });
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Extract positioned edges from cytoscape instance
 | 
			
		||||
 * @param cy - The cytoscape instance after layout
 | 
			
		||||
 * @returns Array of positioned edges
 | 
			
		||||
 */
 | 
			
		||||
export function extractPositionedEdges(cy: cytoscape.Core): PositionedEdge[] {
 | 
			
		||||
  return cy.edges().map((edge) => {
 | 
			
		||||
    const data = edge.data();
 | 
			
		||||
    const rscratch = edge._private.rscratch;
 | 
			
		||||
 | 
			
		||||
    // Create a positioned edge with all original data plus position
 | 
			
		||||
    const positionedEdge: PositionedEdge = {
 | 
			
		||||
      id: data.id,
 | 
			
		||||
      source: data.source,
 | 
			
		||||
      target: data.target,
 | 
			
		||||
      startX: rscratch.startX,
 | 
			
		||||
      startY: rscratch.startY,
 | 
			
		||||
      midX: rscratch.midX,
 | 
			
		||||
      midY: rscratch.midY,
 | 
			
		||||
      endX: rscratch.endX,
 | 
			
		||||
      endY: rscratch.endY,
 | 
			
		||||
    };
 | 
			
		||||
 | 
			
		||||
    // Add all other properties from the original data
 | 
			
		||||
    Object.keys(data).forEach((key) => {
 | 
			
		||||
      if (!['id', 'source', 'target'].includes(key)) {
 | 
			
		||||
        positionedEdge[key] = data[key];
 | 
			
		||||
      }
 | 
			
		||||
    });
 | 
			
		||||
 | 
			
		||||
    return positionedEdge;
 | 
			
		||||
  });
 | 
			
		||||
}
 | 
			
		||||
@@ -0,0 +1,25 @@
 | 
			
		||||
import { render as renderWithCoseBilkent } from './render.js';
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Cose-Bilkent Layout Algorithm for Generic Diagrams
 | 
			
		||||
 *
 | 
			
		||||
 * This module provides a layout algorithm implementation using Cytoscape
 | 
			
		||||
 * with the cose-bilkent algorithm for positioning nodes and edges.
 | 
			
		||||
 *
 | 
			
		||||
 * The algorithm follows the unified rendering pattern and can be used
 | 
			
		||||
 * by any diagram type that provides compatible LayoutData.
 | 
			
		||||
 */
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Render function for the cose-bilkent layout algorithm
 | 
			
		||||
 *
 | 
			
		||||
 * This function follows the unified rendering pattern used by all layout algorithms.
 | 
			
		||||
 * It takes LayoutData, inserts nodes into DOM, runs the cose-bilkent layout algorithm,
 | 
			
		||||
 * and renders the positioned elements to the SVG.
 | 
			
		||||
 *
 | 
			
		||||
 * @param layoutData - Layout data containing nodes, edges, and configuration
 | 
			
		||||
 * @param svg - SVG element to render to
 | 
			
		||||
 * @param helpers - Internal helper functions for rendering
 | 
			
		||||
 * @param options - Rendering options
 | 
			
		||||
 */
 | 
			
		||||
export const render = renderWithCoseBilkent;
 | 
			
		||||
@@ -0,0 +1,253 @@
 | 
			
		||||
import { describe, it, expect, beforeEach, vi } from 'vitest';
 | 
			
		||||
 | 
			
		||||
// Mock cytoscape and cytoscape-cose-bilkent before importing the modules
 | 
			
		||||
vi.mock('cytoscape', () => {
 | 
			
		||||
  const mockCy = {
 | 
			
		||||
    add: vi.fn(),
 | 
			
		||||
    nodes: vi.fn(() => ({
 | 
			
		||||
      forEach: vi.fn(),
 | 
			
		||||
      map: vi.fn((fn) => [
 | 
			
		||||
        fn({
 | 
			
		||||
          data: () => ({
 | 
			
		||||
            id: '1',
 | 
			
		||||
            nodeId: '1',
 | 
			
		||||
            labelText: 'Root',
 | 
			
		||||
            level: 0,
 | 
			
		||||
            type: 0,
 | 
			
		||||
            width: 100,
 | 
			
		||||
            height: 50,
 | 
			
		||||
            padding: 10,
 | 
			
		||||
          }),
 | 
			
		||||
          position: () => ({ x: 100, y: 100 }),
 | 
			
		||||
        }),
 | 
			
		||||
      ]),
 | 
			
		||||
    })),
 | 
			
		||||
    edges: vi.fn(() => ({
 | 
			
		||||
      map: vi.fn((fn) => [
 | 
			
		||||
        fn({
 | 
			
		||||
          data: () => ({
 | 
			
		||||
            id: '1_2',
 | 
			
		||||
            source: '1',
 | 
			
		||||
            target: '2',
 | 
			
		||||
            depth: 0,
 | 
			
		||||
          }),
 | 
			
		||||
          _private: {
 | 
			
		||||
            rscratch: {
 | 
			
		||||
              startX: 100,
 | 
			
		||||
              startY: 100,
 | 
			
		||||
              midX: 150,
 | 
			
		||||
              midY: 150,
 | 
			
		||||
              endX: 200,
 | 
			
		||||
              endY: 200,
 | 
			
		||||
            },
 | 
			
		||||
          },
 | 
			
		||||
        }),
 | 
			
		||||
      ]),
 | 
			
		||||
    })),
 | 
			
		||||
    layout: vi.fn(() => ({
 | 
			
		||||
      run: vi.fn(),
 | 
			
		||||
    })),
 | 
			
		||||
    ready: vi.fn((callback) => callback({})),
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  const mockCytoscape = vi.fn(() => mockCy);
 | 
			
		||||
  (mockCytoscape as any).use = vi.fn();
 | 
			
		||||
 | 
			
		||||
  return {
 | 
			
		||||
    default: mockCytoscape,
 | 
			
		||||
  };
 | 
			
		||||
});
 | 
			
		||||
 | 
			
		||||
vi.mock('cytoscape-cose-bilkent', () => ({
 | 
			
		||||
  default: vi.fn(),
 | 
			
		||||
}));
 | 
			
		||||
 | 
			
		||||
vi.mock('d3', () => ({
 | 
			
		||||
  select: vi.fn(() => ({
 | 
			
		||||
    append: vi.fn(() => ({
 | 
			
		||||
      attr: vi.fn(() => ({
 | 
			
		||||
        attr: vi.fn(() => ({
 | 
			
		||||
          remove: vi.fn(),
 | 
			
		||||
        })),
 | 
			
		||||
      })),
 | 
			
		||||
    })),
 | 
			
		||||
  })),
 | 
			
		||||
}));
 | 
			
		||||
 | 
			
		||||
// Import modules after mocks
 | 
			
		||||
import { validateLayoutData, executeCoseBilkentLayout } from './layout.js';
 | 
			
		||||
import type { LayoutResult } from './types.js';
 | 
			
		||||
import type { MindmapNode } from '../../../diagrams/mindmap/mindmapTypes.js';
 | 
			
		||||
import type { MermaidConfig } from '../../../config.type.js';
 | 
			
		||||
import type { LayoutData } from '../../types.js';
 | 
			
		||||
 | 
			
		||||
describe('Cose-Bilkent Layout Algorithm', () => {
 | 
			
		||||
  let mockConfig: MermaidConfig;
 | 
			
		||||
  let mockRootNode: MindmapNode;
 | 
			
		||||
  let mockLayoutData: LayoutData;
 | 
			
		||||
 | 
			
		||||
  beforeEach(() => {
 | 
			
		||||
    mockConfig = {
 | 
			
		||||
      mindmap: {
 | 
			
		||||
        layoutAlgorithm: 'cose-bilkent',
 | 
			
		||||
        padding: 10,
 | 
			
		||||
        maxNodeWidth: 200,
 | 
			
		||||
        useMaxWidth: true,
 | 
			
		||||
      },
 | 
			
		||||
    } as MermaidConfig;
 | 
			
		||||
 | 
			
		||||
    mockRootNode = {
 | 
			
		||||
      id: 1,
 | 
			
		||||
      nodeId: '1',
 | 
			
		||||
      level: 0,
 | 
			
		||||
      descr: 'Root',
 | 
			
		||||
      type: 0,
 | 
			
		||||
      width: 100,
 | 
			
		||||
      height: 50,
 | 
			
		||||
      padding: 10,
 | 
			
		||||
      x: 0,
 | 
			
		||||
      y: 0,
 | 
			
		||||
      children: [
 | 
			
		||||
        {
 | 
			
		||||
          id: 2,
 | 
			
		||||
          nodeId: '2',
 | 
			
		||||
          level: 1,
 | 
			
		||||
          descr: 'Child 1',
 | 
			
		||||
          type: 0,
 | 
			
		||||
          width: 80,
 | 
			
		||||
          height: 40,
 | 
			
		||||
          padding: 10,
 | 
			
		||||
          x: 0,
 | 
			
		||||
          y: 0,
 | 
			
		||||
        },
 | 
			
		||||
      ],
 | 
			
		||||
    } as MindmapNode;
 | 
			
		||||
 | 
			
		||||
    mockLayoutData = {
 | 
			
		||||
      nodes: [
 | 
			
		||||
        {
 | 
			
		||||
          id: '1',
 | 
			
		||||
          nodeId: '1',
 | 
			
		||||
          level: 0,
 | 
			
		||||
          descr: 'Root',
 | 
			
		||||
          type: 0,
 | 
			
		||||
          width: 100,
 | 
			
		||||
          height: 50,
 | 
			
		||||
          padding: 10,
 | 
			
		||||
          isGroup: false,
 | 
			
		||||
        },
 | 
			
		||||
        {
 | 
			
		||||
          id: '2',
 | 
			
		||||
          nodeId: '2',
 | 
			
		||||
          level: 1,
 | 
			
		||||
          descr: 'Child 1',
 | 
			
		||||
          type: 0,
 | 
			
		||||
          width: 80,
 | 
			
		||||
          height: 40,
 | 
			
		||||
          padding: 10,
 | 
			
		||||
          isGroup: false,
 | 
			
		||||
        },
 | 
			
		||||
      ],
 | 
			
		||||
      edges: [
 | 
			
		||||
        {
 | 
			
		||||
          id: '1_2',
 | 
			
		||||
          source: '1',
 | 
			
		||||
          target: '2',
 | 
			
		||||
          depth: 0,
 | 
			
		||||
        },
 | 
			
		||||
      ],
 | 
			
		||||
      config: mockConfig,
 | 
			
		||||
      rootNode: mockRootNode,
 | 
			
		||||
    };
 | 
			
		||||
  });
 | 
			
		||||
 | 
			
		||||
  describe('validateLayoutData', () => {
 | 
			
		||||
    it('should validate correct layout data', () => {
 | 
			
		||||
      expect(() => validateLayoutData(mockLayoutData)).not.toThrow();
 | 
			
		||||
    });
 | 
			
		||||
 | 
			
		||||
    it('should throw error for missing data', () => {
 | 
			
		||||
      expect(() => validateLayoutData(null as any)).toThrow('Layout data is required');
 | 
			
		||||
    });
 | 
			
		||||
 | 
			
		||||
    it('should throw error for missing root node', () => {
 | 
			
		||||
      const invalidData = { ...mockLayoutData, rootNode: null as any };
 | 
			
		||||
      expect(() => validateLayoutData(invalidData)).toThrow('Root node is required');
 | 
			
		||||
    });
 | 
			
		||||
 | 
			
		||||
    it('should throw error for missing config', () => {
 | 
			
		||||
      const invalidData = { ...mockLayoutData, config: null as any };
 | 
			
		||||
      expect(() => validateLayoutData(invalidData)).toThrow('Configuration is required');
 | 
			
		||||
    });
 | 
			
		||||
 | 
			
		||||
    it('should throw error for invalid nodes array', () => {
 | 
			
		||||
      const invalidData = { ...mockLayoutData, nodes: null as any };
 | 
			
		||||
      expect(() => validateLayoutData(invalidData)).toThrow('No nodes found in layout data');
 | 
			
		||||
    });
 | 
			
		||||
 | 
			
		||||
    it('should throw error for invalid edges array', () => {
 | 
			
		||||
      const invalidData = { ...mockLayoutData, edges: null as any };
 | 
			
		||||
      expect(() => validateLayoutData(invalidData)).toThrow('Edges array is required');
 | 
			
		||||
    });
 | 
			
		||||
  });
 | 
			
		||||
 | 
			
		||||
  describe('layout function', () => {
 | 
			
		||||
    it('should execute layout algorithm successfully', async () => {
 | 
			
		||||
      const result: LayoutResult = await executeCoseBilkentLayout(mockLayoutData, mockConfig);
 | 
			
		||||
 | 
			
		||||
      expect(result).toBeDefined();
 | 
			
		||||
      expect(result.nodes).toBeDefined();
 | 
			
		||||
      expect(result.edges).toBeDefined();
 | 
			
		||||
      expect(Array.isArray(result.nodes)).toBe(true);
 | 
			
		||||
      expect(Array.isArray(result.edges)).toBe(true);
 | 
			
		||||
    });
 | 
			
		||||
 | 
			
		||||
    it('should return positioned nodes with coordinates', async () => {
 | 
			
		||||
      const result: LayoutResult = await executeCoseBilkentLayout(mockLayoutData, mockConfig);
 | 
			
		||||
 | 
			
		||||
      expect(result.nodes.length).toBeGreaterThan(0);
 | 
			
		||||
      result.nodes.forEach((node) => {
 | 
			
		||||
        expect(node.x).toBeDefined();
 | 
			
		||||
        expect(node.y).toBeDefined();
 | 
			
		||||
        expect(typeof node.x).toBe('number');
 | 
			
		||||
        expect(typeof node.y).toBe('number');
 | 
			
		||||
      });
 | 
			
		||||
    });
 | 
			
		||||
 | 
			
		||||
    it('should return positioned edges with coordinates', async () => {
 | 
			
		||||
      const result: LayoutResult = await executeCoseBilkentLayout(mockLayoutData, mockConfig);
 | 
			
		||||
 | 
			
		||||
      expect(result.edges.length).toBeGreaterThan(0);
 | 
			
		||||
      result.edges.forEach((edge) => {
 | 
			
		||||
        expect(edge.startX).toBeDefined();
 | 
			
		||||
        expect(edge.startY).toBeDefined();
 | 
			
		||||
        expect(edge.midX).toBeDefined();
 | 
			
		||||
        expect(edge.midY).toBeDefined();
 | 
			
		||||
        expect(edge.endX).toBeDefined();
 | 
			
		||||
        expect(edge.endY).toBeDefined();
 | 
			
		||||
      });
 | 
			
		||||
    });
 | 
			
		||||
 | 
			
		||||
    it('should handle empty mindmap data gracefully', async () => {
 | 
			
		||||
      const emptyData: LayoutData = {
 | 
			
		||||
        nodes: [],
 | 
			
		||||
        edges: [],
 | 
			
		||||
        config: mockConfig,
 | 
			
		||||
        rootNode: mockRootNode,
 | 
			
		||||
      };
 | 
			
		||||
 | 
			
		||||
      const result: LayoutResult = await executeCoseBilkentLayout(emptyData, mockConfig);
 | 
			
		||||
      expect(result).toBeDefined();
 | 
			
		||||
      expect(result.nodes).toBeDefined();
 | 
			
		||||
      expect(result.edges).toBeDefined();
 | 
			
		||||
      expect(Array.isArray(result.nodes)).toBe(true);
 | 
			
		||||
      expect(Array.isArray(result.edges)).toBe(true);
 | 
			
		||||
    });
 | 
			
		||||
 | 
			
		||||
    it('should throw error for invalid data', async () => {
 | 
			
		||||
      const invalidData = { ...mockLayoutData, rootNode: null as any };
 | 
			
		||||
 | 
			
		||||
      await expect(executeCoseBilkentLayout(invalidData, mockConfig)).rejects.toThrow();
 | 
			
		||||
    });
 | 
			
		||||
  });
 | 
			
		||||
});
 | 
			
		||||
@@ -0,0 +1,77 @@
 | 
			
		||||
import type { MermaidConfig } from '../../../config.type.js';
 | 
			
		||||
import { log } from '../../../logger.js';
 | 
			
		||||
import type { LayoutData } from '../../types.js';
 | 
			
		||||
import type { LayoutResult } from './types.js';
 | 
			
		||||
import {
 | 
			
		||||
  createCytoscapeInstance,
 | 
			
		||||
  extractPositionedNodes,
 | 
			
		||||
  extractPositionedEdges,
 | 
			
		||||
} from './cytoscape-setup.js';
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Execute the cose-bilkent layout algorithm on generic layout data
 | 
			
		||||
 *
 | 
			
		||||
 * This function takes layout data and uses Cytoscape with the cose-bilkent
 | 
			
		||||
 * algorithm to calculate optimal node positions and edge paths.
 | 
			
		||||
 *
 | 
			
		||||
 * @param data - The layout data containing nodes, edges, and configuration
 | 
			
		||||
 * @param config - Mermaid configuration object
 | 
			
		||||
 * @returns Promise resolving to layout result with positioned nodes and edges
 | 
			
		||||
 */
 | 
			
		||||
export async function executeCoseBilkentLayout(
 | 
			
		||||
  data: LayoutData,
 | 
			
		||||
  _config: MermaidConfig
 | 
			
		||||
): Promise<LayoutResult> {
 | 
			
		||||
  log.debug('Starting cose-bilkent layout algorithm');
 | 
			
		||||
 | 
			
		||||
  try {
 | 
			
		||||
    // Validate layout data structure
 | 
			
		||||
    validateLayoutData(data);
 | 
			
		||||
 | 
			
		||||
    // Create and configure cytoscape instance
 | 
			
		||||
    const cy = await createCytoscapeInstance(data);
 | 
			
		||||
 | 
			
		||||
    // Extract positioned nodes and edges after layout
 | 
			
		||||
    const positionedNodes = extractPositionedNodes(cy);
 | 
			
		||||
    const positionedEdges = extractPositionedEdges(cy);
 | 
			
		||||
 | 
			
		||||
    log.debug(`Layout completed: ${positionedNodes.length} nodes, ${positionedEdges.length} edges`);
 | 
			
		||||
 | 
			
		||||
    return {
 | 
			
		||||
      nodes: positionedNodes,
 | 
			
		||||
      edges: positionedEdges,
 | 
			
		||||
    };
 | 
			
		||||
  } catch (error) {
 | 
			
		||||
    log.error('Error in cose-bilkent layout algorithm:', error);
 | 
			
		||||
    throw error;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Validate layout data structure
 | 
			
		||||
 * @param data - The data to validate
 | 
			
		||||
 * @returns True if data is valid, throws error otherwise
 | 
			
		||||
 */
 | 
			
		||||
export function validateLayoutData(data: LayoutData): boolean {
 | 
			
		||||
  if (!data) {
 | 
			
		||||
    throw new Error('Layout data is required');
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  if (!data.config) {
 | 
			
		||||
    throw new Error('Configuration is required in layout data');
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  if (!data.rootNode) {
 | 
			
		||||
    throw new Error('Root node is required');
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  if (!data.nodes || !Array.isArray(data.nodes)) {
 | 
			
		||||
    throw new Error('No nodes found in layout data');
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  if (!Array.isArray(data.edges)) {
 | 
			
		||||
    throw new Error('Edges array is required in layout data');
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  return true;
 | 
			
		||||
}
 | 
			
		||||
@@ -0,0 +1,197 @@
 | 
			
		||||
import type { InternalHelpers, LayoutData, RenderOptions, SVG, SVGGroup } from 'mermaid';
 | 
			
		||||
import { executeCoseBilkentLayout } from './layout.js';
 | 
			
		||||
import type { D3Selection } from '../../../types.js';
 | 
			
		||||
 | 
			
		||||
type Node = Record<string, unknown>;
 | 
			
		||||
 | 
			
		||||
interface NodeWithPosition extends Node {
 | 
			
		||||
  x?: number;
 | 
			
		||||
  y?: number;
 | 
			
		||||
  domId?: string | SVGGroup | D3Selection<SVGAElement>;
 | 
			
		||||
  width?: number;
 | 
			
		||||
  height?: number;
 | 
			
		||||
  id?: string;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Render function for cose-bilkent layout algorithm
 | 
			
		||||
 *
 | 
			
		||||
 * This follows the same pattern as ELK and dagre renderers:
 | 
			
		||||
 * 1. Insert nodes into DOM to get their actual dimensions
 | 
			
		||||
 * 2. Run the layout algorithm to calculate positions
 | 
			
		||||
 * 3. Position the nodes and edges based on layout results
 | 
			
		||||
 */
 | 
			
		||||
export const render = async (
 | 
			
		||||
  data4Layout: LayoutData,
 | 
			
		||||
  svg: SVG,
 | 
			
		||||
  {
 | 
			
		||||
    insertCluster,
 | 
			
		||||
    insertEdge,
 | 
			
		||||
    insertEdgeLabel,
 | 
			
		||||
    insertMarkers,
 | 
			
		||||
    insertNode,
 | 
			
		||||
    log,
 | 
			
		||||
    positionEdgeLabel,
 | 
			
		||||
  }: InternalHelpers,
 | 
			
		||||
  { algorithm: _algorithm }: RenderOptions
 | 
			
		||||
) => {
 | 
			
		||||
  const nodeDb: Record<string, NodeWithPosition> = {};
 | 
			
		||||
  const clusterDb: Record<string, any> = {};
 | 
			
		||||
 | 
			
		||||
  // Insert markers for edges
 | 
			
		||||
  const element = svg.select('g');
 | 
			
		||||
  insertMarkers(element, data4Layout.markers, data4Layout.type, data4Layout.diagramId);
 | 
			
		||||
 | 
			
		||||
  // Create container groups
 | 
			
		||||
  const subGraphsEl = element.insert('g').attr('class', 'subgraphs');
 | 
			
		||||
  const edgePaths = element.insert('g').attr('class', 'edgePaths');
 | 
			
		||||
  const edgeLabels = element.insert('g').attr('class', 'edgeLabels');
 | 
			
		||||
  const nodes = element.insert('g').attr('class', 'nodes');
 | 
			
		||||
 | 
			
		||||
  // Step 1: Insert nodes into DOM to get their actual dimensions
 | 
			
		||||
  log.debug('Inserting nodes into DOM for dimension calculation');
 | 
			
		||||
 | 
			
		||||
  await Promise.all(
 | 
			
		||||
    data4Layout.nodes.map(async (node) => {
 | 
			
		||||
      if (node.isGroup) {
 | 
			
		||||
        // Handle subgraphs/clusters
 | 
			
		||||
        const clusterNode: NodeWithPosition = { ...node };
 | 
			
		||||
        clusterDb[node.id] = clusterNode;
 | 
			
		||||
        nodeDb[node.id] = clusterNode;
 | 
			
		||||
 | 
			
		||||
        // Insert cluster to get dimensions
 | 
			
		||||
        await insertCluster(subGraphsEl, node);
 | 
			
		||||
      } else {
 | 
			
		||||
        // Handle regular nodes
 | 
			
		||||
        const nodeWithPosition: NodeWithPosition = { ...node };
 | 
			
		||||
        nodeDb[node.id] = nodeWithPosition;
 | 
			
		||||
 | 
			
		||||
        // Insert node to get actual dimensions
 | 
			
		||||
        const nodeEl = await insertNode(nodes, node, {
 | 
			
		||||
          config: data4Layout.config,
 | 
			
		||||
          dir: data4Layout.direction || 'TB',
 | 
			
		||||
        });
 | 
			
		||||
 | 
			
		||||
        // Get the actual bounding box after insertion
 | 
			
		||||
        const boundingBox = nodeEl.node()!.getBBox();
 | 
			
		||||
        nodeWithPosition.width = boundingBox.width;
 | 
			
		||||
        nodeWithPosition.height = boundingBox.height;
 | 
			
		||||
        nodeWithPosition.domId = nodeEl;
 | 
			
		||||
 | 
			
		||||
        log.debug(`Node ${node.id} dimensions: ${boundingBox.width}x${boundingBox.height}`);
 | 
			
		||||
      }
 | 
			
		||||
    })
 | 
			
		||||
  );
 | 
			
		||||
 | 
			
		||||
  // Step 2: Run the cose-bilkent layout algorithm
 | 
			
		||||
  log.debug('Running cose-bilkent layout algorithm');
 | 
			
		||||
 | 
			
		||||
  // Update the layout data with actual dimensions
 | 
			
		||||
  const updatedLayoutData = {
 | 
			
		||||
    ...data4Layout,
 | 
			
		||||
    nodes: data4Layout.nodes.map((node) => {
 | 
			
		||||
      const nodeWithDimensions = nodeDb[node.id];
 | 
			
		||||
      return {
 | 
			
		||||
        ...node,
 | 
			
		||||
        width: nodeWithDimensions.width,
 | 
			
		||||
        height: nodeWithDimensions.height,
 | 
			
		||||
      };
 | 
			
		||||
    }),
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  const layoutResult = await executeCoseBilkentLayout(updatedLayoutData, data4Layout.config);
 | 
			
		||||
 | 
			
		||||
  // Step 3: Position the nodes based on layout results
 | 
			
		||||
  log.debug('Positioning nodes based on layout results');
 | 
			
		||||
 | 
			
		||||
  layoutResult.nodes.forEach((positionedNode) => {
 | 
			
		||||
    const node = nodeDb[positionedNode.id];
 | 
			
		||||
    if (node?.domId) {
 | 
			
		||||
      // Position the node at the calculated coordinates
 | 
			
		||||
      // The positionedNode.x/y represents the center of the node, so use directly
 | 
			
		||||
      (node.domId as D3Selection<SVGAElement>).attr(
 | 
			
		||||
        'transform',
 | 
			
		||||
        `translate(${positionedNode.x}, ${positionedNode.y})`
 | 
			
		||||
      );
 | 
			
		||||
 | 
			
		||||
      // Store the final position
 | 
			
		||||
      node.x = positionedNode.x;
 | 
			
		||||
      node.y = positionedNode.y;
 | 
			
		||||
 | 
			
		||||
      log.debug(`Positioned node ${node.id} at center (${positionedNode.x}, ${positionedNode.y})`);
 | 
			
		||||
    }
 | 
			
		||||
  });
 | 
			
		||||
 | 
			
		||||
  layoutResult.edges.forEach((positionedEdge) => {
 | 
			
		||||
    const edge = data4Layout.edges.find((e) => e.id === positionedEdge.id);
 | 
			
		||||
    if (edge) {
 | 
			
		||||
      // Update the edge data with positioned coordinates
 | 
			
		||||
      edge.points = [
 | 
			
		||||
        { x: positionedEdge.startX, y: positionedEdge.startY },
 | 
			
		||||
        { x: positionedEdge.midX, y: positionedEdge.midY },
 | 
			
		||||
        { x: positionedEdge.endX, y: positionedEdge.endY },
 | 
			
		||||
      ];
 | 
			
		||||
    }
 | 
			
		||||
  });
 | 
			
		||||
 | 
			
		||||
  // Step 4: Insert and position edges
 | 
			
		||||
  log.debug('Inserting and positioning edges');
 | 
			
		||||
 | 
			
		||||
  await Promise.all(
 | 
			
		||||
    data4Layout.edges.map(async (edge) => {
 | 
			
		||||
      // Insert edge label first
 | 
			
		||||
      const _edgeLabel = await insertEdgeLabel(edgeLabels, edge);
 | 
			
		||||
 | 
			
		||||
      // Get start and end nodes
 | 
			
		||||
      const startNode = nodeDb[edge.start ?? ''];
 | 
			
		||||
      const endNode = nodeDb[edge.end ?? ''];
 | 
			
		||||
 | 
			
		||||
      if (startNode && endNode) {
 | 
			
		||||
        // Find the positioned edge data
 | 
			
		||||
        const positionedEdge = layoutResult.edges.find((e) => e.id === edge.id);
 | 
			
		||||
 | 
			
		||||
        if (positionedEdge) {
 | 
			
		||||
          log.debug('APA01 positionedEdge', positionedEdge);
 | 
			
		||||
          // Create edge path with positioned coordinates
 | 
			
		||||
          const edgeWithPath = { ...edge };
 | 
			
		||||
 | 
			
		||||
          // Insert the edge path
 | 
			
		||||
          const paths = insertEdge(
 | 
			
		||||
            edgePaths,
 | 
			
		||||
            edgeWithPath,
 | 
			
		||||
            clusterDb,
 | 
			
		||||
            data4Layout.type,
 | 
			
		||||
            startNode,
 | 
			
		||||
            endNode,
 | 
			
		||||
            data4Layout.diagramId
 | 
			
		||||
          );
 | 
			
		||||
 | 
			
		||||
          // Position the edge label
 | 
			
		||||
          positionEdgeLabel(edgeWithPath, paths);
 | 
			
		||||
        } else {
 | 
			
		||||
          // Fallback: create a simple straight line between nodes
 | 
			
		||||
          const edgeWithPath = {
 | 
			
		||||
            ...edge,
 | 
			
		||||
            points: [
 | 
			
		||||
              { x: startNode.x || 0, y: startNode.y || 0 },
 | 
			
		||||
              { x: endNode.x || 0, y: endNode.y || 0 },
 | 
			
		||||
            ],
 | 
			
		||||
          };
 | 
			
		||||
 | 
			
		||||
          const paths = insertEdge(
 | 
			
		||||
            edgePaths,
 | 
			
		||||
            edgeWithPath,
 | 
			
		||||
            clusterDb,
 | 
			
		||||
            data4Layout.type,
 | 
			
		||||
            startNode,
 | 
			
		||||
            endNode,
 | 
			
		||||
            data4Layout.diagramId
 | 
			
		||||
          );
 | 
			
		||||
          positionEdgeLabel(edgeWithPath, paths);
 | 
			
		||||
        }
 | 
			
		||||
      }
 | 
			
		||||
    })
 | 
			
		||||
  );
 | 
			
		||||
 | 
			
		||||
  log.debug('Cose-bilkent rendering completed');
 | 
			
		||||
};
 | 
			
		||||
@@ -0,0 +1,43 @@
 | 
			
		||||
/**
 | 
			
		||||
 * Positioned node after layout calculation
 | 
			
		||||
 */
 | 
			
		||||
export interface PositionedNode {
 | 
			
		||||
  id: string;
 | 
			
		||||
  x: number;
 | 
			
		||||
  y: number;
 | 
			
		||||
  [key: string]: unknown; // Allow additional properties
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Positioned edge after layout calculation
 | 
			
		||||
 */
 | 
			
		||||
export interface PositionedEdge {
 | 
			
		||||
  id: string;
 | 
			
		||||
  source: string;
 | 
			
		||||
  target: string;
 | 
			
		||||
  startX: number;
 | 
			
		||||
  startY: number;
 | 
			
		||||
  midX: number;
 | 
			
		||||
  midY: number;
 | 
			
		||||
  endX: number;
 | 
			
		||||
  endY: number;
 | 
			
		||||
  [key: string]: unknown; // Allow additional properties
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Result of layout algorithm execution
 | 
			
		||||
 */
 | 
			
		||||
export interface LayoutResult {
 | 
			
		||||
  nodes: PositionedNode[];
 | 
			
		||||
  edges: PositionedEdge[];
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Cytoscape layout configuration
 | 
			
		||||
 */
 | 
			
		||||
export interface CytoscapeLayoutConfig {
 | 
			
		||||
  name: 'cose-bilkent';
 | 
			
		||||
  quality: 'proof';
 | 
			
		||||
  styleEnabled: boolean;
 | 
			
		||||
  animate: boolean;
 | 
			
		||||
}
 | 
			
		||||
@@ -4,6 +4,9 @@ import { internalHelpers } from '../internals.js';
 | 
			
		||||
import { log } from '../logger.js';
 | 
			
		||||
import type { LayoutData } from './types.js';
 | 
			
		||||
 | 
			
		||||
// console.log('MUST be removed, this only for keeping dev server working');
 | 
			
		||||
// import tmp from './layout-algorithms/dagre/index.js';
 | 
			
		||||
 | 
			
		||||
export interface RenderOptions {
 | 
			
		||||
  algorithm?: string;
 | 
			
		||||
}
 | 
			
		||||
@@ -39,6 +42,10 @@ const registerDefaultLayoutLoaders = () => {
 | 
			
		||||
      name: 'dagre',
 | 
			
		||||
      loader: async () => await import('./layout-algorithms/dagre/index.js'),
 | 
			
		||||
    },
 | 
			
		||||
    {
 | 
			
		||||
      name: 'cose-bilkent',
 | 
			
		||||
      loader: async () => await import('./layout-algorithms/cose-bilkent/index.js'),
 | 
			
		||||
    },
 | 
			
		||||
  ]);
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -1,9 +1,13 @@
 | 
			
		||||
import { getConfig } from '../../diagram-api/diagramAPI.js';
 | 
			
		||||
import { evaluate, getUrl } from '../../diagrams/common/common.js';
 | 
			
		||||
import { evaluate } from '../../diagrams/common/common.js';
 | 
			
		||||
import { log } from '../../logger.js';
 | 
			
		||||
import { createText } from '../createText.js';
 | 
			
		||||
import utils from '../../utils.js';
 | 
			
		||||
import { getLineFunctionsWithOffset } from '../../utils/lineWithOffset.js';
 | 
			
		||||
import {
 | 
			
		||||
  getLineFunctionsWithOffset,
 | 
			
		||||
  markerOffsets,
 | 
			
		||||
  markerOffsets2,
 | 
			
		||||
} from '../../utils/lineWithOffset.js';
 | 
			
		||||
import { getSubGraphTitleMargins } from '../../utils/subGraphTitleMargins.js';
 | 
			
		||||
 | 
			
		||||
import {
 | 
			
		||||
@@ -27,8 +31,8 @@ import createLabel from './createLabel.js';
 | 
			
		||||
import { addEdgeMarkers } from './edgeMarker.ts';
 | 
			
		||||
import { isLabelStyle } from './shapes/handDrawnShapeStyles.js';
 | 
			
		||||
 | 
			
		||||
const edgeLabels = new Map();
 | 
			
		||||
const terminalLabels = new Map();
 | 
			
		||||
export const edgeLabels = new Map();
 | 
			
		||||
export const terminalLabels = new Map();
 | 
			
		||||
 | 
			
		||||
export const clear = () => {
 | 
			
		||||
  edgeLabels.clear();
 | 
			
		||||
@@ -55,7 +59,7 @@ export const insertEdgeLabel = async (elem, edge) => {
 | 
			
		||||
  const edgeLabel = elem.insert('g').attr('class', 'edgeLabel');
 | 
			
		||||
 | 
			
		||||
  // Create inner g, label, this will be positioned now for centering the text
 | 
			
		||||
  const label = edgeLabel.insert('g').attr('class', 'label');
 | 
			
		||||
  const label = edgeLabel.insert('g').attr('class', 'label').attr('data-id', edge.id);
 | 
			
		||||
  label.node().appendChild(labelElement);
 | 
			
		||||
 | 
			
		||||
  // Center the label
 | 
			
		||||
@@ -352,94 +356,33 @@ const cutPathAtIntersect = (_points, boundaryNode) => {
 | 
			
		||||
  return points;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
function extractCornerPoints(points) {
 | 
			
		||||
  const cornerPoints = [];
 | 
			
		||||
  const cornerPointPositions = [];
 | 
			
		||||
  for (let i = 1; i < points.length - 1; i++) {
 | 
			
		||||
    const prev = points[i - 1];
 | 
			
		||||
    const curr = points[i];
 | 
			
		||||
    const next = points[i + 1];
 | 
			
		||||
    if (
 | 
			
		||||
      prev.x === curr.x &&
 | 
			
		||||
      curr.y === next.y &&
 | 
			
		||||
      Math.abs(curr.x - next.x) > 5 &&
 | 
			
		||||
      Math.abs(curr.y - prev.y) > 5
 | 
			
		||||
    ) {
 | 
			
		||||
      cornerPoints.push(curr);
 | 
			
		||||
      cornerPointPositions.push(i);
 | 
			
		||||
    } else if (
 | 
			
		||||
      prev.y === curr.y &&
 | 
			
		||||
      curr.x === next.x &&
 | 
			
		||||
      Math.abs(curr.x - prev.x) > 5 &&
 | 
			
		||||
      Math.abs(curr.y - next.y) > 5
 | 
			
		||||
    ) {
 | 
			
		||||
      cornerPoints.push(curr);
 | 
			
		||||
      cornerPointPositions.push(i);
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
  return { cornerPoints, cornerPointPositions };
 | 
			
		||||
}
 | 
			
		||||
const generateDashArray = (len, oValueS, oValueE) => {
 | 
			
		||||
  const middleLength = len - oValueS - oValueE;
 | 
			
		||||
  const dashLength = 2; // Length of each dash
 | 
			
		||||
  const gapLength = 2; // Length of each gap
 | 
			
		||||
  const dashGapPairLength = dashLength + gapLength;
 | 
			
		||||
 | 
			
		||||
const findAdjacentPoint = function (pointA, pointB, distance) {
 | 
			
		||||
  const xDiff = pointB.x - pointA.x;
 | 
			
		||||
  const yDiff = pointB.y - pointA.y;
 | 
			
		||||
  const length = Math.sqrt(xDiff * xDiff + yDiff * yDiff);
 | 
			
		||||
  const ratio = distance / length;
 | 
			
		||||
  return { x: pointB.x - ratio * xDiff, y: pointB.y - ratio * yDiff };
 | 
			
		||||
  // Calculate number of complete dash-gap pairs that can fit
 | 
			
		||||
  const numberOfPairs = Math.floor(middleLength / dashGapPairLength);
 | 
			
		||||
 | 
			
		||||
  // Generate the middle pattern array
 | 
			
		||||
  const middlePattern = Array(numberOfPairs).fill(`${dashLength} ${gapLength}`).join(' ');
 | 
			
		||||
 | 
			
		||||
  // Combine all parts
 | 
			
		||||
  const dashArray = `0 ${oValueS} ${middlePattern} ${oValueE}`;
 | 
			
		||||
 | 
			
		||||
  return dashArray;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
const fixCorners = function (lineData) {
 | 
			
		||||
  const { cornerPointPositions } = extractCornerPoints(lineData);
 | 
			
		||||
  const newLineData = [];
 | 
			
		||||
  for (let i = 0; i < lineData.length; i++) {
 | 
			
		||||
    if (cornerPointPositions.includes(i)) {
 | 
			
		||||
      const prevPoint = lineData[i - 1];
 | 
			
		||||
      const nextPoint = lineData[i + 1];
 | 
			
		||||
      const cornerPoint = lineData[i];
 | 
			
		||||
 | 
			
		||||
      const newPrevPoint = findAdjacentPoint(prevPoint, cornerPoint, 5);
 | 
			
		||||
      const newNextPoint = findAdjacentPoint(nextPoint, cornerPoint, 5);
 | 
			
		||||
 | 
			
		||||
      const xDiff = newNextPoint.x - newPrevPoint.x;
 | 
			
		||||
      const yDiff = newNextPoint.y - newPrevPoint.y;
 | 
			
		||||
      newLineData.push(newPrevPoint);
 | 
			
		||||
 | 
			
		||||
      const a = Math.sqrt(2) * 2;
 | 
			
		||||
      let newCornerPoint = { x: cornerPoint.x, y: cornerPoint.y };
 | 
			
		||||
      if (Math.abs(nextPoint.x - prevPoint.x) > 10 && Math.abs(nextPoint.y - prevPoint.y) >= 10) {
 | 
			
		||||
        log.debug(
 | 
			
		||||
          'Corner point fixing',
 | 
			
		||||
          Math.abs(nextPoint.x - prevPoint.x),
 | 
			
		||||
          Math.abs(nextPoint.y - prevPoint.y)
 | 
			
		||||
        );
 | 
			
		||||
        const r = 5;
 | 
			
		||||
        if (cornerPoint.x === newPrevPoint.x) {
 | 
			
		||||
          newCornerPoint = {
 | 
			
		||||
            x: xDiff < 0 ? newPrevPoint.x - r + a : newPrevPoint.x + r - a,
 | 
			
		||||
            y: yDiff < 0 ? newPrevPoint.y - a : newPrevPoint.y + a,
 | 
			
		||||
          };
 | 
			
		||||
        } else {
 | 
			
		||||
          newCornerPoint = {
 | 
			
		||||
            x: xDiff < 0 ? newPrevPoint.x - a : newPrevPoint.x + a,
 | 
			
		||||
            y: yDiff < 0 ? newPrevPoint.y - r + a : newPrevPoint.y + r - a,
 | 
			
		||||
          };
 | 
			
		||||
        }
 | 
			
		||||
      } else {
 | 
			
		||||
        log.debug(
 | 
			
		||||
          'Corner point skipping fixing',
 | 
			
		||||
          Math.abs(nextPoint.x - prevPoint.x),
 | 
			
		||||
          Math.abs(nextPoint.y - prevPoint.y)
 | 
			
		||||
        );
 | 
			
		||||
      }
 | 
			
		||||
      newLineData.push(newCornerPoint, newNextPoint);
 | 
			
		||||
    } else {
 | 
			
		||||
      newLineData.push(lineData[i]);
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
  return newLineData;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
export const insertEdge = function (elem, edge, clusterDb, diagramType, startNode, endNode, id) {
 | 
			
		||||
export const insertEdge = function (
 | 
			
		||||
  elem,
 | 
			
		||||
  edge,
 | 
			
		||||
  clusterDb,
 | 
			
		||||
  diagramType,
 | 
			
		||||
  startNode,
 | 
			
		||||
  endNode,
 | 
			
		||||
  id,
 | 
			
		||||
  skipIntersect = false
 | 
			
		||||
) {
 | 
			
		||||
  const { handDrawnSeed } = getConfig();
 | 
			
		||||
  let points = edge.points;
 | 
			
		||||
  let pointsHasChanged = false;
 | 
			
		||||
@@ -453,11 +396,12 @@ export const insertEdge = function (elem, edge, clusterDb, diagramType, startNod
 | 
			
		||||
    edgeClassStyles.push(edge.cssCompiledStyles[key]);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  if (head.intersect && tail.intersect) {
 | 
			
		||||
  log.debug('UIO intersect check', edge.points, head.x, tail.x);
 | 
			
		||||
  if (head.intersect && tail.intersect && !skipIntersect) {
 | 
			
		||||
    points = points.slice(1, edge.points.length - 1);
 | 
			
		||||
    points.unshift(tail.intersect(points[0]));
 | 
			
		||||
    log.debug(
 | 
			
		||||
      'Last point APA12',
 | 
			
		||||
      'Last point UIO',
 | 
			
		||||
      edge.start,
 | 
			
		||||
      '-->',
 | 
			
		||||
      edge.end,
 | 
			
		||||
@@ -467,6 +411,7 @@ export const insertEdge = function (elem, edge, clusterDb, diagramType, startNod
 | 
			
		||||
    );
 | 
			
		||||
    points.push(head.intersect(points[points.length - 1]));
 | 
			
		||||
  }
 | 
			
		||||
  const pointsStr = btoa(JSON.stringify(points));
 | 
			
		||||
  if (edge.toCluster) {
 | 
			
		||||
    log.info('to cluster abc88', clusterDb.get(edge.toCluster));
 | 
			
		||||
    points = cutPathAtIntersect(edge.points, clusterDb.get(edge.toCluster).node);
 | 
			
		||||
@@ -486,7 +431,7 @@ export const insertEdge = function (elem, edge, clusterDb, diagramType, startNod
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  let lineData = points.filter((p) => !Number.isNaN(p.y));
 | 
			
		||||
  lineData = fixCorners(lineData);
 | 
			
		||||
  //lineData = fixCorners(lineData);
 | 
			
		||||
  let curve = curveBasis;
 | 
			
		||||
  curve = curveLinear;
 | 
			
		||||
  switch (edge.curve) {
 | 
			
		||||
@@ -530,6 +475,10 @@ export const insertEdge = function (elem, edge, clusterDb, diagramType, startNod
 | 
			
		||||
      curve = curveBasis;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  // if (edge.curve) {
 | 
			
		||||
  //   curve = edge.curve;
 | 
			
		||||
  // }
 | 
			
		||||
 | 
			
		||||
  const { x, y } = getLineFunctionsWithOffset(edge);
 | 
			
		||||
  const lineFunction = line().x(x).y(y).curve(curve);
 | 
			
		||||
 | 
			
		||||
@@ -561,10 +510,14 @@ export const insertEdge = function (elem, edge, clusterDb, diagramType, startNod
 | 
			
		||||
      strokeClasses += ' edge-pattern-solid';
 | 
			
		||||
  }
 | 
			
		||||
  let svgPath;
 | 
			
		||||
  let linePath = lineFunction(lineData);
 | 
			
		||||
  const edgeStyles = Array.isArray(edge.style) ? edge.style : edge.style ? [edge.style] : [];
 | 
			
		||||
  let linePath =
 | 
			
		||||
    edge.curve === 'rounded'
 | 
			
		||||
      ? generateRoundedPath(applyMarkerOffsetsToPoints(lineData, edge), 5)
 | 
			
		||||
      : lineFunction(lineData);
 | 
			
		||||
  const edgeStyles = Array.isArray(edge.style) ? edge.style : [edge.style];
 | 
			
		||||
  let strokeColor = edgeStyles.find((style) => style?.startsWith('stroke:'));
 | 
			
		||||
 | 
			
		||||
  let animatedEdge = false;
 | 
			
		||||
  if (edge.look === 'handDrawn') {
 | 
			
		||||
    const rc = rough.svg(elem);
 | 
			
		||||
    Object.assign([], lineData);
 | 
			
		||||
@@ -595,7 +548,10 @@ export const insertEdge = function (elem, edge, clusterDb, diagramType, startNod
 | 
			
		||||
      animationClass = ' edge-animation-' + edge.animation;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    const pathStyle = stylesFromClasses ? stylesFromClasses + ';' + styles + ';' : styles;
 | 
			
		||||
    const pathStyle =
 | 
			
		||||
      (stylesFromClasses ? stylesFromClasses + ';' + styles + ';' : styles) +
 | 
			
		||||
      ';' +
 | 
			
		||||
      (edgeStyles ? edgeStyles.reduce((acc, style) => acc + ';' + style, '') : '');
 | 
			
		||||
    svgPath = elem
 | 
			
		||||
      .append('path')
 | 
			
		||||
      .attr('d', linePath)
 | 
			
		||||
@@ -605,11 +561,38 @@ export const insertEdge = function (elem, edge, clusterDb, diagramType, startNod
 | 
			
		||||
        ' ' + strokeClasses + (edge.classes ? ' ' + edge.classes : '') + (animationClass ?? '')
 | 
			
		||||
      )
 | 
			
		||||
      .attr('style', pathStyle);
 | 
			
		||||
 | 
			
		||||
    //eslint-disable-next-line @typescript-eslint/prefer-regexp-exec
 | 
			
		||||
    strokeColor = pathStyle.match(/stroke:([^;]+)/)?.[1];
 | 
			
		||||
 | 
			
		||||
    // Possible fix to remove eslint-disable-next-line
 | 
			
		||||
    //strokeColor = /stroke:([^;]+)/.exec(pathStyle)?.[1];
 | 
			
		||||
 | 
			
		||||
    animatedEdge =
 | 
			
		||||
      edge.animate === true || !!edge.animation || stylesFromClasses.includes('animation');
 | 
			
		||||
    const len = svgPath.node().getTotalLength();
 | 
			
		||||
    const oValueS = markerOffsets2[edge.arrowTypeStart] || 0;
 | 
			
		||||
    const oValueE = markerOffsets2[edge.arrowTypeEnd] || 0;
 | 
			
		||||
 | 
			
		||||
    if (edge.look === 'neo' && !animatedEdge) {
 | 
			
		||||
      const dashArray =
 | 
			
		||||
        edge.pattern === 'dotted' || edge.pattern === 'dashed'
 | 
			
		||||
          ? generateDashArray(len, oValueS, oValueE)
 | 
			
		||||
          : `0 ${oValueS} ${len - oValueS - oValueE} ${oValueE}`;
 | 
			
		||||
 | 
			
		||||
      // No offset needed because we already start with a zero-length dash that effectively sets us up for a gap at the start.
 | 
			
		||||
      const mOffset = `stroke-dasharray: ${dashArray}; stroke-dashoffset: 0;`;
 | 
			
		||||
      svgPath.attr('style', mOffset + svgPath.attr('style'));
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  // DEBUG code, DO NOT REMOVE
 | 
			
		||||
  // adds a red circle at each edge coordinate
 | 
			
		||||
  // MC Special
 | 
			
		||||
  svgPath.attr('data-edge', true);
 | 
			
		||||
  svgPath.attr('data-et', 'edge');
 | 
			
		||||
  svgPath.attr('data-id', edge.id);
 | 
			
		||||
  svgPath.attr('data-points', pointsStr);
 | 
			
		||||
 | 
			
		||||
  // DEBUG code, adds a red circle at each edge coordinate
 | 
			
		||||
  // cornerPoints.forEach((point) => {
 | 
			
		||||
  //   elem
 | 
			
		||||
  //     .append('circle')
 | 
			
		||||
@@ -619,24 +602,33 @@ export const insertEdge = function (elem, edge, clusterDb, diagramType, startNod
 | 
			
		||||
  //     .attr('cx', point.x)
 | 
			
		||||
  //     .attr('cy', point.y);
 | 
			
		||||
  // });
 | 
			
		||||
  // lineData.forEach((point) => {
 | 
			
		||||
  //   elem
 | 
			
		||||
  //     .append('circle')
 | 
			
		||||
  //     .style('stroke', 'blue')
 | 
			
		||||
  //     .style('fill', 'blue')
 | 
			
		||||
  //     .attr('r', 3)
 | 
			
		||||
  //     .attr('cx', point.x)
 | 
			
		||||
  //     .attr('cy', point.y);
 | 
			
		||||
  // });
 | 
			
		||||
  if (edge.showPoints) {
 | 
			
		||||
    lineData.forEach((point) => {
 | 
			
		||||
      elem
 | 
			
		||||
        .append('circle')
 | 
			
		||||
        .style('stroke', 'red')
 | 
			
		||||
        .style('fill', 'red')
 | 
			
		||||
        .attr('r', 1)
 | 
			
		||||
        .attr('cx', point.x)
 | 
			
		||||
        .attr('cy', point.y);
 | 
			
		||||
    });
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  let url = '';
 | 
			
		||||
  if (getConfig().flowchart.arrowMarkerAbsolute || getConfig().state.arrowMarkerAbsolute) {
 | 
			
		||||
    url = getUrl(true);
 | 
			
		||||
    url =
 | 
			
		||||
      window.location.protocol +
 | 
			
		||||
      '//' +
 | 
			
		||||
      window.location.host +
 | 
			
		||||
      window.location.pathname +
 | 
			
		||||
      window.location.search;
 | 
			
		||||
    url = url.replace(/\(/g, '\\(').replace(/\)/g, '\\)');
 | 
			
		||||
  }
 | 
			
		||||
  log.info('arrowTypeStart', edge.arrowTypeStart);
 | 
			
		||||
  log.info('arrowTypeEnd', edge.arrowTypeEnd);
 | 
			
		||||
 | 
			
		||||
  addEdgeMarkers(svgPath, edge, url, id, diagramType, strokeColor);
 | 
			
		||||
  const useMargin = !animatedEdge && edge?.look === 'neo';
 | 
			
		||||
  addEdgeMarkers(svgPath, edge, url, id, diagramType, useMargin, strokeColor);
 | 
			
		||||
  const midIndex = Math.floor(points.length / 2);
 | 
			
		||||
  const point = points[midIndex];
 | 
			
		||||
  if (!utils.isLabelCoordinateInPath(point, svgPath.attr('d'))) {
 | 
			
		||||
@@ -650,3 +642,134 @@ export const insertEdge = function (elem, edge, clusterDb, diagramType, startNod
 | 
			
		||||
  paths.originalPath = edge.points;
 | 
			
		||||
  return paths;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Generates SVG path data with rounded corners from an array of points.
 | 
			
		||||
 * @param {Array} points - Array of points in the format [{x: Number, y: Number}, ...]
 | 
			
		||||
 * @param {Number} radius - The radius of the rounded corners
 | 
			
		||||
 * @returns {String} - SVG path data string
 | 
			
		||||
 */
 | 
			
		||||
function generateRoundedPath(points, radius) {
 | 
			
		||||
  if (points.length < 2) {
 | 
			
		||||
    return '';
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  let path = '';
 | 
			
		||||
  const size = points.length;
 | 
			
		||||
  const epsilon = 1e-5;
 | 
			
		||||
 | 
			
		||||
  for (let i = 0; i < size; i++) {
 | 
			
		||||
    const currPoint = points[i];
 | 
			
		||||
    const prevPoint = points[i - 1];
 | 
			
		||||
    const nextPoint = points[i + 1];
 | 
			
		||||
 | 
			
		||||
    if (i === 0) {
 | 
			
		||||
      // Move to the first point
 | 
			
		||||
      path += `M${currPoint.x},${currPoint.y}`;
 | 
			
		||||
    } else if (i === size - 1) {
 | 
			
		||||
      // Last point, draw a straight line to the final point
 | 
			
		||||
      path += `L${currPoint.x},${currPoint.y}`;
 | 
			
		||||
    } else {
 | 
			
		||||
      // Calculate vectors for incoming and outgoing segments
 | 
			
		||||
      const dx1 = currPoint.x - prevPoint.x;
 | 
			
		||||
      const dy1 = currPoint.y - prevPoint.y;
 | 
			
		||||
      const dx2 = nextPoint.x - currPoint.x;
 | 
			
		||||
      const dy2 = nextPoint.y - currPoint.y;
 | 
			
		||||
 | 
			
		||||
      const len1 = Math.hypot(dx1, dy1);
 | 
			
		||||
      const len2 = Math.hypot(dx2, dy2);
 | 
			
		||||
 | 
			
		||||
      // Prevent division by zero
 | 
			
		||||
      if (len1 < epsilon || len2 < epsilon) {
 | 
			
		||||
        path += `L${currPoint.x},${currPoint.y}`;
 | 
			
		||||
        continue;
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      // Normalize the vectors
 | 
			
		||||
      const nx1 = dx1 / len1;
 | 
			
		||||
      const ny1 = dy1 / len1;
 | 
			
		||||
      const nx2 = dx2 / len2;
 | 
			
		||||
      const ny2 = dy2 / len2;
 | 
			
		||||
 | 
			
		||||
      // Calculate the angle between the vectors
 | 
			
		||||
      const dot = nx1 * nx2 + ny1 * ny2;
 | 
			
		||||
      // Clamp the dot product to avoid numerical issues with acos
 | 
			
		||||
      const clampedDot = Math.max(-1, Math.min(1, dot));
 | 
			
		||||
      const angle = Math.acos(clampedDot);
 | 
			
		||||
 | 
			
		||||
      // Skip rounding if the angle is too small or too close to 180 degrees
 | 
			
		||||
      if (angle < epsilon || Math.abs(Math.PI - angle) < epsilon) {
 | 
			
		||||
        path += `L${currPoint.x},${currPoint.y}`;
 | 
			
		||||
        continue;
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      // Calculate the distance to offset the control point
 | 
			
		||||
      const cutLen = Math.min(radius / Math.sin(angle / 2), len1 / 2, len2 / 2);
 | 
			
		||||
 | 
			
		||||
      // Calculate the start and end points of the curve
 | 
			
		||||
      const startX = currPoint.x - nx1 * cutLen;
 | 
			
		||||
      const startY = currPoint.y - ny1 * cutLen;
 | 
			
		||||
      const endX = currPoint.x + nx2 * cutLen;
 | 
			
		||||
      const endY = currPoint.y + ny2 * cutLen;
 | 
			
		||||
 | 
			
		||||
      // Draw the line to the start of the curve
 | 
			
		||||
      path += `L${startX},${startY}`;
 | 
			
		||||
 | 
			
		||||
      // Draw the quadratic Bezier curve
 | 
			
		||||
      path += `Q${currPoint.x},${currPoint.y} ${endX},${endY}`;
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  return path;
 | 
			
		||||
}
 | 
			
		||||
// Helper function to calculate delta and angle between two points
 | 
			
		||||
function calculateDeltaAndAngle(point1, point2) {
 | 
			
		||||
  if (!point1 || !point2) {
 | 
			
		||||
    return { angle: 0, deltaX: 0, deltaY: 0 };
 | 
			
		||||
  }
 | 
			
		||||
  const deltaX = point2.x - point1.x;
 | 
			
		||||
  const deltaY = point2.y - point1.y;
 | 
			
		||||
  const angle = Math.atan2(deltaY, deltaX);
 | 
			
		||||
  return { angle, deltaX, deltaY };
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Function to adjust the first and last points of the points array
 | 
			
		||||
function applyMarkerOffsetsToPoints(points, edge) {
 | 
			
		||||
  // Copy the points array to avoid mutating the original data
 | 
			
		||||
  const newPoints = points.map((point) => ({ ...point }));
 | 
			
		||||
 | 
			
		||||
  // Handle the first point (start of the edge)
 | 
			
		||||
  if (points.length >= 2 && markerOffsets[edge.arrowTypeStart]) {
 | 
			
		||||
    const offsetValue = markerOffsets[edge.arrowTypeStart];
 | 
			
		||||
 | 
			
		||||
    const point1 = points[0];
 | 
			
		||||
    const point2 = points[1];
 | 
			
		||||
 | 
			
		||||
    const { angle } = calculateDeltaAndAngle(point1, point2);
 | 
			
		||||
 | 
			
		||||
    const offsetX = offsetValue * Math.cos(angle);
 | 
			
		||||
    const offsetY = offsetValue * Math.sin(angle);
 | 
			
		||||
 | 
			
		||||
    newPoints[0].x = point1.x + offsetX;
 | 
			
		||||
    newPoints[0].y = point1.y + offsetY;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  // Handle the last point (end of the edge)
 | 
			
		||||
  const n = points.length;
 | 
			
		||||
  if (n >= 2 && markerOffsets[edge.arrowTypeEnd]) {
 | 
			
		||||
    const offsetValue = markerOffsets[edge.arrowTypeEnd];
 | 
			
		||||
 | 
			
		||||
    const point1 = points[n - 1];
 | 
			
		||||
    const point2 = points[n - 2];
 | 
			
		||||
 | 
			
		||||
    const { angle } = calculateDeltaAndAngle(point2, point1);
 | 
			
		||||
 | 
			
		||||
    const offsetX = offsetValue * Math.cos(angle);
 | 
			
		||||
    const offsetY = offsetValue * Math.sin(angle);
 | 
			
		||||
 | 
			
		||||
    newPoints[n - 1].x = point1.x - offsetX;
 | 
			
		||||
    newPoints[n - 1].y = point1.y - offsetY;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  return newPoints;
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -2,64 +2,63 @@
 | 
			
		||||
 * Returns the point at which two lines, p and q, intersect or returns undefined if they do not intersect.
 | 
			
		||||
 */
 | 
			
		||||
function intersectLine(p1, p2, q1, q2) {
 | 
			
		||||
  // Algorithm from J. Avro, (ed.) Graphics Gems, No 2, Morgan Kaufmann, 1994,
 | 
			
		||||
  // p7 and p473.
 | 
			
		||||
  {
 | 
			
		||||
    // Algorithm from J. Avro, (ed.) Graphics Gems, No 2, Morgan Kaufmann, 1994,
 | 
			
		||||
    // p7 and p473.
 | 
			
		||||
 | 
			
		||||
  var a1, a2, b1, b2, c1, c2;
 | 
			
		||||
  var r1, r2, r3, r4;
 | 
			
		||||
  var denom, offset, num;
 | 
			
		||||
  var x, y;
 | 
			
		||||
    // Compute a1, b1, c1, where line joining points 1 and 2 is F(x,y) = a1 x +
 | 
			
		||||
    // b1 y + c1 = 0.
 | 
			
		||||
    const a1 = p2.y - p1.y;
 | 
			
		||||
    const b1 = p1.x - p2.x;
 | 
			
		||||
    const c1 = p2.x * p1.y - p1.x * p2.y;
 | 
			
		||||
 | 
			
		||||
  // Compute a1, b1, c1, where line joining points 1 and 2 is F(x,y) = a1 x +
 | 
			
		||||
  // b1 y + c1 = 0.
 | 
			
		||||
  a1 = p2.y - p1.y;
 | 
			
		||||
  b1 = p1.x - p2.x;
 | 
			
		||||
  c1 = p2.x * p1.y - p1.x * p2.y;
 | 
			
		||||
    // Compute r3 and r4.
 | 
			
		||||
    const r3 = a1 * q1.x + b1 * q1.y + c1;
 | 
			
		||||
    const r4 = a1 * q2.x + b1 * q2.y + c1;
 | 
			
		||||
 | 
			
		||||
  // Compute r3 and r4.
 | 
			
		||||
  r3 = a1 * q1.x + b1 * q1.y + c1;
 | 
			
		||||
  r4 = a1 * q2.x + b1 * q2.y + c1;
 | 
			
		||||
    const epsilon = 1e-6;
 | 
			
		||||
 | 
			
		||||
  // Check signs of r3 and r4. If both point 3 and point 4 lie on
 | 
			
		||||
  // same side of line 1, the line segments do not intersect.
 | 
			
		||||
  if (r3 !== 0 && r4 !== 0 && sameSign(r3, r4)) {
 | 
			
		||||
    return /*DON'T_INTERSECT*/;
 | 
			
		||||
    // Check signs of r3 and r4. If both point 3 and point 4 lie on
 | 
			
		||||
    // same side of line 1, the line segments do not intersect.
 | 
			
		||||
    if (r3 !== 0 && r4 !== 0 && sameSign(r3, r4)) {
 | 
			
		||||
      return /*DON'T_INTERSECT*/;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    // Compute a2, b2, c2 where line joining points 3 and 4 is G(x,y) = a2 x + b2 y + c2 = 0
 | 
			
		||||
    const a2 = q2.y - q1.y;
 | 
			
		||||
    const b2 = q1.x - q2.x;
 | 
			
		||||
    const c2 = q2.x * q1.y - q1.x * q2.y;
 | 
			
		||||
 | 
			
		||||
    // Compute r1 and r2
 | 
			
		||||
    const r1 = a2 * p1.x + b2 * p1.y + c2;
 | 
			
		||||
    const r2 = a2 * p2.x + b2 * p2.y + c2;
 | 
			
		||||
 | 
			
		||||
    // Check signs of r1 and r2. If both point 1 and point 2 lie
 | 
			
		||||
    // on same side of second line segment, the line segments do
 | 
			
		||||
    // not intersect.
 | 
			
		||||
    if (Math.abs(r1) < epsilon && Math.abs(r2) < epsilon && sameSign(r1, r2)) {
 | 
			
		||||
      return /*DON'T_INTERSECT*/;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    // Line segments intersect: compute intersection point.
 | 
			
		||||
    const denom = a1 * b2 - a2 * b1;
 | 
			
		||||
    if (denom === 0) {
 | 
			
		||||
      return /*COLLINEAR*/;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    const offset = Math.abs(denom / 2);
 | 
			
		||||
 | 
			
		||||
    // The denom/2 is to get rounding instead of truncating. It
 | 
			
		||||
    // is added or subtracted to the numerator, depending upon the
 | 
			
		||||
    // sign of the numerator.
 | 
			
		||||
    let num = b1 * c2 - b2 * c1;
 | 
			
		||||
    const x = num < 0 ? (num - offset) / denom : (num + offset) / denom;
 | 
			
		||||
 | 
			
		||||
    num = a2 * c1 - a1 * c2;
 | 
			
		||||
    const y = num < 0 ? (num - offset) / denom : (num + offset) / denom;
 | 
			
		||||
 | 
			
		||||
    return { x: x, y: y };
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  // Compute a2, b2, c2 where line joining points 3 and 4 is G(x,y) = a2 x + b2 y + c2 = 0
 | 
			
		||||
  a2 = q2.y - q1.y;
 | 
			
		||||
  b2 = q1.x - q2.x;
 | 
			
		||||
  c2 = q2.x * q1.y - q1.x * q2.y;
 | 
			
		||||
 | 
			
		||||
  // Compute r1 and r2
 | 
			
		||||
  r1 = a2 * p1.x + b2 * p1.y + c2;
 | 
			
		||||
  r2 = a2 * p2.x + b2 * p2.y + c2;
 | 
			
		||||
 | 
			
		||||
  // Check signs of r1 and r2. If both point 1 and point 2 lie
 | 
			
		||||
  // on same side of second line segment, the line segments do
 | 
			
		||||
  // not intersect.
 | 
			
		||||
  if (r1 !== 0 && r2 !== 0 && sameSign(r1, r2)) {
 | 
			
		||||
    return /*DON'T_INTERSECT*/;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  // Line segments intersect: compute intersection point.
 | 
			
		||||
  denom = a1 * b2 - a2 * b1;
 | 
			
		||||
  if (denom === 0) {
 | 
			
		||||
    return /*COLLINEAR*/;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  offset = Math.abs(denom / 2);
 | 
			
		||||
 | 
			
		||||
  // The denom/2 is to get rounding instead of truncating. It
 | 
			
		||||
  // is added or subtracted to the numerator, depending upon the
 | 
			
		||||
  // sign of the numerator.
 | 
			
		||||
  num = b1 * c2 - b2 * c1;
 | 
			
		||||
  x = num < 0 ? (num - offset) / denom : (num + offset) / denom;
 | 
			
		||||
 | 
			
		||||
  num = a2 * c1 - a1 * c2;
 | 
			
		||||
  y = num < 0 ? (num - offset) / denom : (num + offset) / denom;
 | 
			
		||||
 | 
			
		||||
  return { x: x, y: y };
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function sameSign(r1, r2) {
 | 
			
		||||
 
 | 
			
		||||
@@ -61,6 +61,8 @@ import { erBox } from './shapes/erBox.js';
 | 
			
		||||
import { classBox } from './shapes/classBox.js';
 | 
			
		||||
import { requirementBox } from './shapes/requirementBox.js';
 | 
			
		||||
import { kanbanItem } from './shapes/kanbanItem.js';
 | 
			
		||||
import { bang } from './shapes/bang.js';
 | 
			
		||||
import { cloud } from './shapes/cloud.js';
 | 
			
		||||
 | 
			
		||||
type ShapeHandler = <T extends SVGGraphicsElement>(
 | 
			
		||||
  parent: D3Selection<T>,
 | 
			
		||||
@@ -135,6 +137,22 @@ export const shapesDefs = [
 | 
			
		||||
    aliases: ['circ'],
 | 
			
		||||
    handler: circle,
 | 
			
		||||
  },
 | 
			
		||||
  {
 | 
			
		||||
    semanticName: 'Bang',
 | 
			
		||||
    name: 'Bang',
 | 
			
		||||
    shortName: 'bang',
 | 
			
		||||
    description: 'Bang',
 | 
			
		||||
    aliases: ['bang'],
 | 
			
		||||
    handler: bang,
 | 
			
		||||
  },
 | 
			
		||||
  {
 | 
			
		||||
    semanticName: 'Cloud',
 | 
			
		||||
    name: 'Cloud',
 | 
			
		||||
    shortName: 'cloud',
 | 
			
		||||
    description: 'cloud',
 | 
			
		||||
    aliases: ['cloud'],
 | 
			
		||||
    handler: cloud,
 | 
			
		||||
  },
 | 
			
		||||
  {
 | 
			
		||||
    semanticName: 'Decision',
 | 
			
		||||
    name: 'Diamond',
 | 
			
		||||
 
 | 
			
		||||
@@ -0,0 +1,77 @@
 | 
			
		||||
import { log } from '../../../logger.js';
 | 
			
		||||
import { labelHelper, updateNodeBounds, getNodeClasses } from './util.js';
 | 
			
		||||
import intersect from '../intersect/index.js';
 | 
			
		||||
import type { Node } from '../../types.js';
 | 
			
		||||
import { styles2String, userNodeOverrides } from './handDrawnShapeStyles.js';
 | 
			
		||||
import rough from 'roughjs';
 | 
			
		||||
import type { D3Selection } from '../../../types.js';
 | 
			
		||||
import { handleUndefinedAttr } from '../../../utils.js';
 | 
			
		||||
import type { Bounds, Point } from '../../../types.js';
 | 
			
		||||
 | 
			
		||||
export async function bang<T extends SVGGraphicsElement>(parent: D3Selection<T>, node: Node) {
 | 
			
		||||
  const { labelStyles, nodeStyles } = styles2String(node);
 | 
			
		||||
  node.labelStyle = labelStyles;
 | 
			
		||||
  const { shapeSvg, bbox, halfPadding, label } = await labelHelper(
 | 
			
		||||
    parent,
 | 
			
		||||
    node,
 | 
			
		||||
    getNodeClasses(node)
 | 
			
		||||
  );
 | 
			
		||||
 | 
			
		||||
  const w = bbox.width + 8 * halfPadding;
 | 
			
		||||
  const h = bbox.height + 2 * halfPadding;
 | 
			
		||||
  const r = 0.15 * w;
 | 
			
		||||
  const { cssStyles } = node;
 | 
			
		||||
 | 
			
		||||
  // Label centered around (0,0)
 | 
			
		||||
  label.attr('transform', `translate(${-bbox.width / 2}, ${-bbox.height / 2})`);
 | 
			
		||||
 | 
			
		||||
  let bangElem;
 | 
			
		||||
  const path = `M0 0 
 | 
			
		||||
    a${r},${r} 1 0,0 ${w * 0.25},${-1 * h * 0.1}
 | 
			
		||||
    a${r},${r} 1 0,0 ${w * 0.25},${0}
 | 
			
		||||
    a${r},${r} 1 0,0 ${w * 0.25},${0}
 | 
			
		||||
    a${r},${r} 1 0,0 ${w * 0.25},${h * 0.1}
 | 
			
		||||
 | 
			
		||||
    a${r},${r} 1 0,0 ${w * 0.15},${h * 0.33}
 | 
			
		||||
    a${r * 0.8},${r * 0.8} 1 0,0 0,${h * 0.34}
 | 
			
		||||
    a${r},${r} 1 0,0 ${-1 * w * 0.15},${h * 0.33}
 | 
			
		||||
 | 
			
		||||
    a${r},${r} 1 0,0 ${-1 * w * 0.25},${h * 0.15}
 | 
			
		||||
    a${r},${r} 1 0,0 ${-1 * w * 0.25},0
 | 
			
		||||
    a${r},${r} 1 0,0 ${-1 * w * 0.25},0
 | 
			
		||||
    a${r},${r} 1 0,0 ${-1 * w * 0.25},${-1 * h * 0.15}
 | 
			
		||||
 | 
			
		||||
    a${r},${r} 1 0,0 ${-1 * w * 0.1},${-1 * h * 0.33}
 | 
			
		||||
    a${r * 0.8},${r * 0.8} 1 0,0 0,${-1 * h * 0.34}
 | 
			
		||||
    a${r},${r} 1 0,0 ${w * 0.1},${-1 * h * 0.33}
 | 
			
		||||
  H0 V0 Z`;
 | 
			
		||||
 | 
			
		||||
  if (node.look === 'handDrawn') {
 | 
			
		||||
    // @ts-expect-error -- Passing a D3.Selection seems to work for some reason
 | 
			
		||||
    const rc = rough.svg(shapeSvg);
 | 
			
		||||
    const options = userNodeOverrides(node, {});
 | 
			
		||||
    const roughNode = rc.path(path, options);
 | 
			
		||||
    bangElem = shapeSvg.insert(() => roughNode, ':first-child');
 | 
			
		||||
    bangElem.attr('class', 'basic label-container').attr('style', handleUndefinedAttr(cssStyles));
 | 
			
		||||
  } else {
 | 
			
		||||
    bangElem = shapeSvg
 | 
			
		||||
      .insert('path', ':first-child')
 | 
			
		||||
      .attr('class', 'basic label-container')
 | 
			
		||||
      .attr('style', nodeStyles)
 | 
			
		||||
      .attr('d', path);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  // Translate the path (center the shape)
 | 
			
		||||
  bangElem.attr('transform', `translate(${-w / 2}, ${-h / 2})`);
 | 
			
		||||
 | 
			
		||||
  updateNodeBounds(node, bangElem);
 | 
			
		||||
  node.calcIntersect = function (bounds: Bounds, point: Point) {
 | 
			
		||||
    return intersect.rect(bounds, point);
 | 
			
		||||
  };
 | 
			
		||||
  node.intersect = function (point) {
 | 
			
		||||
    log.info('Bang intersect', node, point);
 | 
			
		||||
    return intersect.rect(node, point);
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  return shapeSvg;
 | 
			
		||||
}
 | 
			
		||||
@@ -6,6 +6,7 @@ import { styles2String, userNodeOverrides } from './handDrawnShapeStyles.js';
 | 
			
		||||
import rough from 'roughjs';
 | 
			
		||||
import type { D3Selection } from '../../../types.js';
 | 
			
		||||
import { handleUndefinedAttr } from '../../../utils.js';
 | 
			
		||||
import type { Bounds, Point } from '../../../types.js';
 | 
			
		||||
 | 
			
		||||
export async function circle<T extends SVGGraphicsElement>(parent: D3Selection<T>, node: Node) {
 | 
			
		||||
  const { labelStyles, nodeStyles } = styles2String(node);
 | 
			
		||||
@@ -35,7 +36,10 @@ export async function circle<T extends SVGGraphicsElement>(parent: D3Selection<T
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  updateNodeBounds(node, circleElem);
 | 
			
		||||
 | 
			
		||||
  node.calcIntersect = function (bounds: Bounds, point: Point) {
 | 
			
		||||
    const radius = bounds.width / 2;
 | 
			
		||||
    return intersect.circle(bounds, radius, point);
 | 
			
		||||
  };
 | 
			
		||||
  node.intersect = function (point) {
 | 
			
		||||
    log.info('Circle intersect', node, radius, point);
 | 
			
		||||
    return intersect.circle(node, radius, point);
 | 
			
		||||
 
 | 
			
		||||
@@ -0,0 +1,81 @@
 | 
			
		||||
import { log } from '../../../logger.js';
 | 
			
		||||
import { labelHelper, updateNodeBounds, getNodeClasses } from './util.js';
 | 
			
		||||
import intersect from '../intersect/index.js';
 | 
			
		||||
import type { Node } from '../../types.js';
 | 
			
		||||
import { styles2String, userNodeOverrides } from './handDrawnShapeStyles.js';
 | 
			
		||||
import rough from 'roughjs';
 | 
			
		||||
import type { D3Selection } from '../../../types.js';
 | 
			
		||||
import { handleUndefinedAttr } from '../../../utils.js';
 | 
			
		||||
import type { Bounds, Point } from '../../../types.js';
 | 
			
		||||
 | 
			
		||||
export async function cloud<T extends SVGGraphicsElement>(parent: D3Selection<T>, node: Node) {
 | 
			
		||||
  const { labelStyles, nodeStyles } = styles2String(node);
 | 
			
		||||
  node.labelStyle = labelStyles;
 | 
			
		||||
 | 
			
		||||
  const { shapeSvg, bbox, halfPadding, label } = await labelHelper(
 | 
			
		||||
    parent,
 | 
			
		||||
    node,
 | 
			
		||||
    getNodeClasses(node)
 | 
			
		||||
  );
 | 
			
		||||
 | 
			
		||||
  const w = bbox.width + 2 * halfPadding;
 | 
			
		||||
  const h = bbox.height + 2 * halfPadding;
 | 
			
		||||
 | 
			
		||||
  // Cloud radii
 | 
			
		||||
  const r1 = 0.15 * w;
 | 
			
		||||
  const r2 = 0.25 * w;
 | 
			
		||||
  const r3 = 0.35 * w;
 | 
			
		||||
  const r4 = 0.2 * w;
 | 
			
		||||
 | 
			
		||||
  const { cssStyles } = node;
 | 
			
		||||
  let cloudElem;
 | 
			
		||||
 | 
			
		||||
  // Cloud path
 | 
			
		||||
  const path = `M0 0 
 | 
			
		||||
    a${r1},${r1} 0 0,1 ${w * 0.25},${-1 * w * 0.1}
 | 
			
		||||
    a${r3},${r3} 1 0,1 ${w * 0.4},${-1 * w * 0.1}
 | 
			
		||||
    a${r2},${r2} 1 0,1 ${w * 0.35},${w * 0.2}
 | 
			
		||||
 | 
			
		||||
    a${r1},${r1} 1 0,1 ${w * 0.15},${h * 0.35}
 | 
			
		||||
    a${r4},${r4} 1 0,1 ${-1 * w * 0.15},${h * 0.65}
 | 
			
		||||
 | 
			
		||||
    a${r2},${r1} 1 0,1 ${-1 * w * 0.25},${w * 0.15}
 | 
			
		||||
    a${r3},${r3} 1 0,1 ${-1 * w * 0.5},0
 | 
			
		||||
    a${r1},${r1} 1 0,1 ${-1 * w * 0.25},${-1 * w * 0.15}
 | 
			
		||||
 | 
			
		||||
    a${r1},${r1} 1 0,1 ${-1 * w * 0.1},${-1 * h * 0.35}
 | 
			
		||||
    a${r4},${r4} 1 0,1 ${w * 0.1},${-1 * h * 0.65}
 | 
			
		||||
  H0 V0 Z`;
 | 
			
		||||
 | 
			
		||||
  if (node.look === 'handDrawn') {
 | 
			
		||||
    // @ts-expect-error -- Passing a D3.Selection seems to work for some reason
 | 
			
		||||
    const rc = rough.svg(shapeSvg);
 | 
			
		||||
    const options = userNodeOverrides(node, {});
 | 
			
		||||
    const roughNode = rc.path(path, options);
 | 
			
		||||
    cloudElem = shapeSvg.insert(() => roughNode, ':first-child');
 | 
			
		||||
    cloudElem.attr('class', 'basic label-container').attr('style', handleUndefinedAttr(cssStyles));
 | 
			
		||||
  } else {
 | 
			
		||||
    cloudElem = shapeSvg
 | 
			
		||||
      .insert('path', ':first-child')
 | 
			
		||||
      .attr('class', 'basic label-container')
 | 
			
		||||
      .attr('style', nodeStyles)
 | 
			
		||||
      .attr('d', path);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  label.attr('transform', `translate(${-bbox.width / 2}, ${-bbox.height / 2})`);
 | 
			
		||||
 | 
			
		||||
  // Center the shape
 | 
			
		||||
  cloudElem.attr('transform', `translate(${-w / 2}, ${-h / 2})`);
 | 
			
		||||
 | 
			
		||||
  updateNodeBounds(node, cloudElem);
 | 
			
		||||
 | 
			
		||||
  node.calcIntersect = function (bounds: Bounds, point: Point) {
 | 
			
		||||
    return intersect.rect(bounds, point);
 | 
			
		||||
  };
 | 
			
		||||
  node.intersect = function (point) {
 | 
			
		||||
    log.info('Cloud intersect', node, point);
 | 
			
		||||
    return intersect.rect(node, point);
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  return shapeSvg;
 | 
			
		||||
}
 | 
			
		||||
@@ -6,6 +6,7 @@ import { userNodeOverrides, styles2String } from './handDrawnShapeStyles.js';
 | 
			
		||||
import rough from 'roughjs';
 | 
			
		||||
import type { D3Selection } from '../../../types.js';
 | 
			
		||||
import { handleUndefinedAttr } from '../../../utils.js';
 | 
			
		||||
import type { Bounds, Point } from '../../../types.js';
 | 
			
		||||
 | 
			
		||||
export async function drawRect<T extends SVGGraphicsElement>(
 | 
			
		||||
  parent: D3Selection<T>,
 | 
			
		||||
@@ -62,6 +63,10 @@ export async function drawRect<T extends SVGGraphicsElement>(
 | 
			
		||||
 | 
			
		||||
  updateNodeBounds(node, rect);
 | 
			
		||||
 | 
			
		||||
  node.calcIntersect = function (bounds: Bounds, point: Point) {
 | 
			
		||||
    return intersect.rect(bounds, point);
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  node.intersect = function (point) {
 | 
			
		||||
    return intersect.rect(node, point);
 | 
			
		||||
  };
 | 
			
		||||
 
 | 
			
		||||
@@ -1,4 +1,3 @@
 | 
			
		||||
import { log } from '../../../logger.js';
 | 
			
		||||
import { labelHelper, updateNodeBounds, getNodeClasses } from './util.js';
 | 
			
		||||
import intersect from '../intersect/index.js';
 | 
			
		||||
import type { Node } from '../../types.js';
 | 
			
		||||
@@ -6,6 +5,7 @@ import { styles2String, userNodeOverrides } from './handDrawnShapeStyles.js';
 | 
			
		||||
import rough from 'roughjs';
 | 
			
		||||
import { insertPolygonShape } from './insertPolygonShape.js';
 | 
			
		||||
import type { D3Selection } from '../../../types.js';
 | 
			
		||||
import type { Bounds, Point } from '../../../types.js';
 | 
			
		||||
 | 
			
		||||
export const createDecisionBoxPathD = (x: number, y: number, size: number): string => {
 | 
			
		||||
  return [
 | 
			
		||||
@@ -61,17 +61,26 @@ export async function question<T extends SVGGraphicsElement>(parent: D3Selection
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  updateNodeBounds(node, polygon);
 | 
			
		||||
  node.calcIntersect = function (bounds: Bounds, point: Point) {
 | 
			
		||||
    const s = bounds.width;
 | 
			
		||||
 | 
			
		||||
    // Define polygon points
 | 
			
		||||
    const points = [
 | 
			
		||||
      { x: s / 2, y: 0 },
 | 
			
		||||
      { x: s, y: -s / 2 },
 | 
			
		||||
      { x: s / 2, y: -s },
 | 
			
		||||
      { x: 0, y: -s / 2 },
 | 
			
		||||
    ];
 | 
			
		||||
 | 
			
		||||
    // Calculate the intersection point
 | 
			
		||||
    const res = intersect.polygon(bounds, points, point);
 | 
			
		||||
 | 
			
		||||
    return { x: res.x - 0.5, y: res.y - 0.5 }; // Adjusted result
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  node.intersect = function (point) {
 | 
			
		||||
    log.debug(
 | 
			
		||||
      'APA12 Intersect called SPLIT\npoint:',
 | 
			
		||||
      point,
 | 
			
		||||
      '\nnode:\n',
 | 
			
		||||
      node,
 | 
			
		||||
      '\nres:',
 | 
			
		||||
      intersect.polygon(node, points, point)
 | 
			
		||||
    );
 | 
			
		||||
    return intersect.polygon(node, points, point);
 | 
			
		||||
    // @ts-ignore TODO fix this (KNSV)
 | 
			
		||||
    return this.calcIntersect(node as Bounds, point);
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  return shapeSvg;
 | 
			
		||||
 
 | 
			
		||||
@@ -2,6 +2,7 @@ export type MarkdownWordType = 'normal' | 'strong' | 'em';
 | 
			
		||||
import type { MermaidConfig } from '../config.type.js';
 | 
			
		||||
import type { ClusterShapeID } from './rendering-elements/clusters.js';
 | 
			
		||||
import type { ShapeID } from './rendering-elements/shapes.js';
 | 
			
		||||
import type { Bounds, Point } from '../types.js';
 | 
			
		||||
export interface MarkdownWord {
 | 
			
		||||
  content: string;
 | 
			
		||||
  type: MarkdownWordType;
 | 
			
		||||
@@ -38,11 +39,12 @@ interface BaseNode {
 | 
			
		||||
  linkTarget?: string;
 | 
			
		||||
  tooltip?: string;
 | 
			
		||||
  padding?: number; //REMOVE?, use from LayoutData.config - Keep, this could be shape specific
 | 
			
		||||
  isGroup: boolean;
 | 
			
		||||
  isGroup?: boolean;
 | 
			
		||||
  width?: number;
 | 
			
		||||
  height?: number;
 | 
			
		||||
  // Specific properties for State Diagram nodes TODO remove and use generic properties
 | 
			
		||||
  intersect?: (point: any) => any;
 | 
			
		||||
  calcIntersect?: (bounds: Bounds, point: Point) => any;
 | 
			
		||||
 | 
			
		||||
  // Non-generic properties
 | 
			
		||||
  rx?: number; // Used for rounded corners in Rect, Ellipse, etc.Maybe it to specialized RectNode, EllipseNode, etc.
 | 
			
		||||
@@ -77,14 +79,31 @@ interface BaseNode {
 | 
			
		||||
/**
 | 
			
		||||
 * Group/cluster nodes, e.g. nodes that contain other nodes.
 | 
			
		||||
 */
 | 
			
		||||
export type NodeChildren = Node[];
 | 
			
		||||
 | 
			
		||||
export interface ClusterNode extends BaseNode {
 | 
			
		||||
  shape?: ClusterShapeID;
 | 
			
		||||
  isGroup: true;
 | 
			
		||||
  children?: NodeChildren;
 | 
			
		||||
  nodeId?: string;
 | 
			
		||||
  level?: number;
 | 
			
		||||
  descr?: string;
 | 
			
		||||
  type?: number;
 | 
			
		||||
  height?: number;
 | 
			
		||||
  width?: number;
 | 
			
		||||
  padding?: number;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export interface NonClusterNode extends BaseNode {
 | 
			
		||||
  shape?: ShapeID;
 | 
			
		||||
  isGroup: false;
 | 
			
		||||
  children?: NodeChildren;
 | 
			
		||||
  nodeId?: string;
 | 
			
		||||
  level?: number;
 | 
			
		||||
  descr?: string;
 | 
			
		||||
  type?: number;
 | 
			
		||||
  height?: number;
 | 
			
		||||
  width?: number;
 | 
			
		||||
  padding?: number;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Common properties for any node in the system
 | 
			
		||||
@@ -113,7 +132,7 @@ export interface Edge {
 | 
			
		||||
  start?: string;
 | 
			
		||||
  stroke?: string;
 | 
			
		||||
  text?: string;
 | 
			
		||||
  type: string;
 | 
			
		||||
  type?: string;
 | 
			
		||||
  // Class Diagram specific properties
 | 
			
		||||
  startLabelRight?: string;
 | 
			
		||||
  endLabelLeft?: string;
 | 
			
		||||
@@ -126,6 +145,12 @@ export interface Edge {
 | 
			
		||||
  thickness?: 'normal' | 'thick' | 'invisible' | 'dotted';
 | 
			
		||||
  look?: string;
 | 
			
		||||
  isUserDefinedId?: boolean;
 | 
			
		||||
  points?: Point[];
 | 
			
		||||
  parentId?: string;
 | 
			
		||||
  dir?: string;
 | 
			
		||||
  source?: string;
 | 
			
		||||
  target?: string;
 | 
			
		||||
  depth?: number;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export interface RectOptions {
 | 
			
		||||
 
 | 
			
		||||
@@ -977,6 +977,7 @@ $defs: # JSON Schema definition (maybe we should move these to a separate file)
 | 
			
		||||
      - useMaxWidth
 | 
			
		||||
      - padding
 | 
			
		||||
      - maxNodeWidth
 | 
			
		||||
      - layoutAlgorithm
 | 
			
		||||
    properties:
 | 
			
		||||
      padding:
 | 
			
		||||
        type: number
 | 
			
		||||
@@ -984,6 +985,10 @@ $defs: # JSON Schema definition (maybe we should move these to a separate file)
 | 
			
		||||
      maxNodeWidth:
 | 
			
		||||
        type: number
 | 
			
		||||
        default: 200
 | 
			
		||||
      layoutAlgorithm:
 | 
			
		||||
        description: Layout algorithm to use for positioning mindmap nodes
 | 
			
		||||
        type: string
 | 
			
		||||
        default: 'cose-bilkent'
 | 
			
		||||
 | 
			
		||||
  KanbanDiagramConfig:
 | 
			
		||||
    title: Kanban Diagram Config
 | 
			
		||||
 
 | 
			
		||||
@@ -36,6 +36,10 @@ export interface Point {
 | 
			
		||||
  x: number;
 | 
			
		||||
  y: number;
 | 
			
		||||
}
 | 
			
		||||
export interface Bounds extends Point {
 | 
			
		||||
  width: number;
 | 
			
		||||
  height: number;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export interface TextDimensionConfig {
 | 
			
		||||
  fontSize?: number;
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										4
									
								
								packages/mermaid/src/types/cytoscape-cose-bilkent.d.ts
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								packages/mermaid/src/types/cytoscape-cose-bilkent.d.ts
									
									
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,4 @@
 | 
			
		||||
declare module 'cytoscape-cose-bilkent' {
 | 
			
		||||
  const coseBilkent: any;
 | 
			
		||||
  export default coseBilkent;
 | 
			
		||||
}
 | 
			
		||||
@@ -3,13 +3,23 @@ import type { EdgeData, Point } from '../types.js';
 | 
			
		||||
// We need to draw the lines a bit shorter to avoid drawing
 | 
			
		||||
// under any transparent markers.
 | 
			
		||||
// The offsets are calculated from the markers' dimensions.
 | 
			
		||||
const markerOffsets = {
 | 
			
		||||
  aggregation: 18,
 | 
			
		||||
  extension: 18,
 | 
			
		||||
  composition: 18,
 | 
			
		||||
export const markerOffsets = {
 | 
			
		||||
  aggregation: 17.25,
 | 
			
		||||
  extension: 17.25,
 | 
			
		||||
  composition: 17.25,
 | 
			
		||||
  dependency: 6,
 | 
			
		||||
  lollipop: 13.5,
 | 
			
		||||
  arrow_point: 4,
 | 
			
		||||
  //arrow_cross: 24,
 | 
			
		||||
} as const;
 | 
			
		||||
 | 
			
		||||
// We need to draw the lines a bit shorter to avoid drawing
 | 
			
		||||
// under any transparent markers.
 | 
			
		||||
// The offsets are calculated from the markers' dimensions.
 | 
			
		||||
export const markerOffsets2 = {
 | 
			
		||||
  arrow_point: 9,
 | 
			
		||||
  arrow_cross: 12.5,
 | 
			
		||||
  arrow_circle: 12.5,
 | 
			
		||||
} as const;
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 
 | 
			
		||||
@@ -44,20 +44,16 @@ ValueConverter -->> Package: Return AST
 | 
			
		||||
```
 | 
			
		||||
 | 
			
		||||
- When to override `TokenBuilder`?
 | 
			
		||||
 | 
			
		||||
  - To override keyword rules.
 | 
			
		||||
  - To override terminal rules that need a custom function.
 | 
			
		||||
  - To manually reorder the list of rules.
 | 
			
		||||
 | 
			
		||||
- When to override `Lexer`?
 | 
			
		||||
 | 
			
		||||
  - To modify input before tokenizing.
 | 
			
		||||
  - To insert/modify tokens that cannot or have not been parsed.
 | 
			
		||||
 | 
			
		||||
- When to override `LangiumParser`?
 | 
			
		||||
 | 
			
		||||
  - To insert or modify attributes that can't be parsed.
 | 
			
		||||
 | 
			
		||||
- When to override `ValueConverter`?
 | 
			
		||||
 | 
			
		||||
  - To modify the returned value from the parser.
 | 
			
		||||
 
 | 
			
		||||
@@ -87,7 +87,6 @@
 | 
			
		||||
### Minor Changes
 | 
			
		||||
 | 
			
		||||
- [#6408](https://github.com/mermaid-js/mermaid/pull/6408) [`ad65313`](https://github.com/mermaid-js/mermaid/commit/ad653138e16765d095613a6e5de86dc5e52ac8f0) Thanks [@ashishjain0512](https://github.com/ashishjain0512)! - fix: restore curve type configuration functionality for flowcharts. This fixes the issue where curve type settings were not being applied when configured through any of the following methods:
 | 
			
		||||
 | 
			
		||||
  - Config
 | 
			
		||||
  - Init directive (%%{ init: { 'flowchart': { 'curve': '...' } } }%%)
 | 
			
		||||
  - LinkStyle command (linkStyle default interpolate ...)
 | 
			
		||||
@@ -106,14 +105,12 @@
 | 
			
		||||
### Minor Changes
 | 
			
		||||
 | 
			
		||||
- [#6187](https://github.com/mermaid-js/mermaid/pull/6187) [`7809b5a`](https://github.com/mermaid-js/mermaid/commit/7809b5a93fae127f45727071f5ff14325222c518) Thanks [@ashishjain0512](https://github.com/ashishjain0512)! - Flowchart new syntax for node metadata bugs
 | 
			
		||||
 | 
			
		||||
  - Incorrect label mapping for nodes when using `&`
 | 
			
		||||
  - Syntax error when `}` with trailing spaces before new line
 | 
			
		||||
 | 
			
		||||
- [#6136](https://github.com/mermaid-js/mermaid/pull/6136) [`ec0d9c3`](https://github.com/mermaid-js/mermaid/commit/ec0d9c389aa6018043187654044c1e0b5aa4f600) Thanks [@knsv](https://github.com/knsv)! - Adding support for animation of flowchart edges
 | 
			
		||||
 | 
			
		||||
- [#6373](https://github.com/mermaid-js/mermaid/pull/6373) [`05bdf0e`](https://github.com/mermaid-js/mermaid/commit/05bdf0e20e2629fe77513218fbd4e28e65f75882) Thanks [@ashishjain0512](https://github.com/ashishjain0512)! - Upgrade Requirement and ER diagram to use the common renderer flow
 | 
			
		||||
 | 
			
		||||
  - Added support for directions
 | 
			
		||||
  - Added support for hand drawn look
 | 
			
		||||
 | 
			
		||||
@@ -162,7 +159,6 @@
 | 
			
		||||
- [#5999](https://github.com/mermaid-js/mermaid/pull/5999) [`742ad7c`](https://github.com/mermaid-js/mermaid/commit/742ad7c130964df1fb5544e909d9556081285f68) Thanks [@knsv](https://github.com/knsv)! - Adding Kanban board, a new diagram type
 | 
			
		||||
 | 
			
		||||
- [#5880](https://github.com/mermaid-js/mermaid/pull/5880) [`bdf145f`](https://github.com/mermaid-js/mermaid/commit/bdf145ffe362462176d9c1e68d5f3ff5c9d962b0) Thanks [@yari-dewalt](https://github.com/yari-dewalt)! - Class diagram changes:
 | 
			
		||||
 | 
			
		||||
  - Updates the class diagram to the new unified way of rendering.
 | 
			
		||||
  - Includes a new "classBox" shape to be used in diagrams
 | 
			
		||||
  - Other updates such as:
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										8684
									
								
								pnpm-lock.yaml
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										8684
									
								
								pnpm-lock.yaml
									
									
									
										generated
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
		Reference in New Issue
	
	Block a user