mirror of
				https://github.com/mermaid-js/mermaid.git
				synced 2025-10-26 16:34:08 +01:00 
			
		
		
		
	
		
			
				
	
	
		
			1055 lines
		
	
	
		
			26 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			1055 lines
		
	
	
		
			26 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| import { imgSnapshotTest, renderGraph } from '../../helpers/util.ts';
 | |
| 
 | |
| describe('Flowchart v2', () => {
 | |
|   it('1: should render a simple flowchart', () => {
 | |
|     imgSnapshotTest(
 | |
|       `flowchart TD
 | |
|       A[Christmas] -->|Get money| B(Go shopping)
 | |
|       B --> C{Let me think}
 | |
|       C -->|One| D[Laptop]
 | |
|       C -->|Two| E[iPhone]
 | |
|       C -->|Three| F[fa:fa-car Car]
 | |
|       `,
 | |
|       {}
 | |
|     );
 | |
|   });
 | |
| 
 | |
|   it('2: should render a simple flowchart with diagramPadding set to 0', () => {
 | |
|     imgSnapshotTest(
 | |
|       `flowchart TD
 | |
|       A[Christmas] -->|Get money| B(Go shopping)
 | |
|       B --> C{Let me think}
 | |
|       %% this is a comment
 | |
|       C -->|One| D[Laptop]
 | |
|       C -->|Two| E[iPhone]
 | |
|       C -->|Three| F[fa:fa-car Car]
 | |
|       `,
 | |
|       { flowchart: { diagramPadding: 0 } }
 | |
|     );
 | |
|   });
 | |
| 
 | |
|   it('3: a link with correct arrowhead to a subgraph', () => {
 | |
|     imgSnapshotTest(
 | |
|       `flowchart TD
 | |
|         P1
 | |
|         P1 -->P1.5
 | |
|         subgraph P1.5
 | |
|           P2
 | |
|           P2.5(( A ))
 | |
|           P3
 | |
|         end
 | |
|         P2 --> P4
 | |
|         P3 --> P6
 | |
|         P1.5 --> P5
 | |
|       `,
 | |
|       {}
 | |
|     );
 | |
|   });
 | |
| 
 | |
|   it('4: Length of edges', () => {
 | |
|     imgSnapshotTest(
 | |
|       `flowchart TD
 | |
|       L1 --- L2
 | |
|       L2 --- C
 | |
|       M1 ---> C
 | |
|       R1 .-> R2
 | |
|       R2 <.-> C
 | |
|       C -->|Label 1| E1
 | |
|       C <-- Label 2 ---> E2
 | |
|       C ----> E3
 | |
|       C <-...-> E4
 | |
|       C ======> E5
 | |
|       `,
 | |
|       {}
 | |
|     );
 | |
|   });
 | |
|   it('5: should render escaped without html labels', () => {
 | |
|     imgSnapshotTest(
 | |
|       `flowchart TD
 | |
|         a["<strong>Haiya</strong>"]---->b
 | |
|       `,
 | |
|       { htmlLabels: false, flowchart: { htmlLabels: false } }
 | |
|     );
 | |
|   });
 | |
|   it('6: should render non-escaped with html labels', () => {
 | |
|     imgSnapshotTest(
 | |
|       `flowchart TD
 | |
|         a["<strong>Haiya</strong>"]===>b
 | |
|       `,
 | |
|       { htmlLabels: true, flowchart: { htmlLabels: true }, securityLevel: 'loose' }
 | |
|     );
 | |
|   });
 | |
|   it('7: should render a flowchart when useMaxWidth is true (default)', () => {
 | |
|     renderGraph(
 | |
|       `flowchart TD
 | |
|       A[Christmas] -->|Get money| B(Go shopping)
 | |
|       B --> C{Let me think}
 | |
|       C -->|One| D[Laptop]
 | |
|       C -->|Two| E[iPhone]
 | |
|       C -->|Three| F[fa:fa-car Car]
 | |
|       `,
 | |
|       { flowchart: { useMaxWidth: true } }
 | |
|     );
 | |
|     cy.get('svg').should((svg) => {
 | |
|       expect(svg).to.have.attr('width', '100%');
 | |
|       // expect(svg).to.have.attr('height');
 | |
|       // use within because the absolute value can be slightly different depending on the environment ±5%
 | |
|       // const height = parseFloat(svg.attr('height'));
 | |
|       // expect(height).to.be.within(446 * 0.95, 446 * 1.05);
 | |
|       const style = svg.attr('style');
 | |
|       expect(style).to.match(/^max-width: [\d.]+px;$/);
 | |
|       const maxWidthValue = parseFloat(style.match(/[\d.]+/g).join(''));
 | |
|       expect(maxWidthValue).to.be.within(290 * 0.95 - 1, 290 * 1.05);
 | |
|     });
 | |
|   });
 | |
|   it('8: should render a flowchart when useMaxWidth is false', () => {
 | |
|     renderGraph(
 | |
|       `flowchart TD
 | |
|       A[Christmas] -->|Get money| B(Go shopping)
 | |
|       B --> C{Let me think}
 | |
|       C -->|One| D[Laptop]
 | |
|       C -->|Two| E[iPhone]
 | |
|       C -->|Three| F[fa:fa-car Car]
 | |
|       `,
 | |
|       { flowchart: { useMaxWidth: false } }
 | |
|     );
 | |
|     cy.get('svg').should((svg) => {
 | |
|       // const height = parseFloat(svg.attr('height'));
 | |
|       const width = parseFloat(svg.attr('width'));
 | |
|       // use within because the absolute value can be slightly different depending on the environment ±5%
 | |
|       // expect(height).to.be.within(446 * 0.95, 446 * 1.05);
 | |
|       expect(width).to.be.within(290 * 0.95 - 1, 290 * 1.05);
 | |
|       expect(svg).to.not.have.attr('style');
 | |
|     });
 | |
|   });
 | |
| 
 | |
|   it('V2 - 16: Render Stadium shape', () => {
 | |
|     imgSnapshotTest(
 | |
|       ` flowchart TD
 | |
|       A([stadium shape test])
 | |
|       A -->|Get money| B([Go shopping])
 | |
|       B --> C([Let me think...<br />Do I want something for work,<br />something to spend every free second with,<br />or something to get around?])
 | |
|       C -->|One| D([Laptop])
 | |
|       C -->|Two| E([iPhone])
 | |
|       C -->|Three| F([Car<br/>wroom wroom])
 | |
|       click A "index.html#link-clicked" "link test"
 | |
|       click B testClick "click test"
 | |
|       classDef someclass fill:#f96;
 | |
|       class A someclass;
 | |
|       class C someclass;
 | |
|       `,
 | |
|       { flowchart: { htmlLabels: false }, fontFamily: 'courier' }
 | |
|     );
 | |
|   });
 | |
| 
 | |
|   it('50: handle nested subgraphs in reverse order', () => {
 | |
|     imgSnapshotTest(
 | |
|       `flowchart LR
 | |
|         a -->b
 | |
|         subgraph A
 | |
|         B
 | |
|         end
 | |
|         subgraph B
 | |
|         b
 | |
|         end
 | |
|       `,
 | |
|       { htmlLabels: true, flowchart: { htmlLabels: true }, securityLevel: 'loose' }
 | |
|     );
 | |
|   });
 | |
| 
 | |
|   it('51: handle nested subgraphs in reverse order', () => {
 | |
|     imgSnapshotTest(
 | |
|       `flowchart LR
 | |
|         a -->b
 | |
|         subgraph A
 | |
|         B
 | |
|         end
 | |
|         subgraph B
 | |
|         b
 | |
|         end
 | |
|       `,
 | |
|       { htmlLabels: true, flowchart: { htmlLabels: true }, securityLevel: 'loose' }
 | |
|     );
 | |
|   });
 | |
| 
 | |
|   it('52: handle nested subgraphs in several levels.', () => {
 | |
|     imgSnapshotTest(
 | |
|       `flowchart TB
 | |
|     b-->B
 | |
|     a-->c
 | |
|     subgraph O
 | |
|       A
 | |
|     end
 | |
|     subgraph B
 | |
|       c
 | |
|     end
 | |
|     subgraph A
 | |
|         a
 | |
|         b
 | |
|         B
 | |
|     end
 | |
|       `,
 | |
|       { htmlLabels: true, flowchart: { htmlLabels: true }, securityLevel: 'loose' }
 | |
|     );
 | |
|   });
 | |
| 
 | |
|   it('53: handle nested subgraphs with edges in and out', () => {
 | |
|     imgSnapshotTest(
 | |
|       `flowchart TB
 | |
|   internet
 | |
|   nat
 | |
|   routeur
 | |
|   lb1
 | |
|   lb2
 | |
|   compute1
 | |
|   compute2
 | |
|   subgraph project
 | |
|   routeur
 | |
|   nat
 | |
|     subgraph subnet1
 | |
|       compute1
 | |
|       lb1
 | |
|     end
 | |
|     subgraph subnet2
 | |
|       compute2
 | |
|       lb2
 | |
|     end
 | |
|   end
 | |
|   internet --> routeur
 | |
|   routeur --> subnet1 & subnet2
 | |
|   subnet1 & subnet2 --> nat --> internet
 | |
|       `,
 | |
|       { htmlLabels: true, flowchart: { htmlLabels: true }, securityLevel: 'loose' }
 | |
|     );
 | |
|   });
 | |
| 
 | |
|   it('54: handle nested subgraphs with outgoing links', () => {
 | |
|     imgSnapshotTest(
 | |
|       `flowchart TD
 | |
|   subgraph  main
 | |
|     subgraph subcontainer
 | |
|       subcontainer-child
 | |
|     end
 | |
|      subcontainer-child--> subcontainer-sibling
 | |
|   end
 | |
|       `,
 | |
|       { htmlLabels: true, flowchart: { htmlLabels: true }, securityLevel: 'loose' }
 | |
|     );
 | |
|   });
 | |
| 
 | |
|   it('55: handle nested subgraphs with outgoing links 2', () => {
 | |
|     imgSnapshotTest(
 | |
|       `flowchart TD
 | |
| 
 | |
| subgraph one[One]
 | |
|     subgraph sub_one[Sub One]
 | |
|         _sub_one
 | |
|     end
 | |
|     subgraph sub_two[Sub Two]
 | |
|         _sub_two
 | |
|     end
 | |
|     _one
 | |
| end
 | |
| 
 | |
| %% here, either the first or the second one
 | |
| sub_one --> sub_two
 | |
| _one --> b
 | |
|       `,
 | |
|       { htmlLabels: true, flowchart: { htmlLabels: true }, securityLevel: 'loose' }
 | |
|     );
 | |
|   });
 | |
| 
 | |
|   it('56: handle nested subgraphs with outgoing links 3', () => {
 | |
|     imgSnapshotTest(
 | |
|       `flowchart TB
 | |
|   subgraph container_Beta
 | |
|     process_C-->Process_D
 | |
|   end
 | |
|   subgraph container_Alpha
 | |
|     process_A-->process_B
 | |
|     process_A-->|messages|process_C
 | |
|     end
 | |
|     process_B-->|via_AWSBatch|container_Beta
 | |
|       `,
 | |
|       { htmlLabels: true, flowchart: { htmlLabels: true }, securityLevel: 'loose' }
 | |
|     );
 | |
|   });
 | |
|   it('57: handle nested subgraphs with outgoing links 4', () => {
 | |
|     imgSnapshotTest(
 | |
|       `flowchart LR
 | |
| subgraph A
 | |
| a -->b
 | |
| end
 | |
| subgraph B
 | |
| b
 | |
| end
 | |
|       `,
 | |
|       { htmlLabels: true, flowchart: { htmlLabels: true }, securityLevel: 'loose' }
 | |
|     );
 | |
|   });
 | |
| 
 | |
|   it('57: handle nested subgraphs with outgoing links 2', () => {
 | |
|     imgSnapshotTest(
 | |
|       `flowchart TB
 | |
|     c1-->a2
 | |
|     subgraph one
 | |
|     a1-->a2
 | |
|     end
 | |
|     subgraph two
 | |
|     b1-->b2
 | |
|     end
 | |
|     subgraph three
 | |
|     c1-->c2
 | |
|     end
 | |
|     one --> two
 | |
|     three --> two
 | |
|     two --> c2
 | |
|       `,
 | |
|       { htmlLabels: true, flowchart: { htmlLabels: true }, securityLevel: 'loose' }
 | |
|     );
 | |
|   });
 | |
|   it('57.x: handle nested subgraphs with outgoing links 5', () => {
 | |
|     imgSnapshotTest(
 | |
|       `%% this does not produce the desired result
 | |
| flowchart TB
 | |
|   subgraph container_Beta
 | |
|     process_C-->Process_D
 | |
|   end
 | |
|   subgraph container_Alpha
 | |
|     process_A-->process_B
 | |
|     process_B-->|via_AWSBatch|container_Beta
 | |
|     process_A-->|messages|process_C
 | |
|   end
 | |
|       `,
 | |
|       { htmlLabels: true, flowchart: { htmlLabels: true }, securityLevel: 'loose' }
 | |
|     );
 | |
|   });
 | |
|   it('58: handle styling with style expressions', () => {
 | |
|     imgSnapshotTest(
 | |
|       `
 | |
|     flowchart LR
 | |
|     id1(Start)-->id2(Stop)
 | |
|     style id1 fill:#f9f,stroke:#333,stroke-width:4px
 | |
|     style id2 fill:#bbf,stroke:#f66,stroke-width:2px,color:#fff,stroke-dasharray: 5 5
 | |
|       `,
 | |
|       { htmlLabels: true, flowchart: { htmlLabels: true }, securityLevel: 'loose' }
 | |
|     );
 | |
|   });
 | |
|   it('59: handle styling of subgraphs and links', () => {
 | |
|     imgSnapshotTest(
 | |
|       `
 | |
| flowchart TD
 | |
|   A[Christmas] ==> D
 | |
|   A[Christmas] -->|Get money| B(Go shopping)
 | |
|   A[Christmas] ==> C
 | |
|   subgraph T ["Test"]
 | |
|     A
 | |
|     B
 | |
|     C
 | |
|   end
 | |
| 
 | |
|   classDef Test fill:#F84E68,stroke:#333,color:white;
 | |
|   class A,T Test
 | |
|   classDef TestSub fill:green;
 | |
|   class T TestSub
 | |
|   linkStyle 0,1 color:orange, stroke: orange;
 | |
|       `,
 | |
|       { htmlLabels: true, flowchart: { htmlLabels: true }, securityLevel: 'loose' }
 | |
|     );
 | |
|   });
 | |
|   it('60: handle styling for all node shapes - v2', () => {
 | |
|     imgSnapshotTest(
 | |
|       `
 | |
|       flowchart LR
 | |
|       A[red text] -->|default style| B(blue text)
 | |
|       C([red text]) -->|default style| D[[blue text]]
 | |
|       E[(red text)] -->|default style| F((blue text))
 | |
|       G>red text] -->|default style| H{blue text}
 | |
|       I{{red text}} -->|default style| J[/blue text/]
 | |
|       K[\\ red text\\] -->|default style| L[/blue text\\]
 | |
|       M[\\ red text/] -->|default style| N[blue text];
 | |
|       O(((red text))) -->|default style| P(((blue text)));
 | |
|       linkStyle default color:Sienna;
 | |
|       style A stroke:#ff0000,fill:#ffcccc,color:#ff0000;
 | |
|       style B stroke:#0000ff,fill:#ccccff,color:#0000ff;
 | |
|       style C stroke:#ff0000,fill:#ffcccc,color:#ff0000;
 | |
|       style D stroke:#0000ff,fill:#ccccff,color:#0000ff;
 | |
|       style E stroke:#ff0000,fill:#ffcccc,color:#ff0000;
 | |
|       style F stroke:#0000ff,fill:#ccccff,color:#0000ff;
 | |
|       style G stroke:#ff0000,fill:#ffcccc,color:#ff0000;
 | |
|       style H stroke:#0000ff,fill:#ccccff,color:#0000ff;
 | |
|       style I stroke:#ff0000,fill:#ffcccc,color:#ff0000;
 | |
|       style J stroke:#0000ff,fill:#ccccff,color:#0000ff;
 | |
|       style K stroke:#ff0000,fill:#ffcccc,color:#ff0000;
 | |
|       style L stroke:#0000ff,fill:#ccccff,color:#0000ff;
 | |
|       style M stroke:#ff0000,fill:#ffcccc,color:#ff0000;
 | |
|       style N stroke:#0000ff,fill:#ccccff,color:#0000ff;
 | |
|       style O stroke:#ff0000,fill:#ffcccc,color:#ff0000;
 | |
|       style P stroke:#0000ff,fill:#ccccff,color:#0000ff;
 | |
|       `,
 | |
|       { htmlLabels: true, flowchart: { htmlLabels: true }, securityLevel: 'loose', logLevel: 2 }
 | |
|     );
 | |
|   });
 | |
|   it('61: fontawesome icons in edge labels', () => {
 | |
|     imgSnapshotTest(
 | |
|       `
 | |
|       flowchart TD
 | |
|         C -->|fa:fa-car Car| F[fa:fa-car Car]
 | |
|       `,
 | |
|       { htmlLabels: true, flowchart: { htmlLabels: true }, securityLevel: 'loose' }
 | |
|     );
 | |
|   });
 | |
|   it('62: should render styled subgraphs', () => {
 | |
|     imgSnapshotTest(
 | |
|       `
 | |
|       flowchart TB
 | |
|       A
 | |
|       B
 | |
|       subgraph foo[Foo SubGraph]
 | |
|         C
 | |
|         D
 | |
|       end
 | |
|       subgraph bar[Bar SubGraph]
 | |
|         E
 | |
|         F
 | |
|       end
 | |
|       G
 | |
| 
 | |
|       A-->B
 | |
|       B-->C
 | |
|       C-->D
 | |
|       B-->D
 | |
|       D-->E
 | |
|       E-->A
 | |
|       E-->F
 | |
|       F-->D
 | |
|       F-->G
 | |
|       B-->G
 | |
|       G-->D
 | |
| 
 | |
|       style foo fill:#F99,stroke-width:2px,stroke:#F0F,color:darkred
 | |
|       style bar fill:#999,stroke-width:10px,stroke:#0F0,color:blue
 | |
|       `,
 | |
|       { htmlLabels: true, flowchart: { htmlLabels: true }, securityLevel: 'loose' }
 | |
|     );
 | |
|   });
 | |
|   it('63: title on subgraphs should be themable', () => {
 | |
|     imgSnapshotTest(
 | |
|       `
 | |
|       %%{init:{"theme":"base", "themeVariables": {"primaryColor":"#411d4e", "titleColor":"white", "darkMode":true}}}%%
 | |
|       flowchart LR
 | |
|       subgraph A
 | |
|           a --> b
 | |
|       end
 | |
|       subgraph B
 | |
|           i -->f
 | |
|       end
 | |
|       A --> B
 | |
|       `,
 | |
|       { htmlLabels: true, flowchart: { htmlLabels: true }, securityLevel: 'loose' }
 | |
|     );
 | |
|   });
 | |
|   it('65-1: text-color from classes', () => {
 | |
|     imgSnapshotTest(
 | |
|       `
 | |
|       flowchart LR
 | |
|         classDef dark fill:#000,stroke:#000,stroke-width:4px,color:#fff
 | |
|         Lorem --> Ipsum --> Dolor
 | |
|         class Lorem,Dolor dark
 | |
|       `,
 | |
|       { htmlLabels: true, flowchart: { htmlLabels: true }, securityLevel: 'loose' }
 | |
|     );
 | |
|   });
 | |
|   it('65-2: bold text from classes', () => {
 | |
|     imgSnapshotTest(
 | |
|       `
 | |
|       flowchart
 | |
|         classDef cat fill:#f9d5e5, stroke:#233d4d,stroke-width:2px, font-weight:bold;
 | |
|         CS(A long bold text to be viewed):::cat
 | |
|       `,
 | |
|       { htmlLabels: true, flowchart: { htmlLabels: true }, securityLevel: 'loose' }
 | |
|     );
 | |
|   });
 | |
|   it('65-3: bigger font from classes', () => {
 | |
|     imgSnapshotTest(
 | |
|       `
 | |
| flowchart
 | |
|   Node1:::class1 --> Node2:::class2
 | |
|   Node1:::class1 --> Node3:::class2
 | |
|   Node3 --> Node4((I am a circle)):::larger
 | |
| 
 | |
|   classDef class1 fill:lightblue
 | |
|   classDef class2 fill:pink
 | |
|   classDef larger font-size:30px,fill:yellow
 | |
|       `,
 | |
|       { htmlLabels: true, flowchart: { htmlLabels: true }, securityLevel: 'loose' }
 | |
|     );
 | |
|   });
 | |
|   it('66: More nested subgraph cases (TB)', () => {
 | |
|     imgSnapshotTest(
 | |
|       `
 | |
| flowchart TB
 | |
|     subgraph two
 | |
|     b1
 | |
|     end
 | |
|     subgraph three
 | |
|     c2
 | |
|     end
 | |
| 
 | |
|     three --> two
 | |
|     two --> c2
 | |
| 
 | |
|       `,
 | |
|       { htmlLabels: true, flowchart: { htmlLabels: true }, securityLevel: 'loose' }
 | |
|     );
 | |
|   });
 | |
|   it('67: More nested subgraph cases (RL)', () => {
 | |
|     imgSnapshotTest(
 | |
|       `
 | |
| flowchart RL
 | |
|     subgraph two
 | |
|     b1
 | |
|     end
 | |
|     subgraph three
 | |
|     c2
 | |
|     end
 | |
| 
 | |
|     three --> two
 | |
|     two --> c2
 | |
| 
 | |
|       `,
 | |
|       { htmlLabels: true, flowchart: { htmlLabels: true }, securityLevel: 'loose' }
 | |
|     );
 | |
|   });
 | |
|   it('68: More nested subgraph cases (BT)', () => {
 | |
|     imgSnapshotTest(
 | |
|       `
 | |
| flowchart BT
 | |
|     subgraph two
 | |
|     b1
 | |
|     end
 | |
|     subgraph three
 | |
|     c2
 | |
|     end
 | |
| 
 | |
|     three --> two
 | |
|     two --> c2
 | |
| 
 | |
|       `,
 | |
|       { htmlLabels: true, flowchart: { htmlLabels: true }, securityLevel: 'loose' }
 | |
|     );
 | |
|   });
 | |
|   it('69: More nested subgraph cases (LR)', () => {
 | |
|     imgSnapshotTest(
 | |
|       `
 | |
| flowchart LR
 | |
|     subgraph two
 | |
|     b1
 | |
|     end
 | |
|     subgraph three
 | |
|     c2
 | |
|     end
 | |
| 
 | |
|     three --> two
 | |
|     two --> c2
 | |
| 
 | |
|       `,
 | |
|       { htmlLabels: true, flowchart: { htmlLabels: true }, securityLevel: 'loose' }
 | |
|     );
 | |
|   });
 | |
|   it('70: Handle nested subgraph cases (TB) link out and link between subgraphs', () => {
 | |
|     imgSnapshotTest(
 | |
|       `
 | |
| flowchart TB
 | |
|    subgraph S1
 | |
|     sub1 -->sub2
 | |
|    end
 | |
|   subgraph S2
 | |
|     sub4
 | |
|    end
 | |
|    S1 --> S2
 | |
|    sub1 --> sub4
 | |
|       `,
 | |
|       { htmlLabels: true, flowchart: { htmlLabels: true }, securityLevel: 'loose' }
 | |
|     );
 | |
|   });
 | |
|   it('71: Handle nested subgraph cases (RL) link out and link between subgraphs', () => {
 | |
|     imgSnapshotTest(
 | |
|       `
 | |
| flowchart RL
 | |
|    subgraph S1
 | |
|     sub1 -->sub2
 | |
|    end
 | |
|   subgraph S2
 | |
|     sub4
 | |
|    end
 | |
|    S1 --> S2
 | |
|    sub1 --> sub4
 | |
|       `,
 | |
|       { htmlLabels: true, flowchart: { htmlLabels: true }, securityLevel: 'loose' }
 | |
|     );
 | |
|   });
 | |
|   it('72: Handle nested subgraph cases (BT) link out and link between subgraphs', () => {
 | |
|     imgSnapshotTest(
 | |
|       `
 | |
| flowchart BT
 | |
|    subgraph S1
 | |
|     sub1 -->sub2
 | |
|    end
 | |
|   subgraph S2
 | |
|     sub4
 | |
|    end
 | |
|    S1 --> S2
 | |
|    sub1 --> sub4
 | |
|       `,
 | |
|       { htmlLabels: true, flowchart: { htmlLabels: true }, securityLevel: 'loose' }
 | |
|     );
 | |
|   });
 | |
|   it('74: Handle nested subgraph cases (RL) link out and link between subgraphs', () => {
 | |
|     imgSnapshotTest(
 | |
|       `
 | |
| flowchart RL
 | |
|    subgraph S1
 | |
|     sub1 -->sub2
 | |
|    end
 | |
|   subgraph S2
 | |
|     sub4
 | |
|    end
 | |
|    S1 --> S2
 | |
|    sub1 --> sub4
 | |
|       `,
 | |
|       { htmlLabels: true, flowchart: { htmlLabels: true }, securityLevel: 'loose' }
 | |
|     );
 | |
|   });
 | |
|   it('74: Handle labels for multiple edges from and to the same couple of nodes', () => {
 | |
|     imgSnapshotTest(
 | |
|       `
 | |
| flowchart RL
 | |
|     subgraph one
 | |
|       a1 -- l1 --> a2
 | |
|       a1 -- l2 --> a2
 | |
|     end
 | |
|       `,
 | |
|       { htmlLabels: true, flowchart: { htmlLabels: true }, securityLevel: 'loose' }
 | |
|     );
 | |
|   });
 | |
| 
 | |
|   it('76: handle unicode encoded character with HTML labels true', () => {
 | |
|     imgSnapshotTest(
 | |
|       `flowchart TB
 | |
|       a{{"Lorem 'ipsum' dolor 'sit' amet, 'consectetur' adipiscing 'elit'."}}
 | |
|       --> b{{"Lorem #quot;ipsum#quot; dolor #quot;sit#quot; amet,#quot;consectetur#quot; adipiscing #quot;elit#quot;."}}
 | |
|       `,
 | |
|       { htmlLabels: true, flowchart: { htmlLabels: true }, securityLevel: 'loose' }
 | |
|     );
 | |
|   });
 | |
| 
 | |
|   it('2050: handling of different rendering direction in subgraphs', () => {
 | |
|     imgSnapshotTest(
 | |
|       `
 | |
|     flowchart LR
 | |
| 
 | |
|       subgraph TOP
 | |
|         direction TB
 | |
|         subgraph B1
 | |
|             direction RL
 | |
|             i1 -->f1
 | |
|         end
 | |
|         subgraph B2
 | |
|             direction BT
 | |
|             i2 -->f2
 | |
|         end
 | |
|       end
 | |
|       A --> TOP --> B
 | |
|       B1 --> B2
 | |
|       `,
 | |
|       { htmlLabels: true, flowchart: { htmlLabels: true }, securityLevel: 'loose' }
 | |
|     );
 | |
|   });
 | |
| 
 | |
|   it('2388: handling default in the node name', () => {
 | |
|     imgSnapshotTest(
 | |
|       `
 | |
|       flowchart LR
 | |
|       default-index.js --> dot.template.js
 | |
|       index.js --> module-utl.js
 | |
|       `,
 | |
|       { htmlLabels: true, flowchart: { htmlLabels: true }, securityLevel: 'loose' }
 | |
|     );
 | |
|   });
 | |
|   it('2824: Clipping of edges', () => {
 | |
|     imgSnapshotTest(
 | |
|       `
 | |
|       flowchart TD
 | |
|           A --> B
 | |
|           A --> C
 | |
|           B --> C
 | |
|       `,
 | |
|       { htmlLabels: true, flowchart: { htmlLabels: true }, securityLevel: 'loose' }
 | |
|     );
 | |
|   });
 | |
|   it('1433: should render a titled flowchart with titleTopMargin set to 0', () => {
 | |
|     imgSnapshotTest(
 | |
|       `---
 | |
| title: Simple flowchart
 | |
| ---
 | |
| flowchart TD
 | |
| A --> B
 | |
| `,
 | |
|       { flowchart: { titleTopMargin: 10 } }
 | |
|     );
 | |
|   });
 | |
|   it('3192: It should be possieble to render flowcharts with invisible edges', () => {
 | |
|     imgSnapshotTest(
 | |
|       `---
 | |
| title: Simple flowchart with invisible edges
 | |
| ---
 | |
| flowchart TD
 | |
| A ~~~ B
 | |
| `,
 | |
|       { flowchart: { titleTopMargin: 10 } }
 | |
|     );
 | |
|   });
 | |
|   it('4023: Should render html labels with images and-or text correctly', () => {
 | |
|     imgSnapshotTest(
 | |
|       `flowchart TD
 | |
|     B[<img src='https://mermaid.js.org/mermaid-logo.svg'>]
 | |
|     B-->C[<img src="https://mermaid.js.org/mermaid-logo.svg"> more text <img src='https://mermaid.js.org/mermaid-logo.svg'>]
 | |
|     B-->D(<img src='https://mermaid.js.org/mermaid-logo.svg'> some text)
 | |
|     B-->E(plain)`,
 | |
|       {}
 | |
|     );
 | |
|   });
 | |
| 
 | |
|   it('4439: Should render the graph even if some images are missing', () => {
 | |
|     imgSnapshotTest(
 | |
|       `flowchart TD
 | |
|     B[<img>]
 | |
|     B-->C[<img>]`,
 | |
|       {}
 | |
|     );
 | |
|   });
 | |
| 
 | |
|   it('5064: Should render when subgraph child has links to outside node and subgraph', () => {
 | |
|     imgSnapshotTest(
 | |
|       `flowchart TB
 | |
|     Out --> In
 | |
|     subgraph Sub
 | |
|       In
 | |
|     end
 | |
|     Sub --> In`
 | |
|     );
 | |
|   });
 | |
| 
 | |
|   it('5059: Should render when subgraph contains only subgraphs, has link to outside and itself is part of a link', () => {
 | |
|     imgSnapshotTest(
 | |
|       `flowchart
 | |
| 
 | |
|       subgraph Main
 | |
|         subgraph Child1
 | |
|           Node1
 | |
|           Node2
 | |
|         end
 | |
|         subgraph Child2
 | |
|           Node3
 | |
|           Node4
 | |
|         end
 | |
|       end
 | |
|       Main --> Out1
 | |
|       Child2 --> Out2`
 | |
|     );
 | |
|   });
 | |
| 
 | |
|   it('3258: Should render subgraphs with main graph nodeSpacing and rankSpacing', () => {
 | |
|     imgSnapshotTest(
 | |
|       `---
 | |
|       title: Subgraph nodeSpacing and rankSpacing example
 | |
|       ---
 | |
|       flowchart LR
 | |
|         X --> Y
 | |
|         subgraph X
 | |
|           direction LR
 | |
|           A
 | |
|           C
 | |
|         end
 | |
|         subgraph Y
 | |
|           B
 | |
|           D
 | |
|         end
 | |
|       `,
 | |
|       { flowchart: { nodeSpacing: 1, rankSpacing: 1 } }
 | |
|     );
 | |
|   });
 | |
| 
 | |
|   it('3258: Should render subgraphs with large nodeSpacing and rankSpacing', () => {
 | |
|     imgSnapshotTest(
 | |
|       `---
 | |
|       title: Subgraph nodeSpacing and rankSpacing example
 | |
|       config:
 | |
|         flowchart: 
 | |
|           nodeSpacing: 250
 | |
|           rankSpacing: 250
 | |
|       ---
 | |
|       flowchart LR
 | |
|         X --> Y
 | |
|         subgraph X
 | |
|           direction LR
 | |
|           A
 | |
|           C
 | |
|         end
 | |
|         subgraph Y
 | |
|           B
 | |
|           D
 | |
|         end
 | |
|       `
 | |
|     );
 | |
|   });
 | |
| 
 | |
|   describe('Markdown strings flowchart (#4220)', () => {
 | |
|     describe('html labels', () => {
 | |
|       it('With styling and classes', () => {
 | |
|         imgSnapshotTest(
 | |
|           `%%{init: {"flowchart": {"htmlLabels": true}} }%%
 | |
| flowchart LR
 | |
|     A:::someclass --> B["\`The **cat** in the hat\`"]:::someclass
 | |
|     id1(Start)-->id2(Stop)
 | |
|     style id1 fill:#f9f,stroke:#333,stroke-width:4px
 | |
|     style id2 fill:#bbf,stroke:#f66,stroke-width:2px,color:#fff,stroke-dasharray: 5 5
 | |
|     classDef someclass fill:#f96
 | |
| `,
 | |
|           { flowchart: { titleTopMargin: 0 } }
 | |
|         );
 | |
|       });
 | |
|       it('With formatting in a node', () => {
 | |
|         imgSnapshotTest(
 | |
|           `%%{init: {"flowchart": {"htmlLabels": true}} }%%
 | |
| flowchart LR
 | |
|   a{"\`The **cat** in the hat\`"} -- 1o --> b
 | |
|   a -- 2o --> c
 | |
|   a -- 3o --> d
 | |
|   g --2i--> a
 | |
|   d --1i--> a
 | |
|   h --3i -->a
 | |
|   b --> d(The dog in the hog)
 | |
|   c --> d
 | |
| `,
 | |
|           { flowchart: { titleTopMargin: 0 } }
 | |
|         );
 | |
|       });
 | |
|       it('New line in node and formatted edge label', () => {
 | |
|         imgSnapshotTest(
 | |
|           `%%{init: {"flowchart": {"htmlLabels": true}} }%%
 | |
| flowchart LR
 | |
| b("\`The dog in **the** hog.(1)
 | |
| NL\`") --"\`1o **bold**\`"--> c
 | |
| `,
 | |
|           { flowchart: { titleTopMargin: 0 } }
 | |
|         );
 | |
|       });
 | |
|       it('Wrapping long text with a new line', () => {
 | |
|         imgSnapshotTest(
 | |
|           `%%{init: {"flowchart": {"htmlLabels": true}} }%%
 | |
| flowchart LR
 | |
| b("\`The dog in **the** hog.(1).. a a a a *very long text* about it
 | |
| Word!
 | |
| 
 | |
| Another line with many, many words. Another line with many, many words. Another line with many, many words. Another line with many, many words. Another line with many, many words. Another line with many, many words. Another line with many, many words. Another line with many, many words. \`") --> c
 | |
| 
 | |
| `,
 | |
|           { flowchart: { titleTopMargin: 0 } }
 | |
|         );
 | |
|       });
 | |
|       it('Sub graphs and markdown strings', () => {
 | |
|         imgSnapshotTest(
 | |
|           `%%{init: {"flowchart": {"htmlLabels": true}} }%%
 | |
| flowchart LR
 | |
| subgraph "One"
 | |
|   a("\`The **cat**
 | |
|   in the hat\`") -- "1o" --> b{{"\`The **dog** in the hog\`"}}
 | |
| end
 | |
| subgraph "\`**Two**\`"
 | |
|   c("\`The **cat**
 | |
|   in the hat\`") -- "\`1o **ipa**\`" --> d("The dog in the hog")
 | |
| end
 | |
| 
 | |
| `,
 | |
|           { flowchart: { titleTopMargin: 0 } }
 | |
|         );
 | |
|       });
 | |
|     });
 | |
| 
 | |
|     describe('svg text labels', () => {
 | |
|       it('With styling and classes', () => {
 | |
|         imgSnapshotTest(
 | |
|           `%%{init: {"flowchart": {"htmlLabels": false}} }%%
 | |
| flowchart LR
 | |
|     A:::someclass --> B["\`The **cat** in the hat\`"]:::someclass
 | |
|     id1(Start)-->id2(Stop)
 | |
|     style id1 fill:#f9f,stroke:#333,stroke-width:4px
 | |
|     style id2 fill:#bbf,stroke:#f66,stroke-width:2px,color:#fff,stroke-dasharray: 5 5
 | |
|     classDef someclass fill:#f96
 | |
| `,
 | |
|           { flowchart: { titleTopMargin: 0 } }
 | |
|         );
 | |
|       });
 | |
|       it('With formatting in a node', () => {
 | |
|         imgSnapshotTest(
 | |
|           `%%{init: {"flowchart": {"htmlLabels": false}} }%%
 | |
| flowchart LR
 | |
|   a{"\`The **cat** in the hat\`"} -- 1o --> b
 | |
|   a -- 2o --> c
 | |
|   a -- 3o --> d
 | |
|   g --2i--> a
 | |
|   d --1i--> a
 | |
|   h --3i -->a
 | |
|   b --> d(The dog in the hog)
 | |
|   c --> d
 | |
| `,
 | |
|           { flowchart: { titleTopMargin: 0 } }
 | |
|         );
 | |
|       });
 | |
|       it('New line in node and formatted edge label', () => {
 | |
|         imgSnapshotTest(
 | |
|           `%%{init: {"flowchart": {"htmlLabels": false}} }%%
 | |
| flowchart LR
 | |
| b("\`The dog in **the** hog.(1)
 | |
| NL\`") --"\`1o **bold**\`"--> c
 | |
| `,
 | |
|           { flowchart: { titleTopMargin: 0 } }
 | |
|         );
 | |
|       });
 | |
|       it('Wrapping long text with a new line', () => {
 | |
|         imgSnapshotTest(
 | |
|           `%%{init: {"flowchart": {"htmlLabels": false}} }%%
 | |
| flowchart LR
 | |
| b("\`The dog in **the** hog.(1).. a a a a *very long text* about it
 | |
| Word!
 | |
| 
 | |
| Another line with many, many words. Another line with many, many words. Another line with many, many words. Another line with many, many words. Another line with many, many words. Another line with many, many words. Another line with many, many words. Another line with many, many words. \`") --> c
 | |
| 
 | |
| `,
 | |
|           { flowchart: { titleTopMargin: 0 } }
 | |
|         );
 | |
|       });
 | |
|       it('Sub graphs and markdown strings', () => {
 | |
|         imgSnapshotTest(
 | |
|           `%%{init: {"flowchart": {"htmlLabels": false}} }%%
 | |
| flowchart LR
 | |
| subgraph "One"
 | |
|   a("\`The **cat**
 | |
|   in the hat\`") -- "1o" --> b{{"\`The **dog** in the hog\`"}}
 | |
| end
 | |
| subgraph "\`**Two**\`"
 | |
|   c("\`The **cat**
 | |
|   in the hat\`") -- "\`1o **ipa**\`" --> d("The dog in the hog")
 | |
| end
 | |
| 
 | |
| `,
 | |
|           { flowchart: { titleTopMargin: 0 } }
 | |
|         );
 | |
|       });
 | |
|     });
 | |
| 
 | |
|     it('should not auto wrap when markdownAutoWrap is false', () => {
 | |
|       imgSnapshotTest(
 | |
|         `flowchart TD
 | |
|     angular_velocity["\`**angular_velocity**
 | |
|       *angular_displacement / duration*
 | |
|       [rad/s, 1/s]
 | |
|       {vector}\`"]
 | |
|     frequency["frequency\n(1 / period_duration)\n[Hz, 1/s]"]`,
 | |
|         { markdownAutoWrap: false }
 | |
|       );
 | |
|     });
 | |
|   });
 | |
|   describe('Subgraph title margins', () => {
 | |
|     it('Should render subgraphs with title margins set (LR)', () => {
 | |
|       imgSnapshotTest(
 | |
|         `flowchart LR
 | |
| 
 | |
|           subgraph TOP
 | |
|               direction TB
 | |
|               subgraph B1
 | |
|                   direction RL
 | |
|                   i1 -->f1
 | |
|               end
 | |
|               subgraph B2
 | |
|                   direction BT
 | |
|                   i2 -->f2
 | |
|               end
 | |
|           end
 | |
|           A --> TOP --> B
 | |
|           B1 --> B2
 | |
|         `,
 | |
|         { flowchart: { subGraphTitleMargin: { top: 10, bottom: 5 } } }
 | |
|       );
 | |
|     });
 | |
|     it('Should render subgraphs with title margins set (TD)', () => {
 | |
|       imgSnapshotTest(
 | |
|         `flowchart TD
 | |
| 
 | |
|           subgraph TOP
 | |
|               direction LR
 | |
|               subgraph B1
 | |
|                   direction RL
 | |
|                   i1 -->f1
 | |
|               end
 | |
|               subgraph B2
 | |
|                   direction BT
 | |
|                   i2 -->f2
 | |
|               end
 | |
|           end
 | |
|           A --> TOP --> B
 | |
|           B1 --> B2
 | |
|         `,
 | |
|         { flowchart: { subGraphTitleMargin: { top: 8, bottom: 16 } } }
 | |
|       );
 | |
|     });
 | |
|     it('Should render subgraphs with title margins set (LR) and htmlLabels set to false', () => {
 | |
|       imgSnapshotTest(
 | |
|         `flowchart LR
 | |
| 
 | |
|           subgraph TOP
 | |
|               direction TB
 | |
|               subgraph B1
 | |
|                   direction RL
 | |
|                   i1 -->f1
 | |
|               end
 | |
|               subgraph B2
 | |
|                   direction BT
 | |
|                   i2 -->f2
 | |
|               end
 | |
|           end
 | |
|           A --> TOP --> B
 | |
|           B1 --> B2
 | |
|         `,
 | |
|         {
 | |
|           htmlLabels: false,
 | |
|           flowchart: { htmlLabels: false, subGraphTitleMargin: { top: 10, bottom: 5 } },
 | |
|         }
 | |
|       );
 | |
|     });
 | |
|     it('Should render subgraphs with title margins and edge labels', () => {
 | |
|       imgSnapshotTest(
 | |
|         `flowchart LR
 | |
| 
 | |
|           subgraph TOP
 | |
|               direction TB
 | |
|               subgraph B1
 | |
|                   direction RL
 | |
|                   i1 --lb1-->f1
 | |
|               end
 | |
|               subgraph B2
 | |
|                   direction BT
 | |
|                   i2 --lb2-->f2
 | |
|               end
 | |
|           end
 | |
|           A --lb3--> TOP --lb4--> B
 | |
|           B1 --lb5--> B2
 | |
|         `,
 | |
|         { flowchart: { subGraphTitleMargin: { top: 10, bottom: 5 } } }
 | |
|       );
 | |
|     });
 | |
|   });
 | |
| });
 | 
