mirror of
				https://github.com/mermaid-js/mermaid.git
				synced 2025-10-26 01:14:09 +02:00 
			
		
		
		
	Compare commits
	
		
			7 Commits
		
	
	
		
			@mermaid-j
			...
			sidv/testA
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | aa036d831b | ||
|   | 4f2005eec8 | ||
|   | 8c99ee8da1 | ||
|   | 364ccfe6a6 | ||
|   | 10d6c17de8 | ||
|   | 3f58ebe063 | ||
|   | 1a6632e377 | 
| @@ -22,9 +22,9 @@ export const packageOptions = { | ||||
|     packageName: 'mermaid-zenuml', | ||||
|     file: 'detector.ts', | ||||
|   }, | ||||
|   'mermaid-layout-elk': { | ||||
|     name: 'mermaid-layout-elk', | ||||
|     packageName: 'mermaid-layout-elk', | ||||
|     file: 'layouts.ts', | ||||
|   'mermaid-flowchart-elk': { | ||||
|     name: 'mermaid-flowchart-elk', | ||||
|     packageName: 'mermaid-flowchart-elk', | ||||
|     file: 'detector.ts', | ||||
|   }, | ||||
| } as const; | ||||
|   | ||||
| @@ -1,7 +1,6 @@ | ||||
| import jison from 'jison'; | ||||
|  | ||||
| export const transformJison = (src: string): string => { | ||||
|   // @ts-ignore - Jison is not typed properly | ||||
|   const parser = new jison.Generator(src, { | ||||
|     moduleType: 'js', | ||||
|     'token-stack': true, | ||||
|   | ||||
| @@ -1,8 +0,0 @@ | ||||
| # Changesets | ||||
|  | ||||
| Hello and welcome! This folder has been automatically generated by `@changesets/cli`, a build tool that works | ||||
| with multi-package repos, or single-package repos to help you version and publish your code. You can | ||||
| find the full documentation for it [in our repository](https://github.com/changesets/changesets) | ||||
|  | ||||
| We have a quick list of common questions to get you started engaging with this project in | ||||
| [our documentation](https://github.com/changesets/changesets/blob/main/docs/common-questions.md) | ||||
| @@ -1,12 +0,0 @@ | ||||
| { | ||||
|   "$schema": "https://unpkg.com/@changesets/config@3.0.0/schema.json", | ||||
|   "changelog": ["@changesets/changelog-github", { "repo": "mermaid-js/mermaid" }], | ||||
|   "commit": false, | ||||
|   "fixed": [], | ||||
|   "linked": [], | ||||
|   "access": "public", | ||||
|   "baseBranch": "master", | ||||
|   "updateInternalDependencies": "patch", | ||||
|   "bumpVersionsWithWorkspaceProtocolOnly": true, | ||||
|   "ignore": ["@mermaid-js/docs", "@mermaid-js/webpack-test", "@mermaid-js/mermaid-example-diagram"] | ||||
| } | ||||
| @@ -28,7 +28,6 @@ controly | ||||
| CSSCLASS | ||||
| CYLINDEREND | ||||
| CYLINDERSTART | ||||
| DAGA | ||||
| datakey | ||||
| DEND | ||||
| descr | ||||
| @@ -91,7 +90,6 @@ reqs | ||||
| rewritelinks | ||||
| rgba | ||||
| RIGHTOF | ||||
| roughjs | ||||
| sankey | ||||
| sequencenumber | ||||
| shrc | ||||
| @@ -111,7 +109,6 @@ strikethrough | ||||
| stringifying | ||||
| struct | ||||
| STYLECLASS | ||||
| STYLEDEF | ||||
| STYLEOPTS | ||||
| subcomponent | ||||
| subcomponents | ||||
| @@ -120,8 +117,6 @@ SUBROUTINEEND | ||||
| SUBROUTINESTART | ||||
| Subschemas | ||||
| substr | ||||
| SVGG | ||||
| SVGSVG | ||||
| TAGEND | ||||
| TAGSTART | ||||
| techn | ||||
|   | ||||
| @@ -55,10 +55,8 @@ presetAttributify | ||||
| pyplot | ||||
| redmine | ||||
| rehype | ||||
| roughjs | ||||
| rscratch | ||||
| shiki | ||||
| Slidev | ||||
| sparkline | ||||
| sphinxcontrib | ||||
| ssim | ||||
|   | ||||
| @@ -9,7 +9,6 @@ elems | ||||
| gantt | ||||
| gitgraph | ||||
| gzipped | ||||
| handDrawn | ||||
| knsv | ||||
| Knut | ||||
| marginx | ||||
| @@ -18,7 +17,6 @@ Markdownish | ||||
| mermaidjs | ||||
| mindmap | ||||
| mindmaps | ||||
| mrtree | ||||
| multigraph | ||||
| nodesep | ||||
| NOTEGROUP | ||||
|   | ||||
| @@ -1,6 +1,4 @@ | ||||
| BRANDES | ||||
| circo | ||||
| handDrawn | ||||
| handdrawn | ||||
| KOEPF | ||||
| neato | ||||
| newbranch | ||||
|   | ||||
| @@ -8,8 +8,8 @@ import { defaultOptions, getBuildConfig } from './util.js'; | ||||
| const shouldVisualize = process.argv.includes('--visualize'); | ||||
|  | ||||
| const buildPackage = async (entryName: keyof typeof packageOptions) => { | ||||
|   const commonOptions: MermaidBuildOptions = { ...defaultOptions, entryName } as const; | ||||
|   const buildConfigs: MermaidBuildOptions[] = [ | ||||
|   const commonOptions = { ...defaultOptions, entryName } as const; | ||||
|   const buildConfigs = [ | ||||
|     // package.mjs | ||||
|     { ...commonOptions }, | ||||
|     // package.min.mjs | ||||
|   | ||||
| @@ -8,7 +8,7 @@ import { jisonPlugin } from './jisonPlugin.js'; | ||||
|  | ||||
| const __dirname = fileURLToPath(new URL('.', import.meta.url)); | ||||
|  | ||||
| export interface MermaidBuildOptions extends BuildOptions { | ||||
| export interface MermaidBuildOptions { | ||||
|   minify: boolean; | ||||
|   core: boolean; | ||||
|   metafile: boolean; | ||||
|   | ||||
							
								
								
									
										2
									
								
								.github/pull_request_template.md
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/pull_request_template.md
									
									
									
									
										vendored
									
									
								
							| @@ -15,4 +15,4 @@ Make sure you | ||||
| - [ ] :book: have read the [contribution guidelines](https://mermaid.js.org/community/contributing.html) | ||||
| - [ ] :computer: have added necessary unit/e2e tests. | ||||
| - [ ] :notebook: have added documentation. Make sure [`MERMAID_RELEASE_VERSION`](https://mermaid.js.org/community/contributing.html#update-documentation) is used for all new features. | ||||
| - [ ] :butterfly: If your PR makes a change that should be noted in one or more packages' changelogs, generate a changeset by running `pnpm changeset` and following the prompts. Changesets that add features should be `minor` and those that fix bugs should be `patch`. Please prefix changeset messages with `feat:`, `fix:`, or `chore:`. | ||||
| - [ ] :bookmark: targeted `develop` branch | ||||
|   | ||||
							
								
								
									
										36
									
								
								.github/release-drafter.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										36
									
								
								.github/release-drafter.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,36 @@ | ||||
| name-template: '$NEXT_PATCH_VERSION' | ||||
| tag-template: '$NEXT_PATCH_VERSION' | ||||
| categories: | ||||
|   - title: '🚨 **Breaking Changes**' | ||||
|     labels: | ||||
|       - 'Breaking Change' | ||||
|   - title: '🚀 Features' | ||||
|     labels: | ||||
|       - 'Type: Enhancement' | ||||
|       - 'feature' # deprecated, new PRs shouldn't have this | ||||
|   - title: '🐛 Bug Fixes' | ||||
|     labels: | ||||
|       - 'Type: Bug / Error' | ||||
|       - 'fix' # deprecated, new PRs shouldn't have this | ||||
|   - title: '🧰 Maintenance' | ||||
|     labels: | ||||
|       - 'Type: Other' | ||||
|       - 'chore' # deprecated, new PRs shouldn't have this | ||||
|   - title: '⚡️ Performance' | ||||
|     labels: | ||||
|       - 'Type: Performance' | ||||
|   - title: '📚 Documentation' | ||||
|     labels: | ||||
|       - 'Area: Documentation' | ||||
| change-template: '- $TITLE (#$NUMBER) @$AUTHOR' | ||||
| sort-by: title | ||||
| sort-direction: ascending | ||||
| exclude-labels: | ||||
|   - 'Skip changelog' | ||||
| no-changes-template: 'This release contains minor changes and bugfixes.' | ||||
| template: | | ||||
|   # Release Notes | ||||
|  | ||||
|   $CHANGES | ||||
|  | ||||
|   🎉 **Thanks to all contributors helping with this release!** 🎉 | ||||
							
								
								
									
										43
									
								
								.github/workflows/autofix.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										43
									
								
								.github/workflows/autofix.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,43 +0,0 @@ | ||||
| name: autofix.ci # needed to securely identify the workflow | ||||
|  | ||||
| on: | ||||
|   pull_request: | ||||
|     branches-ignore: | ||||
|       - 'renovate/**' | ||||
| permissions: | ||||
|   contents: read | ||||
|  | ||||
| jobs: | ||||
|   autofix: | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - uses: actions/checkout@v4 | ||||
|  | ||||
|       - uses: pnpm/action-setup@v4 | ||||
|         # uses version from "packageManager" field in package.json | ||||
|  | ||||
|       - name: Setup Node.js | ||||
|         uses: actions/setup-node@v4 | ||||
|         with: | ||||
|           cache: pnpm | ||||
|           node-version-file: '.node-version' | ||||
|  | ||||
|       - name: Install Packages | ||||
|         run: | | ||||
|           pnpm install --frozen-lockfile | ||||
|         env: | ||||
|           CYPRESS_CACHE_FOLDER: .cache/Cypress | ||||
|  | ||||
|       - name: Fix Linting | ||||
|         shell: bash | ||||
|         run: pnpm -w run lint:fix | ||||
|  | ||||
|       - name: Sync `./src/config.type.ts` with `./src/schemas/config.schema.yaml` | ||||
|         shell: bash | ||||
|         run: pnpm run --filter mermaid types:build-config | ||||
|  | ||||
|       - name: Build Docs | ||||
|         working-directory: ./packages/mermaid | ||||
|         run: pnpm run docs:build | ||||
|  | ||||
|       - uses: autofix-ci/action@ff86a557419858bb967097bfc916833f5647fa8c | ||||
							
								
								
									
										8
									
								
								.github/workflows/e2e.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										8
									
								
								.github/workflows/e2e.yml
									
									
									
									
										vendored
									
									
								
							| @@ -117,7 +117,7 @@ jobs: | ||||
|         id: cypress | ||||
|         # If CYPRESS_RECORD_KEY is set, run in parallel on all containers | ||||
|         # Otherwise (e.g. if running from fork), we run on a single container only | ||||
|         if: ${{ env.shouldRunParallel == 'true' || ( matrix.containers == 1 ) }} | ||||
|         if: ${{ env.shouldRunParallel || ( matrix.containers == 1 ) }} | ||||
|         with: | ||||
|           install: false | ||||
|           start: pnpm run dev:coverage | ||||
| @@ -125,14 +125,14 @@ jobs: | ||||
|           browser: chrome | ||||
|           # Disable recording if we don't have an API key | ||||
|           # e.g. if this action was run from a fork | ||||
|           record: ${{ env.shouldRunParallel == 'true' }} | ||||
|           parallel: ${{ env.shouldRunParallel == 'true' }} | ||||
|           record: ${{ env.shouldRunParallel }} | ||||
|           parallel: ${{ env.shouldRunParallel }} | ||||
|         env: | ||||
|           CYPRESS_RECORD_KEY: ${{ secrets.CYPRESS_RECORD_KEY }} | ||||
|           VITEST_COVERAGE: true | ||||
|           CYPRESS_COMMIT: ${{ github.sha }} | ||||
|           ARGOS_TOKEN: ${{ secrets.ARGOS_TOKEN }} | ||||
|           ARGOS_PARALLEL: ${{ env.shouldRunParallel == 'true' }} | ||||
|           ARGOS_PARALLEL: ${{ env.shouldRunParallel }} | ||||
|           ARGOS_PARALLEL_TOTAL: 4 | ||||
|           ARGOS_PARALLEL_INDEX: ${{ matrix.containers }} | ||||
|  | ||||
|   | ||||
							
								
								
									
										12
									
								
								.github/workflows/lint.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										12
									
								
								.github/workflows/lint.yml
									
									
									
									
										vendored
									
									
								
							| @@ -82,3 +82,15 @@ jobs: | ||||
|         working-directory: ./packages/mermaid | ||||
|         continue-on-error: ${{ github.event_name == 'push' }} | ||||
|         run: pnpm run docs:verify | ||||
|  | ||||
|       - name: Rebuild Docs | ||||
|         if: ${{ steps.verifyDocs.outcome == 'failure' && github.event_name == 'push' }} | ||||
|         working-directory: ./packages/mermaid | ||||
|         run: pnpm run docs:build | ||||
|  | ||||
|       - name: Commit changes | ||||
|         uses: EndBug/add-and-commit@v9 | ||||
|         if: ${{ steps.verifyDocs.outcome == 'failure' && github.event_name == 'push' }} | ||||
|         with: | ||||
|           message: 'Update docs' | ||||
|           add: 'docs/*' | ||||
|   | ||||
							
								
								
									
										23
									
								
								.github/workflows/release-draft.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										23
									
								
								.github/workflows/release-draft.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,23 @@ | ||||
| name: Draft Release | ||||
|  | ||||
| on: | ||||
|   push: | ||||
|     branches: | ||||
|       - master | ||||
|  | ||||
| permissions: | ||||
|   contents: read | ||||
|  | ||||
| jobs: | ||||
|   draft-release: | ||||
|     runs-on: ubuntu-latest | ||||
|     permissions: | ||||
|       contents: write # write permission is required to create a GitHub release | ||||
|       pull-requests: read # required to read PR titles/labels | ||||
|     steps: | ||||
|       - name: Draft Release | ||||
|         uses: release-drafter/release-drafter@v6 | ||||
|         with: | ||||
|           disable-autolabeler: true | ||||
|         env: | ||||
|           GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} | ||||
							
								
								
									
										37
									
								
								.github/workflows/release.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										37
									
								
								.github/workflows/release.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,37 +0,0 @@ | ||||
| name: Release | ||||
|  | ||||
| on: | ||||
|   push: | ||||
|     branches: | ||||
|       - master | ||||
|  | ||||
| concurrency: ${{ github.workflow }}-${{ github.ref }} | ||||
|  | ||||
| jobs: | ||||
|   release: | ||||
|     name: Release | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - name: Checkout Repo | ||||
|         uses: actions/checkout@v3 | ||||
|  | ||||
|       - uses: pnpm/action-setup@v4 | ||||
|  | ||||
|       - name: Setup Node.js | ||||
|         uses: actions/setup-node@v4 | ||||
|         with: | ||||
|           cache: pnpm | ||||
|           node-version-file: '.node-version' | ||||
|  | ||||
|       - name: Install Packages | ||||
|         run: pnpm install --frozen-lockfile | ||||
|  | ||||
|       - name: Create Release Pull Request or Publish to npm | ||||
|         id: changesets | ||||
|         uses: changesets/action@v1 | ||||
|         with: | ||||
|           version: pnpm changeset:version | ||||
|           publish: pnpm changeset:publish | ||||
|         env: | ||||
|           GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} | ||||
|           NPM_TOKEN: ${{ secrets.NPM_TOKEN }} | ||||
							
								
								
									
										2
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -35,7 +35,7 @@ cypress/snapshots/ | ||||
| .tsbuildinfo | ||||
| tsconfig.tsbuildinfo | ||||
|  | ||||
| #knsv*.html | ||||
| knsv*.html | ||||
| local*.html | ||||
| stats/ | ||||
|  | ||||
|   | ||||
| @@ -16,5 +16,3 @@ generated/ | ||||
| # Ignore the files creates in /demos/dev except for example.html | ||||
| demos/dev/** | ||||
| !/demos/dev/example.html | ||||
| # TODO: Lots of errors to fix | ||||
| cypress/platform/state-refactor.html | ||||
|   | ||||
							
								
								
									
										14
									
								
								cypress/integration/other/flowchart-elk.spec.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								cypress/integration/other/flowchart-elk.spec.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,14 @@ | ||||
| import { urlSnapshotTest, openURLAndVerifyRendering } from '../../helpers/util.ts'; | ||||
|  | ||||
| describe('Flowchart elk', () => { | ||||
|   it('should use dagre as fallback', () => { | ||||
|     urlSnapshotTest('http://localhost:9000/flow-elk.html', { | ||||
|       name: 'flow-elk fallback to dagre', | ||||
|     }); | ||||
|   }); | ||||
|   it('should allow overriding with external package', () => { | ||||
|     urlSnapshotTest('http://localhost:9000/flow-elk.html?elk=true', { | ||||
|       name: 'flow-elk overriding dagre with elk', | ||||
|     }); | ||||
|   }); | ||||
| }); | ||||
| @@ -11,27 +11,6 @@ describe('Git Graph diagram', () => { | ||||
|       {} | ||||
|     ); | ||||
|   }); | ||||
|   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 } } } | ||||
|     ); | ||||
|   }); | ||||
|   // it(`ultraFastTest`, function () { | ||||
|   //   // Navigate to the url we want to test | ||||
|   //   // ⭐️ Note to see visual bugs, run the test using the above URL for the 1st run. | ||||
|   | ||||
| @@ -236,7 +236,7 @@ describe('Block diagram', () => { | ||||
|     ); | ||||
|   }); | ||||
|  | ||||
|   it('BL17: width alignment - blocks shold be equal in width', () => { | ||||
|   it('BL16: width alignment - blocks shold be equal in width', () => { | ||||
|     imgSnapshotTest( | ||||
|       `block-beta | ||||
|     A("This is the text") | ||||
| @@ -247,7 +247,7 @@ describe('Block diagram', () => { | ||||
|     ); | ||||
|   }); | ||||
|  | ||||
|   it('BL18: block types 1 - square, rounded and circle', () => { | ||||
|   it('BL17: block types 1 - square, rounded and circle', () => { | ||||
|     imgSnapshotTest( | ||||
|       `block-beta | ||||
|     A["square"] | ||||
| @@ -258,7 +258,7 @@ describe('Block diagram', () => { | ||||
|     ); | ||||
|   }); | ||||
|  | ||||
|   it('BL19: block types 2 - odd, diamond and hexagon', () => { | ||||
|   it('BL18: block types 2 - odd, diamond and hexagon', () => { | ||||
|     imgSnapshotTest( | ||||
|       `block-beta | ||||
|     A>"rect_left_inv_arrow"] | ||||
| @@ -269,7 +269,7 @@ describe('Block diagram', () => { | ||||
|     ); | ||||
|   }); | ||||
|  | ||||
|   it('BL20: block types 3 - stadium', () => { | ||||
|   it('BL19: block types 3 - stadium', () => { | ||||
|     imgSnapshotTest( | ||||
|       `block-beta | ||||
|     A(["stadium"]) | ||||
| @@ -278,7 +278,7 @@ describe('Block diagram', () => { | ||||
|     ); | ||||
|   }); | ||||
|  | ||||
|   it('BL21: block types 4 - lean right, lean left, trapezoid and inv trapezoid', () => { | ||||
|   it('BL20: block types 4 - lean right, lean left, trapezoid and inv trapezoid', () => { | ||||
|     imgSnapshotTest( | ||||
|       `block-beta | ||||
|     A[/"lean right"/] | ||||
| @@ -290,7 +290,7 @@ describe('Block diagram', () => { | ||||
|     ); | ||||
|   }); | ||||
|  | ||||
|   it('BL22: block types 1 - square, rounded and circle', () => { | ||||
|   it('BL21: block types 1 - square, rounded and circle', () => { | ||||
|     imgSnapshotTest( | ||||
|       `block-beta | ||||
|     A["square"] | ||||
| @@ -301,7 +301,7 @@ describe('Block diagram', () => { | ||||
|     ); | ||||
|   }); | ||||
|  | ||||
|   it('BL23: sizing - it should be possible to make a block wider', () => { | ||||
|   it('BL22: sizing - it should be possible to make a block wider', () => { | ||||
|     imgSnapshotTest( | ||||
|       `block-beta | ||||
|       A("rounded"):2 | ||||
| @@ -312,7 +312,7 @@ describe('Block diagram', () => { | ||||
|     ); | ||||
|   }); | ||||
|  | ||||
|   it('BL24: sizing - it should be possible to make a composite block wider', () => { | ||||
|   it('BL23: sizing - it should be possible to make a composite block wider', () => { | ||||
|     imgSnapshotTest( | ||||
|       `block-beta | ||||
|       block:2 | ||||
| @@ -324,7 +324,7 @@ describe('Block diagram', () => { | ||||
|     ); | ||||
|   }); | ||||
|  | ||||
|   it('BL25: block in the middle with space on each side', () => { | ||||
|   it('BL24: block in the middle with space on each side', () => { | ||||
|     imgSnapshotTest( | ||||
|       `block-beta | ||||
|         columns 3 | ||||
| @@ -335,7 +335,7 @@ describe('Block diagram', () => { | ||||
|       {} | ||||
|     ); | ||||
|   }); | ||||
|   it('BL26: space and an edge', () => { | ||||
|   it('BL25: space and an edge', () => { | ||||
|     imgSnapshotTest( | ||||
|       `block-beta | ||||
|   columns 5 | ||||
| @@ -345,7 +345,7 @@ describe('Block diagram', () => { | ||||
|       {} | ||||
|     ); | ||||
|   }); | ||||
|   it('BL27: block sizes for regular blocks', () => { | ||||
|   it('BL26: block sizes for regular blocks', () => { | ||||
|     imgSnapshotTest( | ||||
|       `block-beta | ||||
|   columns 3 | ||||
| @@ -354,7 +354,7 @@ describe('Block diagram', () => { | ||||
|       {} | ||||
|     ); | ||||
|   }); | ||||
|   it('BL28: composite block with a set width - f should use the available space', () => { | ||||
|   it('BL27: composite block with a set width - f should use the available space', () => { | ||||
|     imgSnapshotTest( | ||||
|       `block-beta | ||||
|   columns 3 | ||||
| @@ -363,12 +363,11 @@ describe('Block diagram', () => { | ||||
|       f | ||||
|   end | ||||
|   g | ||||
|   `, | ||||
|       `, | ||||
|       {} | ||||
|     ); | ||||
|   }); | ||||
|  | ||||
|   it('BL29: composite block with a set width - f and g should split the available space', () => { | ||||
|   it('BL23: composite block with a set width - f and g should split the available space', () => { | ||||
|     imgSnapshotTest( | ||||
|       `block-beta | ||||
|   columns 3 | ||||
| @@ -380,7 +379,7 @@ describe('Block diagram', () => { | ||||
|   h | ||||
|   i | ||||
|   j | ||||
|   `, | ||||
|       `, | ||||
|       {} | ||||
|     ); | ||||
|   }); | ||||
|   | ||||
| @@ -1,7 +1,7 @@ | ||||
| import { imgSnapshotTest, renderGraph } from '../../helpers/util.ts'; | ||||
|  | ||||
| describe('C4 diagram', () => { | ||||
|   it('C4.1 should render a simple C4Context diagram', () => { | ||||
|   it('should render a simple C4Context diagram', () => { | ||||
|     imgSnapshotTest( | ||||
|       ` | ||||
|       C4Context | ||||
| @@ -31,7 +31,7 @@ describe('C4 diagram', () => { | ||||
|       {} | ||||
|     ); | ||||
|   }); | ||||
|   it('C4.2 should render a simple C4Container diagram', () => { | ||||
|   it('should render a simple C4Container diagram', () => { | ||||
|     imgSnapshotTest( | ||||
|       ` | ||||
|       C4Container | ||||
| @@ -50,7 +50,7 @@ describe('C4 diagram', () => { | ||||
|       {} | ||||
|     ); | ||||
|   }); | ||||
|   it('C4.3 should render a simple C4Component diagram', () => { | ||||
|   it('should render a simple C4Component diagram', () => { | ||||
|     imgSnapshotTest( | ||||
|       ` | ||||
|       C4Component | ||||
| @@ -68,7 +68,7 @@ describe('C4 diagram', () => { | ||||
|       {} | ||||
|     ); | ||||
|   }); | ||||
|   it('C4.4 should render a simple C4Dynamic diagram', () => { | ||||
|   it('should render a simple C4Dynamic diagram', () => { | ||||
|     imgSnapshotTest( | ||||
|       ` | ||||
|       C4Dynamic | ||||
| @@ -91,7 +91,7 @@ describe('C4 diagram', () => { | ||||
|       {} | ||||
|     ); | ||||
|   }); | ||||
|   it('C4.5 should render a simple C4Deployment diagram', () => { | ||||
|   it('should render a simple C4Deployment diagram', () => { | ||||
|     imgSnapshotTest( | ||||
|       ` | ||||
|       C4Deployment | ||||
|   | ||||
| @@ -76,7 +76,7 @@ describe('Class diagram V2', () => { | ||||
|     ); | ||||
|   }); | ||||
|  | ||||
|   it('2.1 should render a simple class diagram with different visibilities', () => { | ||||
|   it('should render a simple class diagram with different visibilities', () => { | ||||
|     imgSnapshotTest( | ||||
|       ` | ||||
|     classDiagram-v2 | ||||
| @@ -93,7 +93,7 @@ describe('Class diagram V2', () => { | ||||
|     ); | ||||
|   }); | ||||
|  | ||||
|   it('3: should render multiple class diagrams', () => { | ||||
|   it('should render multiple class diagrams', () => { | ||||
|     imgSnapshotTest( | ||||
|       [ | ||||
|         ` | ||||
|   | ||||
| @@ -3,7 +3,7 @@ import { imgSnapshotTest } from '../../helpers/util'; | ||||
| describe('Error Diagrams', () => { | ||||
|   beforeEach(() => { | ||||
|     cy.on('uncaught:exception', (err) => { | ||||
|       expect(err.message).to.include('error'); | ||||
|       expect(err.message).to.include('Parse error'); | ||||
|       // return false to prevent the error from | ||||
|       // failing this test | ||||
|       return false; | ||||
|   | ||||
| @@ -837,26 +837,6 @@ subgraph "\`**Two**\`" | ||||
|   in the hat\`") -- "\`1o **ipa**\`" --> d("The dog in the hog") | ||||
| end | ||||
|  | ||||
| `, | ||||
|           { flowchart: { titleTopMargin: 0 } } | ||||
|         ); | ||||
|       }); | ||||
|       it('Sub graphs and markdown strings', () => { | ||||
|         imgSnapshotTest( | ||||
|           `--- | ||||
| config: | ||||
|   layout: elk | ||||
| --- | ||||
|  | ||||
| flowchart LR | ||||
|  subgraph subgraph_ko6czgs5u["Untitled subgraph"] | ||||
|         D["Option 1"] | ||||
|   end | ||||
|     C{"Evaluate"} -- One --> D | ||||
|     C -- Two --> E(("Option 2")) | ||||
|     D --> E | ||||
|       A["A"] | ||||
|  | ||||
| `, | ||||
|           { flowchart: { titleTopMargin: 0 } } | ||||
|         ); | ||||
| @@ -875,7 +855,7 @@ describe('Title and arrow styling #4813', () => { | ||||
|       flowchart LR | ||||
|       A-->B | ||||
|       A-->C`, | ||||
|       { layout: 'elk' } | ||||
|       { flowchart: { defaultRenderer: 'elk' } } | ||||
|     ); | ||||
|     cy.get('svg').should((svg) => { | ||||
|       const title = svg[0].querySelector('text'); | ||||
| @@ -891,14 +871,15 @@ describe('Title and arrow styling #4813', () => { | ||||
|       B-.-oC | ||||
|       C==xD | ||||
|       D ~~~ A`, | ||||
|       { layout: 'elk' } | ||||
|       { flowchart: { defaultRenderer: 'elk' } } | ||||
|     ); | ||||
|     cy.get('svg').should((svg) => { | ||||
|       const edges = svg[0].querySelectorAll('.edges path'); | ||||
|       expect(edges[0].getAttribute('class')).to.contain('edge-pattern-solid'); | ||||
|       expect(edges[1].getAttribute('class')).to.contain('edge-pattern-dotted'); | ||||
|       expect(edges[2].getAttribute('class')).to.contain('edge-thickness-thick'); | ||||
|       expect(edges[3].getAttribute('class')).to.contain('edge-thickness-invisible'); | ||||
|       console.log(edges); | ||||
|       expect(edges[0]).to.have.attr('pattern', 'solid'); | ||||
|       expect(edges[1]).to.have.attr('pattern', 'dotted'); | ||||
|       expect(edges[2]).to.have.css('stroke-width', '3.5px'); | ||||
|       expect(edges[3]).to.have.css('stroke-width', '1.5px'); | ||||
|     }); | ||||
|   }); | ||||
| }); | ||||
|   | ||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -99,7 +99,7 @@ describe('Flowchart v2', () => { | ||||
|       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(417 * 0.95, 417 * 1.05); | ||||
|       expect(maxWidthValue).to.be.within(290 * 0.95 - 1, 290 * 1.05); | ||||
|     }); | ||||
|   }); | ||||
|   it('8: should render a flowchart when useMaxWidth is false', () => { | ||||
| @@ -118,7 +118,7 @@ describe('Flowchart v2', () => { | ||||
|       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(417 * 0.95, 417 * 1.05); | ||||
|       expect(width).to.be.within(290 * 0.95 - 1, 290 * 1.05); | ||||
|       expect(svg).to.not.have.attr('style'); | ||||
|     }); | ||||
|   }); | ||||
| @@ -1047,9 +1047,7 @@ end | ||||
|           A --lb3--> TOP --lb4--> B | ||||
|           B1 --lb5--> B2 | ||||
|         `, | ||||
|         { | ||||
|           flowchart: { subGraphTitleMargin: { top: 10, bottom: 5 } }, | ||||
|         } | ||||
|         { flowchart: { subGraphTitleMargin: { top: 10, bottom: 5 } } } | ||||
|       ); | ||||
|     }); | ||||
|   }); | ||||
|   | ||||
| @@ -733,7 +733,7 @@ describe('Graph', () => { | ||||
|   }); | ||||
|   it('38: should render a flowchart when useMaxWidth is true (default)', () => { | ||||
|     renderGraph( | ||||
|       `flowchart TD | ||||
|       `graph TD | ||||
|       A[Christmas] -->|Get money| B(Go shopping) | ||||
|       B --> C{Let me think} | ||||
|       C -->|One| D[Laptop] | ||||
| @@ -751,7 +751,7 @@ describe('Graph', () => { | ||||
|       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(446 * 0.9, 446 * 1.1); | ||||
|       expect(maxWidthValue).to.be.within(300 * 0.9, 300 * 1.1); | ||||
|     }); | ||||
|   }); | ||||
|   it('39: should render a flowchart when useMaxWidth is false', () => { | ||||
| @@ -770,7 +770,7 @@ describe('Graph', () => { | ||||
|       const width = parseFloat(svg.attr('width')); | ||||
|       // use within because the absolute value can be slightly different depending on the environment ±10% | ||||
|       // expect(height).to.be.within(446 * 0.95, 446 * 1.05); | ||||
|       expect(width).to.be.within(446 * 0.9, 446 * 1.1); | ||||
|       expect(width).to.be.within(300 * 0.9, 300 * 1.1); | ||||
|       expect(svg).to.not.have.attr('style'); | ||||
|     }); | ||||
|   }); | ||||
| @@ -905,16 +905,13 @@ graph TD | ||||
|   it('67: should be able to style default node independently', () => { | ||||
|     imgSnapshotTest( | ||||
|       ` | ||||
|       flowchart TD | ||||
|     flowchart TD | ||||
|       classDef default fill:#a34 | ||||
|       hello --> default | ||||
|  | ||||
|       style default stroke:#000,stroke-width:4px | ||||
|     `, | ||||
|       { | ||||
|         flowchart: { htmlLabels: true }, | ||||
|         securityLevel: 'loose', | ||||
|       } | ||||
|       { htmlLabels: true, flowchart: { htmlLabels: true }, securityLevel: 'loose' } | ||||
|     ); | ||||
|   }); | ||||
| }); | ||||
|   | ||||
| @@ -1532,41 +1532,5 @@ gitGraph TB: | ||||
|         {} | ||||
|       ); | ||||
|     }); | ||||
|     it('75: should render a gitGraph with multiple tags on a merge commit on bottom-to-top orientation', () => { | ||||
|       imgSnapshotTest( | ||||
|         `gitGraph BT: | ||||
|         commit id: "ZERO" | ||||
|         branch develop | ||||
|         commit id:"A" | ||||
|         checkout main | ||||
|         commit id:"ONE" | ||||
|         checkout develop | ||||
|         commit id:"B" | ||||
|         checkout main | ||||
|         merge develop id:"Release 1.0" type:HIGHLIGHT tag: "SAML v2.0" tag: "OpenID v1.1" | ||||
|         commit id:"TWO" | ||||
|         checkout develop | ||||
|         commit id:"C"`, | ||||
|         {} | ||||
|       ); | ||||
|     }); | ||||
|   }); | ||||
|   it('76: should render a gitGraph with multiple tags on a merge commit on left-to-right orientation', () => { | ||||
|     imgSnapshotTest( | ||||
|       `gitGraph | ||||
|     commit id: "ZERO" | ||||
|     branch develop | ||||
|     commit id:"A" | ||||
|     checkout main | ||||
|     commit id:"ONE" | ||||
|     checkout develop | ||||
|     commit id:"B" | ||||
|     checkout main | ||||
|     merge develop id:"Release 1.0" type:HIGHLIGHT tag: "SAML v2.0" tag: "OpenID v1.1" | ||||
|     commit id:"TWO" | ||||
|     checkout develop | ||||
|     commit id:"C"`, | ||||
|       {} | ||||
|     ); | ||||
|   }); | ||||
| }); | ||||
|   | ||||
| @@ -10,15 +10,6 @@ describe('packet structure', () => { | ||||
|     ); | ||||
|   }); | ||||
|  | ||||
|   it('should render a simple packet diagram without ranges', () => { | ||||
|     imgSnapshotTest( | ||||
|       `packet-beta | ||||
|   0: "h" | ||||
|   1: "i" | ||||
| ` | ||||
|     ); | ||||
|   }); | ||||
|  | ||||
|   it('should render a complex packet diagram', () => { | ||||
|     imgSnapshotTest( | ||||
|       `packet-beta | ||||
|   | ||||
| @@ -20,7 +20,7 @@ describe('Quadrant Chart', () => { | ||||
|     quadrant-2 Need to promote | ||||
|     quadrant-3 Re-evaluate | ||||
|     quadrant-4 May be improved | ||||
|     Campaign A: [0.3, 0.6] | ||||
|     Campaign A: [0.3, 0.61] | ||||
|     Campaign B: [0.45, 0.23] | ||||
|     Campaign C: [0.57, 0.69] | ||||
|     Campaign D: [0.78, 0.34] | ||||
| @@ -54,7 +54,7 @@ describe('Quadrant Chart', () => { | ||||
|     x-axis Low Reach --> High Reach | ||||
|     y-axis Low Engagement --> High Engagement | ||||
|     quadrant-1 We should expand | ||||
|     quadrant-2 Need to promote | ||||
|     quadrant-2 Need to promot | ||||
|     quadrant-3 Re-evaluate | ||||
|     quadrant-4 May be improved | ||||
|       `, | ||||
|   | ||||
| @@ -542,43 +542,6 @@ stateDiagram-v2 | ||||
|         { logLevel: 0, fontFamily: 'courier' } | ||||
|       ); | ||||
|     }); | ||||
|     it(' can have styles applied ', () => { | ||||
|       imgSnapshotTest( | ||||
|         ` | ||||
| stateDiagram-v2 | ||||
| AState | ||||
| style AState fill:#636,border:1px solid red,color:white; | ||||
|         `, | ||||
|         { logLevel: 0, fontFamily: 'courier' } | ||||
|       ); | ||||
|     }); | ||||
|     it(' should let styles take preceedence over classes', () => { | ||||
|       imgSnapshotTest( | ||||
|         ` | ||||
| stateDiagram-v2 | ||||
| AState: Should NOT be white | ||||
| BState | ||||
| classDef exampleStyleClass fill:#fff,color: blue; | ||||
| class AState,BState exampleStyleClass | ||||
| style AState fill:#636,border:1px solid red,color:white; | ||||
|         `, | ||||
|         { logLevel: 0, fontFamily: 'courier' } | ||||
|       ); | ||||
|     }); | ||||
|     it(' should allow styles to take effect in stubgraphs', () => { | ||||
|       imgSnapshotTest( | ||||
|         ` | ||||
|   stateDiagram | ||||
|     state roundWithTitle { | ||||
|       C: Black with white text | ||||
|     } | ||||
|     D: Black with white text | ||||
|  | ||||
|     style C,D stroke:#00f, fill:black, color:white | ||||
|         `, | ||||
|         { logLevel: 0, fontFamily: 'courier' } | ||||
|       ); | ||||
|     }); | ||||
|   }); | ||||
|   it('1433: should render a simple state diagram with a title', () => { | ||||
|     imgSnapshotTest( | ||||
| @@ -588,20 +551,6 @@ title: simple state diagram | ||||
| stateDiagram-v2 | ||||
| [*] --> State1 | ||||
| State1 --> [*] | ||||
| `, | ||||
|       {} | ||||
|     ); | ||||
|   }); | ||||
|   it('should align dividers correctly', () => { | ||||
|     imgSnapshotTest( | ||||
|       `stateDiagram-v2 | ||||
|   state s2 { | ||||
|       s3 | ||||
|       -- | ||||
|       s4 | ||||
|       -- | ||||
|       55 | ||||
|   } | ||||
| `, | ||||
|       {} | ||||
|     ); | ||||
|   | ||||
| @@ -205,7 +205,7 @@ describe('XY Chart', () => { | ||||
|         title "Sales Revenue" | ||||
|         x-axis Months [jan, feb, mar, apr, may, jun, jul, aug, sep, oct, nov, dec] | ||||
|         y-axis "Revenue (in $)" 4000 --> 11000 | ||||
|         bar [5000, 6000, 7500, 8200, 9500, 10500, 11000, 10200, 9200, 8500, 7000, 6000] | ||||
|         bar [5000, 6000, 7500, 8200, 9500, 10500, 11000, 10200, 9200, 8500, 7000, 6010] | ||||
|         line [5000, 6000, 7500, 8200, 9500, 10500, 11000, 10200, 9200, 8500, 7000, 6000] | ||||
|       `, | ||||
|       {} | ||||
| @@ -239,15 +239,15 @@ describe('XY Chart', () => { | ||||
|       config: | ||||
|         xyChart: | ||||
|           xAxis: | ||||
|             showTick: false | ||||
|             showTick: true | ||||
|           yAxis: | ||||
|             showTick: false | ||||
|             showTick: true | ||||
|       --- | ||||
|       xychart-beta | ||||
|         title "Sales Revenue" | ||||
|         x-axis Months [jan, feb, mar, apr, may, jun, jul, aug, sep, oct, nov, dec] | ||||
|         y-axis "Revenue (in $)" 4000 --> 11000 | ||||
|         bar [5000, 6000, 7500, 8200, 9500, 10500, 11000, 10200, 9200, 8500, 7000, 6000] | ||||
|         bar [5000, 6000, 7500, 8200, 9500, 10500, 11000, 10200, 9200, 8500, 7000, 5000] | ||||
|         line [5000, 6000, 7500, 8200, 9500, 10500, 11000, 10200, 9200, 8500, 7000, 6000] | ||||
|       `, | ||||
|       {} | ||||
| @@ -268,7 +268,7 @@ describe('XY Chart', () => { | ||||
|         title "Sales Revenue" | ||||
|         x-axis Months [jan, feb, mar, apr, may, jun, jul, aug, sep, oct, nov, dec] | ||||
|         y-axis "Revenue (in $)" 4000 --> 11000 | ||||
|         bar [5000, 6000, 7500, 8200, 9500, 10500, 11000, 10200, 9200, 8500, 7000, 6000] | ||||
|         bar [5000, 6000, 7500, 8200, 9500, 10500, 11000, 10200, 9200, 8500, 7000, 6001] | ||||
|         line [5000, 6000, 7500, 8200, 9500, 10500, 11000, 10200, 9200, 8500, 7000, 6000] | ||||
|       `, | ||||
|       {} | ||||
|   | ||||
| @@ -1,866 +0,0 @@ | ||||
| <html> | ||||
|   <head> | ||||
|     <link href="https://fonts.googleapis.com/css?family=Montserrat&display=swap" rel="stylesheet" /> | ||||
|     <link href="https://unpkg.com/tailwindcss@^1.0/dist/tailwind.min.css" rel="stylesheet" /> | ||||
|     <link | ||||
|       rel="stylesheet" | ||||
|       href="https://cdnjs.cloudflare.com/ajax/libs/font-awesome/4.7.0/css/font-awesome.min.css" | ||||
|     /> | ||||
|     <link | ||||
|       href="https://cdn.jsdelivr.net/npm/@mdi/font@6.9.96/css/materialdesignicons.min.css" | ||||
|       rel="stylesheet" | ||||
|     /> | ||||
|     <link | ||||
|       href="https://fonts.googleapis.com/css?family=Noto+Sans+SC&display=swap" | ||||
|       rel="stylesheet" | ||||
|     /> | ||||
|     <link rel="preconnect" href="https://fonts.googleapis.com" /> | ||||
|     <link rel="preconnect" href="https://fonts.gstatic.com" crossorigin /> | ||||
|     <link | ||||
|       href="https://fonts.googleapis.com/css2?family=Kalam:wght@300;400;700&display=swap" | ||||
|       rel="stylesheet" | ||||
|     /> | ||||
|     <link | ||||
|       href="https://fonts.googleapis.com/css2?family=Caveat:wght@400..700&family=Kalam:wght@300;400;700&family=Rubik+Mono+One&display=swap" | ||||
|       rel="stylesheet" | ||||
|     /> | ||||
|     <link | ||||
|       href="https://fonts.googleapis.com/css2?family=Kalam:wght@300;400;700&family=Rubik+Mono+One&display=swap" | ||||
|       rel="stylesheet" | ||||
|     /> | ||||
|  | ||||
|     <style> | ||||
|       body { | ||||
|         font-family: 'Arial'; | ||||
|       } | ||||
|  | ||||
|       table { | ||||
|         width: 100%; | ||||
|         border-collapse: collapse; | ||||
|         table-layout: fixed; | ||||
|       } | ||||
|  | ||||
|       th, | ||||
|       td { | ||||
|         border: 1px solid black; | ||||
|         padding: 10px; | ||||
|         text-align: center; | ||||
|         vertical-align: middle; | ||||
|       } | ||||
|  | ||||
|       .separator { | ||||
|         height: 20px; | ||||
|         background-color: #f0f0f0; | ||||
|       } | ||||
|  | ||||
|       .vertical-header { | ||||
|         text-align: center; | ||||
|       } | ||||
|  | ||||
|       .collapsible { | ||||
|         background-color: #f9f9f9; | ||||
|         color: #444; | ||||
|         cursor: pointer; | ||||
|         padding: 18px; | ||||
|         width: 100%; | ||||
|         border: none; | ||||
|         text-align: left; | ||||
|         outline: none; | ||||
|         font-size: 15px; | ||||
|       } | ||||
|  | ||||
|       .active, | ||||
|       .collapsible:hover { | ||||
|         background-color: #ccc; | ||||
|       } | ||||
|  | ||||
|       .collapsible:after { | ||||
|         content: '\002B'; | ||||
|         color: #777; | ||||
|         font-weight: bold; | ||||
|         float: right; | ||||
|         margin-left: 2px; | ||||
|       } | ||||
|  | ||||
|       .active:after { | ||||
|         content: '\2212'; | ||||
|       } | ||||
|  | ||||
|       .content { | ||||
|         padding: 0 5px; | ||||
|         max-height: 0; | ||||
|         overflow: hidden; | ||||
|         transition: max-height 0.2s ease-out; | ||||
|         background-color: #f1f1f1; | ||||
|       } | ||||
|  | ||||
|       .content .pre-scrollable { | ||||
|         max-height: 200px; | ||||
|         overflow-y: scroll; | ||||
|       } | ||||
|     </style> | ||||
|   </head> | ||||
|  | ||||
|   <body> | ||||
|     <table> | ||||
|       <tr> | ||||
|         <th></th> | ||||
|         <!-- Placeholder for the top left corner --> | ||||
|         <th>Dagre</th> | ||||
|         <th>Dagre with rough</th> | ||||
|         <th>ELK</th> | ||||
|         <th>ELK with rough</th> | ||||
|       </tr> | ||||
|       <tr> | ||||
|         <th class="vertical-header"> | ||||
|           <button class="collapsible">Stadium shape</button> | ||||
|           <div class="content"> | ||||
|             <div class="pre-scrollable"> | ||||
|               <pre> | ||||
|               flowchart LR | ||||
|               id1([This is the text in the box]) | ||||
|             </pre | ||||
|               > | ||||
|             </div> | ||||
|           </div> | ||||
|         </th> | ||||
|         <td> | ||||
|           <pre id="diagram1" class="mermaid"> | ||||
| flowchart LR | ||||
|     id1([This is the text in the box]) | ||||
|  | ||||
|       </pre | ||||
|           > | ||||
|         </td> | ||||
|         <td> | ||||
|           <pre id="diagram2" class="mermaid"> | ||||
| %%{init: {"look": "handDrawn"} }%% | ||||
| flowchart LR | ||||
|     id1([This is the text in the box]) | ||||
|       </pre | ||||
|           > | ||||
|         </td> | ||||
|         <td> | ||||
|           <pre id="diagram3" class="mermaid"> | ||||
| %%{init: {"handDrawn": false, "layout": "elk"} }%% | ||||
| flowchart LR | ||||
|     id1([This is the text in the box]) | ||||
|       </pre | ||||
|           > | ||||
|         </td> | ||||
|         <td> | ||||
|           <pre id="diagram4" class="mermaid"> | ||||
| %%{init: {"look": "handDrawn", "layout": "elk"} }%% | ||||
| flowchart LR | ||||
|     id1([This is the text in the box]) | ||||
|       </pre | ||||
|           > | ||||
|         </td> | ||||
|       </tr> | ||||
|       <!-- Separator row --> | ||||
|       <tr class="separator"> | ||||
|         <td colspan="5"></td> | ||||
|         <!-- This cell spans all columns including the vertical header --> | ||||
|       </tr> | ||||
|  | ||||
|       <tr> | ||||
|         <th class="vertical-header"> | ||||
|           <button class="collapsible">Sub-Routine shape</button> | ||||
|           <div class="content"> | ||||
|             <div class="pre-scrollable"> | ||||
|               <pre> | ||||
|       flowchart LR | ||||
|     id1[[This is the text in the box]] | ||||
|     </pre | ||||
|               > | ||||
|             </div> | ||||
|           </div> | ||||
|         </th> | ||||
|         <td> | ||||
|           <pre id="diagram5" class="mermaid"> | ||||
| flowchart LR | ||||
|     id1[[This is the text in the box]] | ||||
|       </pre | ||||
|           > | ||||
|         </td> | ||||
|         <td> | ||||
|           <pre id="diagram6" class="mermaid"> | ||||
| %%{init: {"look": "handDrawn"} }%% | ||||
| flowchart LR | ||||
|     id1[[This is the text in the box]] | ||||
|       </pre | ||||
|           > | ||||
|         </td> | ||||
|         <td> | ||||
|           <pre id="diagram7" class="mermaid"> | ||||
| %%{init: {"handDrawn": false, "layout": "elk"} }%% | ||||
| flowchart LR | ||||
|     id1[[This is the text in the box]] | ||||
|       </pre | ||||
|           > | ||||
|         </td> | ||||
|         <td> | ||||
|           <pre id="diagram8" class="mermaid"> | ||||
| %%{init: {"look": "handDrawn", "layout": "elk"} }%% | ||||
| flowchart LR | ||||
|     id1[[This is the text in the box]] | ||||
|       </pre | ||||
|           > | ||||
|         </td> | ||||
|       </tr> | ||||
|       <!-- Separator row --> | ||||
|       <tr class="separator"> | ||||
|         <td colspan="5"></td> | ||||
|         <!-- This cell spans all columns including the vertical header --> | ||||
|       </tr> | ||||
|  | ||||
|       <tr> | ||||
|         <th class="vertical-header"> | ||||
|           <button class="collapsible">Cylindrical shape</button> | ||||
|           <div class="content"> | ||||
|             <div class="pre-scrollable"> | ||||
|               <pre> | ||||
|               flowchart LR | ||||
|     id1[(Database)] | ||||
|     </pre | ||||
|               > | ||||
|             </div> | ||||
|           </div> | ||||
|         </th> | ||||
|         <td> | ||||
|           <pre id="diagram9" class="mermaid"> | ||||
|           flowchart LR | ||||
|     id1[(Database)] | ||||
|       </pre | ||||
|           > | ||||
|         </td> | ||||
|         <td> | ||||
|           <pre id="diagram10" class="mermaid"> | ||||
|           %%{init: {"look": "handDrawn"} }%% | ||||
|           flowchart LR | ||||
|     id1[(Database)] | ||||
|       </pre | ||||
|           > | ||||
|         </td> | ||||
|         <td> | ||||
|           <pre id="diagram11" class="mermaid"> | ||||
|           %%{init: {"handDrawn": false, "layout": "elk"} }%% | ||||
|           flowchart LR | ||||
|     id1[(Database)] | ||||
|       </pre | ||||
|           > | ||||
|         </td> | ||||
|         <td> | ||||
|           <pre id="diagram12" class="mermaid"> | ||||
|           %%{init: {"look": "handDrawn", "layout": "elk"} }%% | ||||
|           flowchart LR | ||||
|     id1[(Database)] | ||||
|       </pre | ||||
|           > | ||||
|         </td> | ||||
|       </tr> | ||||
|       <!-- Separator row --> | ||||
|       <tr class="separator"> | ||||
|         <td colspan="5"></td> | ||||
|         <!-- This cell spans all columns including the vertical header --> | ||||
|       </tr> | ||||
|  | ||||
|       <tr> | ||||
|         <th class="vertical-header"> | ||||
|           <button class="collapsible">Circle shape</button> | ||||
|           <div class="content"> | ||||
|             <div class="pre-scrollable"> | ||||
|               <pre> | ||||
|               flowchart LR | ||||
|     id1((This is the text in the circle)) | ||||
|     </pre | ||||
|               > | ||||
|             </div> | ||||
|           </div> | ||||
|         </th> | ||||
|         <td> | ||||
|           <pre id="diagram13" class="mermaid"> | ||||
|           flowchart LR | ||||
|     id1((This is the text in the circle)) | ||||
|       </pre | ||||
|           > | ||||
|         </td> | ||||
|         <td> | ||||
|           <pre id="diagram14" class="mermaid"> | ||||
|           %%{init: {"look": "handDrawn"} }%% | ||||
|           flowchart LR | ||||
|     id1((This is the text in the circle)) | ||||
|       </pre | ||||
|           > | ||||
|         </td> | ||||
|         <td> | ||||
|           <pre id="diagram15" class="mermaid"> | ||||
|           %%{init: {"handDrawn": false, "layout": "elk"} }%% | ||||
|           flowchart LR | ||||
|     id1((This is the text in the circle)) | ||||
|       </pre | ||||
|           > | ||||
|         </td> | ||||
|         <td> | ||||
|           <pre id="diagram16" class="mermaid"> | ||||
|           %%{init: {"look": "handDrawn", "layout": "elk"} }%% | ||||
|           flowchart LR | ||||
|     id1((This is the text in the circle)) | ||||
|       </pre | ||||
|           > | ||||
|         </td> | ||||
|       </tr> | ||||
|       <!-- Separator row --> | ||||
|       <tr class="separator"> | ||||
|         <td colspan="5"></td> | ||||
|         <!-- This cell spans all columns including the vertical header --> | ||||
|       </tr> | ||||
|  | ||||
|       <tr> | ||||
|         <th class="vertical-header"> | ||||
|           <button class="collapsible">Double Circle shape</button> | ||||
|           <div class="content"> | ||||
|             <div class="pre-scrollable"> | ||||
|               <pre> | ||||
|               flowchart TD | ||||
|     id1(((This is the text in the circle))) | ||||
|     </pre | ||||
|               > | ||||
|             </div> | ||||
|           </div> | ||||
|         </th> | ||||
|         <td> | ||||
|           <pre id="diagram17" class="mermaid"> | ||||
|           flowchart TD | ||||
|     id1(((This is the text in the circle))) | ||||
|       </pre | ||||
|           > | ||||
|         </td> | ||||
|         <td> | ||||
|           <pre id="diagram18" class="mermaid"> | ||||
|           %%{init: {"look": "handDrawn"} }%% | ||||
|           flowchart TD | ||||
|     id1(((This is the text in the circle))) | ||||
|       </pre | ||||
|           > | ||||
|         </td> | ||||
|         <td> | ||||
|           <pre id="diagram19" class="mermaid"> | ||||
|           %%{init: {"handDrawn": false, "layout": "elk"} }%% | ||||
|           flowchart TD | ||||
|     id1(((This is the text in the circle))) | ||||
|       </pre | ||||
|           > | ||||
|         </td> | ||||
|         <td> | ||||
|           <pre id="diagram20" class="mermaid"> | ||||
|           %%{init: {"look": "handDrawn", "layout": "elk"} }%% | ||||
|           flowchart TD | ||||
|     id1(((This is the text in the circle))) | ||||
|       </pre | ||||
|           > | ||||
|         </td> | ||||
|       </tr> | ||||
|       <!-- Separator row --> | ||||
|       <tr class="separator"> | ||||
|         <td colspan="5"></td> | ||||
|         <!-- This cell spans all columns including the vertical header --> | ||||
|       </tr> | ||||
|  | ||||
|       <tr> | ||||
|         <th class="vertical-header"> | ||||
|           <button class="collapsible">Asymmetric shape</button> | ||||
|           <div class="content"> | ||||
|             <div class="pre-scrollable"> | ||||
|               <pre> | ||||
|               flowchart LR | ||||
|     id1>This is the text in the box] | ||||
|     </pre | ||||
|               > | ||||
|             </div> | ||||
|           </div> | ||||
|         </th> | ||||
|         <td> | ||||
|           <pre id="diagram21" class="mermaid"> | ||||
|           flowchart LR | ||||
|     id1>This is the text in the box] | ||||
|       </pre | ||||
|           > | ||||
|         </td> | ||||
|         <td> | ||||
|           <pre id="diagram22" class="mermaid"> | ||||
|           %%{init: {"look": "handDrawn"} }%% | ||||
|           flowchart LR | ||||
|     id1>This is the text in the box]   | ||||
|       </pre | ||||
|           > | ||||
|         </td> | ||||
|         <td> | ||||
|           <pre id="diagram23" class="mermaid"> | ||||
|           %%{init: {"handDrawn": false, "layout": "elk"} }%% | ||||
|           flowchart LR | ||||
|     id1>This is the text in the box]   | ||||
|       </pre | ||||
|           > | ||||
|         </td> | ||||
|         <td> | ||||
|           <pre id="diagram24" class="mermaid"> | ||||
|           %%{init: {"look": "handDrawn", "layout": "elk"} }%% | ||||
|           flowchart LR | ||||
|     id1>This is the text in the box] | ||||
|       </pre | ||||
|           > | ||||
|         </td> | ||||
|       </tr> | ||||
|       <!-- Separator row --> | ||||
|       <tr class="separator"> | ||||
|         <td colspan="5"></td> | ||||
|         <!-- This cell spans all columns including the vertical header --> | ||||
|       </tr> | ||||
|  | ||||
|       <tr> | ||||
|         <th class="vertical-header"> | ||||
|           <button class="collapsible">Rhombus/Diamond/Question shape</button> | ||||
|           <div class="content"> | ||||
|             <div class="pre-scrollable"> | ||||
|               <pre> | ||||
|               flowchart LR | ||||
|     id1{This is the text in the box} | ||||
|     </pre | ||||
|               > | ||||
|             </div> | ||||
|           </div> | ||||
|         </th> | ||||
|         <td> | ||||
|           <pre id="diagram25" class="mermaid"> | ||||
|           flowchart LR | ||||
|     id1{This is the text in the box} | ||||
|       </pre | ||||
|           > | ||||
|         </td> | ||||
|         <td> | ||||
|           <pre id="diagram26" class="mermaid"> | ||||
|           %%{init: {"look": "handDrawn"} }%% | ||||
|           flowchart LR | ||||
|     id1{This is the text in the box} | ||||
|       </pre | ||||
|           > | ||||
|         </td> | ||||
|         <td> | ||||
|           <pre id="diagram27" class="mermaid"> | ||||
|           %%{init: {"handDrawn": false, "layout": "elk"} }%% | ||||
|           flowchart LR | ||||
|     id1{This is the text in the box} | ||||
|       </pre | ||||
|           > | ||||
|         </td> | ||||
|         <td> | ||||
|           <pre id="diagram28" class="mermaid"> | ||||
|           %%{init: {"look": "handDrawn", "layout": "elk"} }%% | ||||
|           flowchart LR | ||||
|     id1{This is the text in the box} | ||||
|       </pre | ||||
|           > | ||||
|         </td> | ||||
|       </tr> | ||||
|       <!-- Separator row --> | ||||
|       <tr class="separator"> | ||||
|         <td colspan="5"></td> | ||||
|         <!-- This cell spans all columns including the vertical header --> | ||||
|       </tr> | ||||
|  | ||||
|       <tr> | ||||
|         <th class="vertical-header"> | ||||
|           <button class="collapsible">Hexagon shape</button> | ||||
|           <div class="content"> | ||||
|             <div class="pre-scrollable"> | ||||
|               <pre> | ||||
|               flowchart LR | ||||
|     id1{{This is the text in the box}} | ||||
|     </pre | ||||
|               > | ||||
|             </div> | ||||
|           </div> | ||||
|         </th> | ||||
|         <td> | ||||
|           <pre id="diagram29" class="mermaid"> | ||||
|           flowchart LR | ||||
|     id1{{This is the text in the box}} | ||||
|       </pre | ||||
|           > | ||||
|         </td> | ||||
|         <td> | ||||
|           <pre id="diagram31" class="mermaid"> | ||||
|           %%{init: {"handDrawn": false, "layout": "elk"} }%% | ||||
|           flowchart LR | ||||
|     id1{{This is the text in the box}} | ||||
|       </pre | ||||
|           > | ||||
|         </td> | ||||
|         <td> | ||||
|           <pre id="diagram32" class="mermaid"> | ||||
|           %%{init: {"look": "handDrawn", "layout": "elk"} }%% | ||||
|           flowchart LR | ||||
|     id1{{This is the text in the box}} | ||||
|       </pre | ||||
|           > | ||||
|         </td> | ||||
|       </tr> | ||||
|       <!-- Separator row --> | ||||
|       <tr class="separator"> | ||||
|         <td colspan="5"></td> | ||||
|         <!-- This cell spans all columns including the vertical header --> | ||||
|       </tr> | ||||
|  | ||||
|       <tr> | ||||
|         <th class="vertical-header"> | ||||
|           <button class="collapsible">Parallelogram shape</button> | ||||
|           <div class="content"> | ||||
|             <div class="pre-scrollable"> | ||||
|               <pre> | ||||
|               flowchart TD | ||||
|     id1[/This is the text in the box/] | ||||
|     </pre | ||||
|               > | ||||
|             </div> | ||||
|           </div> | ||||
|         </th> | ||||
|         <td> | ||||
|           <pre id="diagram33" class="mermaid"> | ||||
|           flowchart TD | ||||
|     id1[/This is the text in the box/] | ||||
|       </pre | ||||
|           > | ||||
|         </td> | ||||
|         <td> | ||||
|           <pre id="diagram34" class="mermaid"> | ||||
|           %%{init: {"look": "handDrawn"} }%% | ||||
|           flowchart TD | ||||
|     id1[/This is the text in the box/]   | ||||
|       </pre | ||||
|           > | ||||
|         </td> | ||||
|         <td> | ||||
|           <pre id="diagram35" class="mermaid"> | ||||
|           %%{init: {"handDrawn": false, "layout": "elk"} }%% | ||||
|           flowchart TD | ||||
|     id1[/This is the text in the box/]  | ||||
|       </pre | ||||
|           > | ||||
|         </td> | ||||
|         <td> | ||||
|           <pre id="diagram36" class="mermaid"> | ||||
|           %%{init: {"look": "handDrawn", "layout": "elk"} }%% | ||||
|           flowchart TD | ||||
|     id1[/This is the text in the box/] | ||||
|       </pre | ||||
|           > | ||||
|         </td> | ||||
|       </tr> | ||||
|       <!-- Separator row --> | ||||
|       <tr class="separator"> | ||||
|         <td colspan="5"></td> | ||||
|         <!-- This cell spans all columns including the vertical header --> | ||||
|       </tr> | ||||
|  | ||||
|       <tr> | ||||
|         <th class="vertical-header"> | ||||
|           <button class="collapsible">Parallelogram Alt shape</button> | ||||
|           <div class="content"> | ||||
|             <div class="pre-scrollable"> | ||||
|               <pre> | ||||
|               flowchart TD | ||||
|     id1[\This is the text in the box\] | ||||
|     </pre | ||||
|               > | ||||
|             </div> | ||||
|           </div> | ||||
|         </th> | ||||
|         <td> | ||||
|           <pre id="diagram37" class="mermaid"> | ||||
|           flowchart TD | ||||
|     id1[\This is the text in the box\] | ||||
|       </pre | ||||
|           > | ||||
|         </td> | ||||
|         <td> | ||||
|           <pre id="diagram38" class="mermaid"> | ||||
|           %%{init: {"look": "handDrawn"} }%% | ||||
|           flowchart TD | ||||
|     id1[\This is the text in the box\] | ||||
|       </pre | ||||
|           > | ||||
|         </td> | ||||
|         <td> | ||||
|           <pre id="diagram39" class="mermaid"> | ||||
|           %%{init: {"handDrawn": false, "layout": "elk"} }%% | ||||
|           flowchart TD | ||||
|     id1[\This is the text in the box\] | ||||
|       </pre | ||||
|           > | ||||
|         </td> | ||||
|         <td> | ||||
|           <pre id="diagram40" class="mermaid"> | ||||
|           %%{init: {"look": "handDrawn", "layout": "elk"} }%% | ||||
|           flowchart TD | ||||
|     id1[\This is the text in the box\] | ||||
|  | ||||
|       </pre | ||||
|           > | ||||
|         </td> | ||||
|       </tr> | ||||
|       <!-- Separator row --> | ||||
|       <tr class="separator"> | ||||
|         <td colspan="5"></td> | ||||
|         <!-- This cell spans all columns including the vertical header --> | ||||
|       </tr> | ||||
|  | ||||
|       <tr> | ||||
|         <th class="vertical-header"> | ||||
|           <button class="collapsible">Trapezoid shape</button> | ||||
|           <div class="content"> | ||||
|             <div class="pre-scrollable"> | ||||
|               <pre> | ||||
|               flowchart TD | ||||
|     A[/Christmas\] | ||||
|     </pre | ||||
|               > | ||||
|             </div> | ||||
|           </div> | ||||
|         </th> | ||||
|         <td> | ||||
|           <pre id="diagram41" class="mermaid"> | ||||
|           flowchart TD | ||||
|     A[/Christmas\] | ||||
|       </pre | ||||
|           > | ||||
|         </td> | ||||
|         <td> | ||||
|           <pre id="diagram42" class="mermaid"> | ||||
|           %%{init: {"look": "handDrawn"} }%% | ||||
|           flowchart TD | ||||
|     A[/Christmas\] | ||||
|       </pre | ||||
|           > | ||||
|         </td> | ||||
|         <td> | ||||
|           <pre id="diagram43" class="mermaid"> | ||||
|           %%{init: {"handDrawn": false, "layout": "elk"} }%% | ||||
|           flowchart TD | ||||
|     A[/Christmas\] | ||||
|       </pre | ||||
|           > | ||||
|         </td> | ||||
|         <td> | ||||
|           <pre id="diagram44" class="mermaid"> | ||||
|           %%{init: {"look": "handDrawn", "layout": "elk"} }%% | ||||
|           flowchart TD | ||||
|     A[/Christmas\] | ||||
|       </pre | ||||
|           > | ||||
|         </td> | ||||
|       </tr> | ||||
|       <!-- Separator row --> | ||||
|       <tr class="separator"> | ||||
|         <td colspan="5"></td> | ||||
|         <!-- This cell spans all columns including the vertical header --> | ||||
|       </tr> | ||||
|  | ||||
|       <tr> | ||||
|         <th class="vertical-header"> | ||||
|           <button class="collapsible">Trapezoid Alt shape</button> | ||||
|           <div class="content"> | ||||
|             <div class="pre-scrollable"> | ||||
|               <pre> | ||||
|               flowchart TD | ||||
|     A[\Christmas/] | ||||
|     </pre | ||||
|               > | ||||
|             </div> | ||||
|           </div> | ||||
|         </th> | ||||
|         <td> | ||||
|           <pre id="diagram45" class="mermaid"> | ||||
|           flowchart TD | ||||
|     A[\Christmas/] | ||||
|       </pre | ||||
|           > | ||||
|         </td> | ||||
|         <td> | ||||
|           <pre id="diagram46" class="mermaid"> | ||||
|           %%{init: {"look": "handDrawn"} }%% | ||||
|           flowchart TD | ||||
|     A[\Christmas/] | ||||
|       </pre | ||||
|           > | ||||
|         </td> | ||||
|         <td> | ||||
|           <pre id="diagram47" class="mermaid"> | ||||
|           %%{init: {"handDrawn": false, "layout": "elk"} }%% | ||||
|           flowchart TD | ||||
|     A[\Christmas/] | ||||
|       </pre | ||||
|           > | ||||
|         </td> | ||||
|         <td> | ||||
|           <pre id="diagram48" class="mermaid"> | ||||
|           %%{init: {"look": "handDrawn", "layout": "elk"} }%% | ||||
|           flowchart TD | ||||
|     A[\Christmas/]   | ||||
|       </pre | ||||
|           > | ||||
|         </td> | ||||
|       </tr> | ||||
|       <!-- Separator row --> | ||||
|       <tr class="separator"> | ||||
|         <td colspan="5"></td> | ||||
|         <!-- This cell spans all columns including the vertical header --> | ||||
|       </tr> | ||||
|  | ||||
|       <tr> | ||||
|         <th class="vertical-header"> | ||||
|           <button class="collapsible">Rect with rounded corner</button> | ||||
|           <div class="content"> | ||||
|             <div class="pre-scrollable"> | ||||
|               <pre> | ||||
|               flowchart LR | ||||
|     id1(This is the text in the box) | ||||
|     </pre | ||||
|               > | ||||
|             </div> | ||||
|           </div> | ||||
|         </th> | ||||
|         <td> | ||||
|           <pre id="diagram49" class="mermaid"> | ||||
|           flowchart LR | ||||
|     id1(This is the text in the box) | ||||
|       </pre | ||||
|           > | ||||
|         </td> | ||||
|         <td> | ||||
|           <pre id="diagram50" class="mermaid"> | ||||
|           %%{init: {"look": "handDrawn"} }%% | ||||
|           flowchart LR | ||||
|     id1(This is the text in the box) | ||||
|       </pre | ||||
|           > | ||||
|         </td> | ||||
|         <td> | ||||
|           <pre id="diagram51" class="mermaid"> | ||||
|           %%{init: {"handDrawn": false, "layout": "elk"} }%% | ||||
|           flowchart LR | ||||
|     id1(This is the text in the box) | ||||
|       </pre | ||||
|           > | ||||
|         </td> | ||||
|         <td> | ||||
|           <pre id="diagram52" class="mermaid"> | ||||
|           %%{init: {"look": "handDrawn", "layout": "elk"} }%% | ||||
|           flowchart LR | ||||
|     id1(This is the text in the box)  | ||||
|       </pre | ||||
|           > | ||||
|         </td> | ||||
|       </tr> | ||||
|       <!-- Separator row --> | ||||
|       <tr class="separator"> | ||||
|         <td colspan="5"></td> | ||||
|         <!-- This cell spans all columns including the vertical header --> | ||||
|       </tr> | ||||
|  | ||||
|       <tr> | ||||
|         <th class="vertical-header"> | ||||
|           <button class="collapsible">Rect with sharp corner</button> | ||||
|           <div class="content"> | ||||
|             <div class="pre-scrollable"> | ||||
|               <pre> | ||||
|               flowchart LR | ||||
|     id1[This is the text in the box] | ||||
|     </pre | ||||
|               > | ||||
|             </div> | ||||
|           </div> | ||||
|         </th> | ||||
|         <td> | ||||
|           <pre id="diagram53" class="mermaid"> | ||||
|           flowchart LR | ||||
|     id1[This is the text in the box] | ||||
|       </pre | ||||
|           > | ||||
|         </td> | ||||
|         <td> | ||||
|           <pre id="diagram54" class="mermaid"> | ||||
|           %%{init: {"look": "handDrawn"} }%% | ||||
|           flowchart LR | ||||
|     id1[This is the text in the box] | ||||
|       </pre | ||||
|           > | ||||
|         </td> | ||||
|         <td> | ||||
|           <pre id="diagram55" class="mermaid"> | ||||
|           %%{init: {"handDrawn": false, "layout": "elk"} }%% | ||||
|           flowchart LR | ||||
|     id1[This is the text in the box] | ||||
|       </pre | ||||
|           > | ||||
|         </td> | ||||
|         <td> | ||||
|           <pre id="diagram56" class="mermaid"> | ||||
|           %%{init: {"look": "handDrawn", "layout": "elk"} }%% | ||||
|           flowchart LR | ||||
|     id1[This is the text in the box] | ||||
|       </pre | ||||
|           > | ||||
|         </td> | ||||
|       </tr> | ||||
|       <!-- Separator row --> | ||||
|       <tr class="separator"> | ||||
|         <td colspan="5"></td> | ||||
|         <!-- This cell spans all columns including the vertical header --> | ||||
|       </tr> | ||||
|     </table> | ||||
|  | ||||
|     <script type="module"> | ||||
|       import mermaid from './mermaid.esm.mjs'; | ||||
|       import layouts from './mermaid-layout-elk.esm.mjs'; | ||||
|       mermaid.registerLayoutLoaders(layouts); | ||||
|       mermaid.parseError = function (err, hash) {}; | ||||
|  | ||||
|       mermaid.initialize({ | ||||
|         handDrawn: false, | ||||
|         mergeEdges: true, | ||||
|         layout: 'dagre', | ||||
|         flowchart: { titleTopMargin: 10 }, | ||||
|         // fontFamily: 'Caveat', | ||||
|         fontFamily: 'Kalam', | ||||
|         sequence: { | ||||
|           actorFontFamily: 'courier', | ||||
|           noteFontFamily: 'courier', | ||||
|           messageFontFamily: 'courier', | ||||
|         }, | ||||
|         fontSize: 16, | ||||
|         logLevel: 0, | ||||
|       }); | ||||
|       function callback() { | ||||
|         alert('It worked'); | ||||
|       } | ||||
|       mermaid.parseError = function (err, hash) { | ||||
|         console.error('In parse error:'); | ||||
|         console.error(err); | ||||
|       }; | ||||
|  | ||||
|       let coll = document.getElementsByClassName('collapsible'); | ||||
|       for (const element of coll) { | ||||
|         element.addEventListener('click', function () { | ||||
|           this.classList.toggle('active'); | ||||
|           let content = this.nextElementSibling; | ||||
|           if (content.style.maxHeight) { | ||||
|             content.style.maxHeight = null; | ||||
|           } else { | ||||
|             content.style.maxHeight = content.scrollHeight + 'px'; | ||||
|           } | ||||
|         }); | ||||
|       } | ||||
|     </script> | ||||
|   </body> | ||||
| </html> | ||||
| @@ -1,191 +0,0 @@ | ||||
| <html> | ||||
|   <head> | ||||
|     <link href="https://fonts.googleapis.com/css?family=Montserrat&display=swap" rel="stylesheet" /> | ||||
|     <link href="https://unpkg.com/tailwindcss@^1.0/dist/tailwind.min.css" rel="stylesheet" /> | ||||
|     <link | ||||
|       rel="stylesheet" | ||||
|       href="https://cdnjs.cloudflare.com/ajax/libs/font-awesome/4.7.0/css/font-awesome.min.css" | ||||
|     /> | ||||
|     <link | ||||
|       href="https://cdn.jsdelivr.net/npm/@mdi/font@6.9.96/css/materialdesignicons.min.css" | ||||
|       rel="stylesheet" | ||||
|     /> | ||||
|     <link | ||||
|       href="https://fonts.googleapis.com/css?family=Noto+Sans+SC&display=swap" | ||||
|       rel="stylesheet" | ||||
|     /> | ||||
|     <link rel="preconnect" href="https://fonts.googleapis.com" /> | ||||
|     <link rel="preconnect" href="https://fonts.gstatic.com" crossorigin /> | ||||
|     <link | ||||
|       href="https://fonts.googleapis.com/css2?family=Kalam:wght@300;400;700&display=swap" | ||||
|       rel="stylesheet" | ||||
|     /> | ||||
|     <link | ||||
|       href="https://fonts.googleapis.com/css2?family=Caveat:wght@400..700&family=Kalam:wght@300;400;700&family=Rubik+Mono+One&display=swap" | ||||
|       rel="stylesheet" | ||||
|     /> | ||||
|     <link | ||||
|       href="https://fonts.googleapis.com/css2?family=Kalam:wght@300;400;700&family=Rubik+Mono+One&display=swap" | ||||
|       rel="stylesheet" | ||||
|     /> | ||||
|  | ||||
|     <style> | ||||
|       body { | ||||
|         font-family: 'Arial'; | ||||
|       } | ||||
|  | ||||
|       table { | ||||
|         width: 100%; | ||||
|         border-collapse: collapse; | ||||
|         table-layout: fixed; | ||||
|       } | ||||
|  | ||||
|       th, | ||||
|       td { | ||||
|         border: 1px solid black; | ||||
|         padding: 10px; | ||||
|         text-align: center; | ||||
|         vertical-align: middle; | ||||
|       } | ||||
|  | ||||
|       .separator { | ||||
|         height: 20px; | ||||
|         background-color: #f0f0f0; | ||||
|       } | ||||
|  | ||||
|       .vertical-header { | ||||
|         text-align: center; | ||||
|       } | ||||
|  | ||||
|       .collapsible { | ||||
|         background-color: #f9f9f9; | ||||
|         color: #444; | ||||
|         cursor: pointer; | ||||
|         padding: 18px; | ||||
|         width: 100%; | ||||
|         border: none; | ||||
|         text-align: left; | ||||
|         outline: none; | ||||
|         font-size: 15px; | ||||
|       } | ||||
|  | ||||
|       .active, | ||||
|       .collapsible:hover { | ||||
|         background-color: #ccc; | ||||
|       } | ||||
|  | ||||
|       .collapsible:after { | ||||
|         content: '\002B'; | ||||
|         color: #777; | ||||
|         font-weight: bold; | ||||
|         float: right; | ||||
|         margin-left: 2px; | ||||
|       } | ||||
|  | ||||
|       .active:after { | ||||
|         content: '\2212'; | ||||
|       } | ||||
|  | ||||
|       .content { | ||||
|         padding: 0 5px; | ||||
|         max-height: 0; | ||||
|         overflow: hidden; | ||||
|         transition: max-height 0.2s ease-out; | ||||
|         background-color: #f1f1f1; | ||||
|       } | ||||
|  | ||||
|       .content .pre-scrollable { | ||||
|         max-height: 200px; | ||||
|         overflow-y: scroll; | ||||
|       } | ||||
|     </style> | ||||
|   </head> | ||||
|  | ||||
|   <body> | ||||
|     <table> | ||||
|       <tr> | ||||
|         <th></th> | ||||
|         <!-- Placeholder for the top left corner --> | ||||
|         <th>State rough</th> | ||||
|         <th>Flowchart rough</th> | ||||
|       </tr> | ||||
|       <tr> | ||||
|         <th class="vertical-header"> | ||||
|           <button class="collapsible">Stadium shape</button> | ||||
|           <div class="content"> | ||||
|             <div class="pre-scrollable"> | ||||
|               <pre> | ||||
|       flowchart LR | ||||
|     id1([This is the text in the box]) | ||||
|  | ||||
|   </pre | ||||
|               > | ||||
|             </div> | ||||
|           </div> | ||||
|         </th> | ||||
|         <td> | ||||
|           <pre id="diagram1" class="mermaid"> | ||||
| %%{init: {"look": "handDrawn"} }%% | ||||
| stateDiagram-v2 | ||||
|     stateA | ||||
|  | ||||
|       </pre | ||||
|           > | ||||
|         </td> | ||||
|         <td> | ||||
|           <pre id="diagram2" class="mermaid"> | ||||
| %%{init: {"look": "handDrawn"} }%% | ||||
| flowchart LR | ||||
|     id1[[This is the text in the box]] | ||||
|  | ||||
|  | ||||
|       </pre | ||||
|           > | ||||
|         </td> | ||||
|       </tr> | ||||
|     </table> | ||||
|  | ||||
|     <script type="module"> | ||||
|       import mermaid from './mermaid.esm.mjs'; | ||||
|       import layouts from './mermaid-layout-elk.esm.mjs'; | ||||
|       mermaid.registerLayoutLoaders(layouts); | ||||
|       mermaid.parseError = function (err, hash) {}; | ||||
|  | ||||
|       mermaid.initialize({ | ||||
|         handDrawn: false, | ||||
|         mergeEdges: true, | ||||
|         layout: 'dagre', | ||||
|         flowchart: { titleTopMargin: 10 }, | ||||
|         // fontFamily: 'Caveat', | ||||
|         fontFamily: 'Kalam', | ||||
|         sequence: { | ||||
|           actorFontFamily: 'courier', | ||||
|           noteFontFamily: 'courier', | ||||
|           messageFontFamily: 'courier', | ||||
|         }, | ||||
|         fontSize: 16, | ||||
|         logLevel: 0, | ||||
|       }); | ||||
|       function callback() { | ||||
|         alert('It worked'); | ||||
|       } | ||||
|       mermaid.parseError = function (err, hash) { | ||||
|         console.error('In parse error:'); | ||||
|         console.error(err); | ||||
|       }; | ||||
|  | ||||
|       let coll = document.getElementsByClassName('collapsible'); | ||||
|       for (const element of coll) { | ||||
|         element.addEventListener('click', function () { | ||||
|           this.classList.toggle('active'); | ||||
|           let content = this.nextElementSibling; | ||||
|           if (content.style.maxHeight) { | ||||
|             content.style.maxHeight = null; | ||||
|           } else { | ||||
|             content.style.maxHeight = content.scrollHeight + 'px'; | ||||
|           } | ||||
|         }); | ||||
|       } | ||||
|     </script> | ||||
|   </body> | ||||
| </html> | ||||
| @@ -1,433 +0,0 @@ | ||||
| <html> | ||||
|   <head> | ||||
|     <link href="https://fonts.googleapis.com/css?family=Montserrat&display=swap" rel="stylesheet" /> | ||||
|     <link href="https://unpkg.com/tailwindcss@^1.0/dist/tailwind.min.css" rel="stylesheet" /> | ||||
|     <link | ||||
|       rel="stylesheet" | ||||
|       href="https://cdnjs.cloudflare.com/ajax/libs/font-awesome/4.7.0/css/font-awesome.min.css" | ||||
|     /> | ||||
|     <link | ||||
|       href="https://cdn.jsdelivr.net/npm/@mdi/font@6.9.96/css/materialdesignicons.min.css" | ||||
|       rel="stylesheet" | ||||
|     /> | ||||
|     <link | ||||
|       href="https://fonts.googleapis.com/css?family=Noto+Sans+SC&display=swap" | ||||
|       rel="stylesheet" | ||||
|     /> | ||||
|     <style> | ||||
|       body { | ||||
|         /* background: rgb(221, 208, 208); */ | ||||
|         /* background:#333; */ | ||||
|         font-family: 'Arial'; | ||||
|         /* font-size: 18px !important; */ | ||||
|       } | ||||
|       h1 { | ||||
|         color: grey; | ||||
|       } | ||||
|       .mermaid2 { | ||||
|         display: none; | ||||
|       } | ||||
|       .mermaid svg { | ||||
|         /* font-size: 18px !important; */ | ||||
|         background-color: #efefef; | ||||
|         background-image: radial-gradient(#fff 51%, transparent 91%), | ||||
|           radial-gradient(#fff 51%, transparent 91%); | ||||
|         background-size: 20px 20px; | ||||
|         background-position: | ||||
|           0 0, | ||||
|           10px 10px; | ||||
|         background-repeat: repeat; | ||||
|       } | ||||
|       .malware { | ||||
|         position: fixed; | ||||
|         bottom: 0; | ||||
|         left: 0; | ||||
|         right: 0; | ||||
|         height: 150px; | ||||
|         background: red; | ||||
|         color: black; | ||||
|         display: flex; | ||||
|         display: flex; | ||||
|         justify-content: center; | ||||
|         align-items: center; | ||||
|         font-family: monospace; | ||||
|         font-size: 72px; | ||||
|       } | ||||
|       /* tspan { | ||||
|         font-size: 6px !important; | ||||
|       } */ | ||||
|     </style> | ||||
|   </head> | ||||
|   <body> | ||||
|     <pre id="diagram" class="mermaid"> | ||||
| stateDiagram-v2 | ||||
|     [*] --> Still | ||||
|     Still --> [*] | ||||
|     Still --> Moving | ||||
|     Moving --> Still | ||||
|     Moving --> Crash | ||||
|     Crash --> [*]    </pre | ||||
|     > | ||||
|     <pre id="diagram" class="mermaid2"> | ||||
| flowchart RL | ||||
|     subgraph "`one`" | ||||
|       a1 -- l1 --> a2 | ||||
|       a1 -- l2 --> a2 | ||||
|     end | ||||
|     </pre> | ||||
|     <pre id="diagram" class="mermaid"> | ||||
| flowchart RL | ||||
|     subgraph "`one`" | ||||
|       a1 -- l1 --> a2 | ||||
|       a1 -- l2 --> a2 | ||||
|     end | ||||
|     </pre> | ||||
|     <pre id="diagram" class="mermaid2"> | ||||
| flowchart | ||||
| id["`A root with a long text that wraps to keep the node size in check. A root with a long text that wraps to keep the node size in check`"]</pre | ||||
|     > | ||||
|     <pre id="diagram" class="mermaid2"> | ||||
| flowchart LR | ||||
|     A[A text that needs to be wrapped wraps to another line] | ||||
|     B[A text that needs to be<br/>wrapped wraps to another line] | ||||
|     C["`A text that needs to be wrapped to another line`"]</pre> | ||||
|     <pre id="diagram" class="mermaid2"> | ||||
| flowchart LR | ||||
|     C["`A text | ||||
|         that needs | ||||
|         to be wrapped | ||||
|         in another | ||||
|         way`"] | ||||
|   </pre | ||||
|     > | ||||
|     <pre id="diagram" class="mermaid"> | ||||
|       classDiagram-v2 | ||||
|         note "I love this diagram!\nDo you love it?" | ||||
|     </pre> | ||||
|     <pre id="diagram" class="mermaid"> | ||||
|     stateDiagram-v2 | ||||
|     State1: The state with a note with minus - and plus + in it | ||||
|     note left of State1 | ||||
|       Important information! You can write | ||||
|       notes with . and  in them. | ||||
|     end note    </pre | ||||
|     > | ||||
|     <pre id="diagram" class="mermaid2"> | ||||
| mindmap | ||||
| root | ||||
|   Child3(A node with an icon and with a long text that wraps to keep the node size in check) | ||||
| </pre | ||||
|     > | ||||
|     <pre id="diagram" class="mermaid2"> | ||||
|       %%{init: {"theme": "forest"} }%% | ||||
| mindmap | ||||
|     id1[**Start2**<br/>end] | ||||
|       id2[**Start2**<br />end] | ||||
|       %% Another comment | ||||
|       id3[**Start2**<br>end] %% Comment | ||||
|       id4[**Start2**<br >end<br    >the very end] | ||||
|     </pre> | ||||
|     <pre id="diagram" class="mermaid2"> | ||||
| mindmap | ||||
|     id1["`**Start2** | ||||
|     second line 😎 with long text that is wrapping to the next line`"] | ||||
|       id2["`Child **with bold** text`"] | ||||
|       id3["`Children of which some | ||||
|       is using *italic type of* text`"] | ||||
|       id4[Child] | ||||
|       id5["`Child | ||||
|       Row | ||||
|       and another | ||||
|       `"] | ||||
|     </pre> | ||||
|     <pre id="diagram" class="mermaid2"> | ||||
| mindmap | ||||
|     id1("`**Root**`"] | ||||
|       id2["`A formatted text... with **bold** and *italics*`"] | ||||
|       id3[Regular labels works as usual] | ||||
|       id4["`Emojis and unicode works too: 🤓 | ||||
|       शान्तिः سلام  和平 `"] | ||||
|  | ||||
|     </pre> | ||||
|     <pre id="diagram" class="mermaid"> | ||||
| %%{init: {"flowchart": {"defaultRenderer": "elk"}} }%% | ||||
| flowchart TB | ||||
|   %% I could not figure out how to use double quotes in labels in Mermaid | ||||
|   subgraph ibm[IBM Espresso CPU] | ||||
|     core0[IBM PowerPC Broadway Core 0] | ||||
|     core1[IBM PowerPC Broadway Core 1] | ||||
|     core2[IBM PowerPC Broadway Core 2] | ||||
|  | ||||
|     rom[16 KB ROM] | ||||
|  | ||||
|     core0 --- core2 | ||||
|  | ||||
|     rom --> core2 | ||||
|   end | ||||
|  | ||||
|   subgraph amd["`**AMD** Latte GPU`"] | ||||
|     mem[Memory & I/O Bridge] | ||||
|     dram[DRAM Controller] | ||||
|     edram[32 MB EDRAM MEM1] | ||||
|     rom[512 B SEEPROM] | ||||
|  | ||||
|     sata[SATA IF] | ||||
|     exi[EXI] | ||||
|  | ||||
|     subgraph gx[GX] | ||||
|       sram[3 MB 1T-SRAM] | ||||
|     end | ||||
|  | ||||
|     radeon[AMD Radeon R7xx GX2] | ||||
|  | ||||
|     mem --- gx | ||||
|     mem --- radeon | ||||
|  | ||||
|     rom --- mem | ||||
|  | ||||
|     mem --- sata | ||||
|     mem --- exi | ||||
|  | ||||
|     dram --- sata | ||||
|     dram --- exi | ||||
|   end | ||||
|  | ||||
|   ddr3[2 GB DDR3 RAM MEM2] | ||||
|  | ||||
|   mem --- ddr3 | ||||
|   dram --- ddr3 | ||||
|   edram --- ddr3 | ||||
|  | ||||
|   core1 --- mem | ||||
|  | ||||
|   exi --- rtc | ||||
|   rtc{{rtc}} | ||||
| </pre | ||||
|     > | ||||
|     <pre id="diagram" class="mermaid2"> | ||||
| %%{init: {"flowchart": {"defaultRenderer": "elk", "htmlLabels": false}} }%% | ||||
| flowchart TB | ||||
|   %% I could not figure out how to use double quotes in labels in Mermaid | ||||
|   subgraph ibm[IBM Espresso CPU] | ||||
|     core0[IBM PowerPC Broadway Core 0] | ||||
|     core1[IBM PowerPC Broadway Core 1] | ||||
|     core2[IBM PowerPC Broadway Core 2] | ||||
|  | ||||
|     rom[16 KB ROM] | ||||
|  | ||||
|     core0 --- core2 | ||||
|  | ||||
|     rom --> core2 | ||||
|   end | ||||
|  | ||||
|   subgraph amd["`**AMD** Latte GPU`"] | ||||
|     mem[Memory & I/O Bridge] | ||||
|     dram[DRAM Controller] | ||||
|     edram[32 MB EDRAM MEM1] | ||||
|     rom[512 B SEEPROM] | ||||
|  | ||||
|     sata[SATA IF] | ||||
|     exi[EXI] | ||||
|  | ||||
|     subgraph gx[GX] | ||||
|       sram[3 MB 1T-SRAM] | ||||
|     end | ||||
|  | ||||
|     radeon[AMD Radeon R7xx GX2] | ||||
|  | ||||
|     mem --- gx | ||||
|     mem --- radeon | ||||
|  | ||||
|     rom --- mem | ||||
|  | ||||
|     mem --- sata | ||||
|     mem --- exi | ||||
|  | ||||
|     dram --- sata | ||||
|     dram --- exi | ||||
|   end | ||||
|  | ||||
|   ddr3[2 GB DDR3 RAM MEM2] | ||||
|  | ||||
|   mem --- ddr3 | ||||
|   dram --- ddr3 | ||||
|   edram --- ddr3 | ||||
|  | ||||
|   core1 --- mem | ||||
|  | ||||
|   exi --- rtc | ||||
|   rtc{{rtc}} | ||||
| </pre | ||||
|     > | ||||
|  | ||||
|     <br /> | ||||
|     <pre id="diagram" class="mermaid2"> | ||||
| flowchart TB | ||||
|   %% I could not figure out how to use double quotes in labels in Mermaid | ||||
|   subgraph ibm[IBM Espresso CPU] | ||||
|     core0[IBM PowerPC Broadway Core 0] | ||||
|     core1[IBM PowerPC Broadway Core 1] | ||||
|     core2[IBM PowerPC Broadway Core 2] | ||||
|  | ||||
|     rom[16 KB ROM] | ||||
|  | ||||
|     core0 --- core2 | ||||
|  | ||||
|     rom --> core2 | ||||
|   end | ||||
|  | ||||
|   subgraph amd[AMD Latte GPU] | ||||
|     mem[Memory & I/O Bridge] | ||||
|     dram[DRAM Controller] | ||||
|     edram[32 MB EDRAM MEM1] | ||||
|     rom[512 B SEEPROM] | ||||
|  | ||||
|     sata[SATA IF] | ||||
|     exi[EXI] | ||||
|  | ||||
|     subgraph gx[GX] | ||||
|       sram[3 MB 1T-SRAM] | ||||
|     end | ||||
|  | ||||
|     radeon[AMD Radeon R7xx GX2] | ||||
|  | ||||
|     mem --- gx | ||||
|     mem --- radeon | ||||
|  | ||||
|     rom --- mem | ||||
|  | ||||
|     mem --- sata | ||||
|     mem --- exi | ||||
|  | ||||
|     dram --- sata | ||||
|     dram --- exi | ||||
|   end | ||||
|  | ||||
|   ddr3[2 GB DDR3 RAM MEM2] | ||||
|  | ||||
|   mem --- ddr3 | ||||
|   dram --- ddr3 | ||||
|   edram --- ddr3 | ||||
|  | ||||
|   core1 --- mem | ||||
|  | ||||
|   exi --- rtc | ||||
|   rtc{{rtc}} | ||||
| </pre | ||||
|     > | ||||
|     <br /> | ||||
|       | ||||
|     <pre id="diagram" class="mermaid2"> | ||||
|       flowchart LR | ||||
|   B1 --be be--x B2 | ||||
|   B1 --bo bo--o B3 | ||||
|   subgraph Ugge | ||||
|       B2 | ||||
|       B3 | ||||
|       subgraph inner | ||||
|           B4 | ||||
|           B5 | ||||
|       end | ||||
|       subgraph inner2 | ||||
|         subgraph deeper | ||||
|           C4 | ||||
|           C5 | ||||
|         end | ||||
|         C6 | ||||
|       end | ||||
|  | ||||
|       B4 --> C4 | ||||
|  | ||||
|       B3 -- X --> B4 | ||||
|       B2 --> inner | ||||
|  | ||||
|       C4 --> C5 | ||||
|   end | ||||
|  | ||||
|   subgraph outer | ||||
|       B6 | ||||
|   end | ||||
|   B6 --> B5 | ||||
|   </pre | ||||
|     > | ||||
|     <pre id="diagram" class="mermaid2"> | ||||
| sequenceDiagram | ||||
|     Customer->>+Stripe: Makes a payment request | ||||
|     Stripe->>+Bank: Forwards the payment request to the bank | ||||
|     Bank->>+Customer: Asks for authorization | ||||
|     Customer->>+Bank: Provides authorization | ||||
|     Bank->>+Stripe: Sends a response with payment details | ||||
|     Stripe->>+Merchant: Sends a notification of payment receipt | ||||
|     Merchant->>+Stripe: Confirms the payment | ||||
|     Stripe->>+Customer: Sends a confirmation of payment | ||||
|     Customer->>+Merchant: Receives goods or services | ||||
|         </pre | ||||
|     > | ||||
|     <pre id="diagram" class="mermaid2"> | ||||
| 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> | ||||
|     <br /> | ||||
|     <pre id="diagram" class="mermaid2"> | ||||
|   example-diagram | ||||
|     </pre> | ||||
|  | ||||
|     <!-- <div id="cy"></div> --> | ||||
|     <!-- <script src="http://localhost:9000/packages/mermaid-mindmap/dist/mermaid-mindmap-detector.js"></script> --> | ||||
|     <!-- <script src="./mermaid-example-diagram-detector.js"></script>    --> | ||||
|     <!-- <script src="//cdn.jsdelivr.net/npm/mermaid@9.1.7/dist/mermaid.min.js"></script> --> | ||||
|     <!-- <script src="./mermaid.js"></script> --> | ||||
|  | ||||
|     <scrpt> | ||||
|       // import mindmap from '../../packages/mermaid-mindmap/src/detector'; // import example from | ||||
|       '../../packages/mermaid-example-diagram/src/mermaid-example-diagram.core.mjs'; import mermaid | ||||
|       from './mermaid.esm.mjs'; // await mermaid.registerExternalDiagrams([example]); | ||||
|       mermaid.parseError = function (err, hash) { // console.error('Mermaid error: ', err); }; | ||||
|       mermaid.initialize({ // theme: 'forest', startOnLoad: true, logLevel: 0, flowchart: { // | ||||
|       defaultRenderer: 'elk', useMaxWidth: false, // htmlLabels: false, htmlLabels: true, }, // | ||||
|       htmlLabels: false, gantt: { useMaxWidth: false, }, useMaxWidth: false, }); function callback() | ||||
|       { alert('It worked'); } mermaid.parseError = function (err, hash) { console.error('In parse | ||||
|       error:'); console.error(err); }; // mermaid.test1('first_slow', 1200).then((r) => | ||||
|       console.info(r)); // mermaid.test1('second_fast', 200).then((r) => console.info(r)); // | ||||
|       mermaid.test1('third_fast', 200).then((r) => console.info(r)); // mermaid.test1('forth_slow', | ||||
|       1200).then((r) => console.info(r)); | ||||
|     </scrpt> | ||||
|     <script | ||||
|       type="text/javascript" | ||||
|       src="https://cdn.jsdelivr.net/npm/mermaid@10.2.0/dist/mermaid.min.js" | ||||
|     ></script> | ||||
|     <script type="module"> | ||||
|       import mermaid from 'https://cdn.jsdelivr.net/npm/mermaid@10.2.0/dist/mermaid.min.js'; | ||||
|       (function () { | ||||
|         mermaid.initialize({ startOnLoad: false }); | ||||
|         const elements = document.getElementsByClassName('mermaid'); | ||||
|         console.log(elements); | ||||
|         let id = 0; | ||||
|         [...elements].forEach((elem) => { | ||||
|           const insertSvg = function (svgCode) { | ||||
|             elem.innerHTML = svgCode; | ||||
|           }; | ||||
|  | ||||
|           console.log(atob(elem.innerText)); | ||||
|  | ||||
|           mermaid.render(`graphDiv-${id++}`, atob(elem.innerText), insertSvg); | ||||
|         }); | ||||
|       })(); | ||||
|     </script> | ||||
|   </body> | ||||
| </html> | ||||
| @@ -4,7 +4,7 @@ | ||||
|     <link href="https://unpkg.com/tailwindcss@^1.0/dist/tailwind.min.css" rel="stylesheet" /> | ||||
|     <link | ||||
|       rel="stylesheet" | ||||
|       href="https://cdnjs.cloudflare.com/ajax/libs/font-awesome/4.7.0/css/font-awesome.min.css" | ||||
|       href="https://cdnjs.cloudflare.com/ajax/libs/font-awesome/6.5.2/css/font-awesome.min.css" | ||||
|     /> | ||||
|     <link | ||||
|       href="https://cdn.jsdelivr.net/npm/@mdi/font@6.9.96/css/materialdesignicons.min.css" | ||||
| @@ -14,50 +14,34 @@ | ||||
|       href="https://fonts.googleapis.com/css?family=Noto+Sans+SC&display=swap" | ||||
|       rel="stylesheet" | ||||
|     /> | ||||
|     <link rel="preconnect" href="https://fonts.googleapis.com" /> | ||||
|     <link rel="preconnect" href="https://fonts.gstatic.com" crossorigin /> | ||||
|     <link | ||||
|       href="https://fonts.googleapis.com/css2?family=Kalam:wght@300;400;700&display=swap" | ||||
|       rel="stylesheet" | ||||
|     /> | ||||
|     <link | ||||
|       href="https://fonts.googleapis.com/css2?family=Caveat:wght@400..700&family=Kalam:wght@300;400;700&family=Rubik+Mono+One&display=swap" | ||||
|       rel="stylesheet" | ||||
|     /> | ||||
|     <link | ||||
|       href="https://fonts.googleapis.com/css2?family=Kalam:wght@300;400;700&family=Rubik+Mono+One&display=swap" | ||||
|       rel="stylesheet" | ||||
|     /> | ||||
|  | ||||
|     <style> | ||||
|       body { | ||||
|         /* background: rgb(221, 208, 208); */ | ||||
|         /* background: #333; */ | ||||
|         background: #333; | ||||
|         font-family: 'Arial'; | ||||
|         /* font-size: 18px !important; */ | ||||
|       } | ||||
|  | ||||
|       h1 { | ||||
|         color: grey; | ||||
|       } | ||||
|  | ||||
|       .mermaid { | ||||
|         border: 1px solid #ddd; | ||||
|         margin: 10px; | ||||
|       } | ||||
|       .mermaid2 { | ||||
|         display: none; | ||||
|       } | ||||
|  | ||||
|       .mermaid svg { | ||||
|         /* font-size: 18px !important; */ | ||||
|  | ||||
|         /* background-color: #efefef; | ||||
|         background-image: radial-gradient(#fff 51%, transparent 91%), | ||||
|           radial-gradient(#fff 51%, transparent 91%); | ||||
|         /* background-color: #efefef; */ | ||||
|         background-color: #333; | ||||
|         background-image: radial-gradient(#333 51%, transparent 91%), | ||||
|           radial-gradient(#333 51%, transparent 91%); | ||||
|         background-size: 20px 20px; | ||||
|         background-position: | ||||
|           0 0, | ||||
|           10px 10px; | ||||
|         background-repeat: repeat; */ | ||||
|         background-position: 0 0, 10px 10px; | ||||
|         background-repeat: repeat; | ||||
|         border: 2px solid rgb(131, 142, 205); | ||||
|       } | ||||
|  | ||||
|       .malware { | ||||
|         position: fixed; | ||||
|         bottom: 0; | ||||
| @@ -73,130 +57,546 @@ | ||||
|         font-family: monospace; | ||||
|         font-size: 72px; | ||||
|       } | ||||
|  | ||||
|       /* tspan { | ||||
|               font-size: 6px !important; | ||||
|             } */ | ||||
|         font-size: 6px !important; | ||||
|       } */ | ||||
|     </style> | ||||
|   </head> | ||||
|  | ||||
|   <body> | ||||
|     <div class="flex"> | ||||
|       <pre id="diagram" class="mermaid"> | ||||
| --- | ||||
|   title: hello2 | ||||
|   config: | ||||
|     look: handDrawn | ||||
|     layout: elk | ||||
|     elk: | ||||
|         nodePlacementStrategy: BRANDES_KOEPF | ||||
| --- | ||||
| flowchart LR | ||||
|   A[Start] --Some text--> B(Continue) | ||||
|   B --> C{Evaluate} | ||||
|   C -- One --> D[Option 1] | ||||
|   C -- Two --> E[Option 2] | ||||
|   C -- Three --> F[fa:fa-car Option 3] | ||||
|  | ||||
|  | ||||
|  | ||||
| </pre | ||||
|       > | ||||
|       <pre id="diagram" class="mermaid2"> | ||||
| --- | ||||
| config: | ||||
|   look: handdrawn | ||||
|   flowchart: | ||||
|     htmlLabels: true | ||||
| --- | ||||
| flowchart | ||||
|       A[I am a long text, where do I go??? handdrawn - true] | ||||
| </pre | ||||
|       > | ||||
|     </div> | ||||
|     <div class="flex"> | ||||
|       <pre id="diagram" class="mermaid2"> | ||||
| --- | ||||
| config: | ||||
|   flowchart: | ||||
|     htmlLabels: false | ||||
| --- | ||||
| flowchart | ||||
|       A[I am a long text, where do I go??? classic - false] | ||||
| </pre | ||||
|       > | ||||
|       <pre id="diagram" class="mermaid2"> | ||||
| --- | ||||
| config: | ||||
|   flowchart: | ||||
|     htmlLabels: true | ||||
| --- | ||||
| flowchart | ||||
|       A[I am a long text, where do I go??? classic - true] | ||||
| </pre | ||||
|       > | ||||
|     </div> | ||||
|     <pre id="diagram2" class="mermaid2"> | ||||
| 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 | ||||
|  | ||||
|     <pre id="diagram" class="mermaid"> | ||||
|       block-beta | ||||
|   blockArrowId<["Label"]>(right) | ||||
|   blockArrowId2<["Label"]>(left) | ||||
|   blockArrowId3<["Label"]>(up) | ||||
|   blockArrowId4<["Label"]>(down) | ||||
|   blockArrowId5<["Label"]>(x) | ||||
|   blockArrowId6<["Label"]>(y) | ||||
|   blockArrowId6<["Label"]>(x, down) | ||||
|     </pre> | ||||
|     <pre id="diagram" class="mermaid"> | ||||
| block-beta | ||||
|   block:e:4 | ||||
|     columns 2 | ||||
|       f | ||||
|       g | ||||
|   end | ||||
|  | ||||
|     </pre> | ||||
|     <pre id="diagram" class="mermaid"> | ||||
| block-beta | ||||
|   block:e:4 | ||||
|     columns 2 | ||||
|       f | ||||
|       g | ||||
|       h | ||||
|   end | ||||
|  | ||||
|     <pre id="diagram3" class="mermaid2"> | ||||
|     </pre> | ||||
|     <pre id="diagram" class="mermaid"> | ||||
| block-beta | ||||
|   columns 4 | ||||
|   a b c d | ||||
|   block:e:4 | ||||
|     columns 2 | ||||
|       f | ||||
|       g | ||||
|       h | ||||
|   end | ||||
|   i:4 | ||||
|  | ||||
|     </pre> | ||||
|     <pre id="diagram" class="mermaid2"> | ||||
| flowchart LR | ||||
|   X-- "y" -->z | ||||
|     </pre> | ||||
|     <pre id="diagram" class="mermaid2"> | ||||
| block-beta | ||||
| columns 5 | ||||
|    A space B | ||||
|    A --x B | ||||
|     </pre> | ||||
|     <pre id="diagram" class="mermaid2"> | ||||
| block-beta | ||||
| columns 3 | ||||
|   a["A wide one"] b:2 c:2 d | ||||
|     </pre> | ||||
|     <pre id="diagram" class="mermaid2"> | ||||
| block-beta | ||||
|   block:e | ||||
|       f | ||||
|   end | ||||
|     </pre> | ||||
|     <pre id="diagram" class="mermaid2"> | ||||
| block-beta | ||||
|   columns 3 | ||||
|   a:3 | ||||
|   block:e:3 | ||||
|       f | ||||
|   end | ||||
|   g | ||||
|     </pre> | ||||
|     <pre id="diagram" class="mermaid2"> | ||||
| block-beta | ||||
|   columns 3 | ||||
|   a:3 | ||||
|   block:e:3 | ||||
|       f | ||||
|       g | ||||
|   end | ||||
|   h | ||||
|   i | ||||
|   j | ||||
|  | ||||
|     </pre> | ||||
|     <pre id="diagram" class="mermaid2"> | ||||
| block-beta | ||||
| columns 3 | ||||
|   a b:2 | ||||
|   block:e:3 | ||||
|       f | ||||
|   end | ||||
|   g h i | ||||
|     </pre> | ||||
|     <pre id="diagram" class="mermaid"> | ||||
| block-beta | ||||
| columns 3 | ||||
|   a b c | ||||
|   e:3 | ||||
|   f g h | ||||
|     </pre> | ||||
|     <pre id="diagram" class="mermaid"> | ||||
| block-beta | ||||
| columns 1 | ||||
|   db(("DB")) | ||||
|   blockArrowId6<["   "]>(down) | ||||
|   block:ID | ||||
|     A | ||||
|     B["A wide one in the middle"] | ||||
|     C | ||||
|   end | ||||
|   space | ||||
|   D | ||||
|   ID --> D | ||||
|   C --> D | ||||
|   style B fill:#f9F,stroke:#333,stroke-width:4px | ||||
|     </pre> | ||||
|     <pre id="diagram" class="mermaid"> | ||||
| block-beta | ||||
|   columns 5 | ||||
|   A1:3 | ||||
|   A2:1 | ||||
|   A3 | ||||
|   B1 B2 B3:3 | ||||
|     </pre> | ||||
|     <pre id="diagram" class="mermaid2"> | ||||
| block-beta | ||||
|   block | ||||
|     D | ||||
|     E | ||||
|   end | ||||
|   db("This is the text in the box") | ||||
|     </pre> | ||||
|     <pre id="diagram" class="mermaid2"> | ||||
| block-beta | ||||
|  | ||||
|       block | ||||
|         D | ||||
|       end | ||||
|       A["A: I am a wide one"] | ||||
|     </pre> | ||||
|     <pre id="diagram" class="mermaid2"> | ||||
| block-beta | ||||
|     A["square"] | ||||
|     B("rounded") | ||||
|     C(("circle")) | ||||
|     </pre> | ||||
|     <pre id="diagram" class="mermaid2"> | ||||
| block-beta | ||||
|     A>"rect_left_inv_arrow"] | ||||
|     B{"diamond"} | ||||
|     C{{"hexagon"}} | ||||
|     </pre> | ||||
|     <pre id="diagram" class="mermaid2"> | ||||
| block-beta | ||||
|     A(["stadium"]) | ||||
|     </pre> | ||||
|     <pre id="diagram" class="mermaid2"> | ||||
| block-beta | ||||
|     %% A[["subroutine"]] | ||||
|     %% B[("cylinder")] | ||||
|     C>"surprise"] | ||||
|     </pre> | ||||
|     <pre id="diagram" class="mermaid2"> | ||||
| block-beta | ||||
|     A[/"lean right"/] | ||||
|     B[\"lean left"\] | ||||
|     C[/"trapezoid"\] | ||||
|     D[\"trapezoid"/] | ||||
|     </pre> | ||||
|  | ||||
|     <pre id="diagram" class="mermaid2"> | ||||
| flowchart | ||||
|       B | ||||
|       style B fill:#f9F,stroke:#333,stroke-width:4px | ||||
|     </pre> | ||||
|  | ||||
|     <pre id="diagram" class="mermaid2"> | ||||
|       flowchart LR | ||||
|     A:::foo & B:::bar --> C:::foobar | ||||
|     classDef foo stroke:#f00 | ||||
|     classDef bar stroke:#0f0 | ||||
|     classDef ash color:red | ||||
|     class C ash | ||||
|     style C stroke:#00f, fill:black | ||||
|  | ||||
|       a1 -- apa --> b1 | ||||
|     </pre> | ||||
|  | ||||
|     <pre id="diagram4" class="mermaid2"> | ||||
|       stateDiagram | ||||
|     A:::foo | ||||
|     B:::bar --> C:::foobar | ||||
|     classDef foo stroke:#f00 | ||||
|     classDef bar stroke:#0f0 | ||||
|     style C stroke:#00f, fill:black, color:white | ||||
|     <pre id="diagram" class="mermaid2"> | ||||
| flowchart RL | ||||
|   subgraph "`one`" | ||||
|     id | ||||
|   end | ||||
|     </pre> | ||||
|     <pre id="diagram" class="mermaid2"> | ||||
| flowchart RL | ||||
|     subgraph "`one`" | ||||
|       a1 -- l1 --> a2 | ||||
|       a1 -- l2 --> a2 | ||||
|     end | ||||
|     </pre> | ||||
|     <pre id="diagram" class="mermaid2"> | ||||
| flowchart | ||||
| id["`A root with a long text that wraps to keep the node size in check. A root with a long text that wraps to keep the node size in check`"]</pre | ||||
|     > | ||||
|     <pre id="diagram" class="mermaid2"> | ||||
| flowchart LR | ||||
|     A[A text that needs to be wrapped wraps to another line] | ||||
|     B[A text that needs to be<br/>wrapped wraps to another line] | ||||
|     C["`A text that needs to be wrapped to another line`"]</pre> | ||||
|     <pre id="diagram" class="mermaid2"> | ||||
| flowchart LR | ||||
|     C["`A text | ||||
|         that needs | ||||
|         to be wrapped | ||||
|         in another | ||||
|         way`"] | ||||
|   </pre | ||||
|     > | ||||
|     <pre id="diagram" class="mermaid2"> | ||||
|       classDiagram-v2 | ||||
|         note "I love this diagram!\nDo you love it?" | ||||
|     </pre> | ||||
|     <pre id="diagram" class="mermaid2"> | ||||
|     stateDiagram-v2 | ||||
|     State1: The state with a note with minus - and plus + in it | ||||
|     note left of State1 | ||||
|       Important information! You can write | ||||
|       notes with . and  in them. | ||||
|     end note    </pre | ||||
|     > | ||||
|     <pre id="diagram" class="mermaid2"> | ||||
| mindmap | ||||
| root | ||||
|   Child3(A node with an icon and with a long text that wraps to keep the node size in check) | ||||
| </pre | ||||
|     > | ||||
|     <pre id="diagram" class="mermaid2"> | ||||
|       %%{init: {"theme": "forest"} }%% | ||||
| mindmap | ||||
|     id1[**Start2**<br/>end] | ||||
|       id2[**Start2**<br />end] | ||||
|       %% Another comment | ||||
|       id3[**Start2**<br>end] %% Comment | ||||
|       id4[**Start2**<br >end<br    >the very end] | ||||
|     </pre> | ||||
|     <pre id="diagram" class="mermaid2"> | ||||
| mindmap | ||||
|     id1["`**Start2** | ||||
|     second line 😎 with long text that is wrapping to the next line`"] | ||||
|       id2["`Child **with bold** text`"] | ||||
|       id3["`Children of which some | ||||
|       is using *italic type of* text`"] | ||||
|       id4[Child] | ||||
|       id5["`Child | ||||
|       Row | ||||
|       and another | ||||
|       `"] | ||||
|     </pre> | ||||
|     <pre id="diagram" class="mermaid2"> | ||||
| mindmap | ||||
|     id1("`**Root**`"] | ||||
|       id2["`A formatted text... with **bold** and *italics*`"] | ||||
|       id3[Regular labels works as usual] | ||||
|       id4["`Emojis and unicode works too: 🤓 | ||||
|       शान्तिः سلام  和平 `"] | ||||
|  | ||||
|     </pre> | ||||
|     <pre id="diagram" class="mermaid2"> | ||||
| %%{init: {"flowchart": {"defaultRenderer": "elk"}} }%% | ||||
| flowchart TB | ||||
|   %% I could not figure out how to use double quotes in labels in Mermaid | ||||
|   subgraph ibm[IBM Espresso CPU] | ||||
|     core0[IBM PowerPC Broadway Core 0] | ||||
|     core1[IBM PowerPC Broadway Core 1] | ||||
|     core2[IBM PowerPC Broadway Core 2] | ||||
|  | ||||
|     rom[16 KB ROM] | ||||
|  | ||||
|     core0 --- core2 | ||||
|  | ||||
|     rom --> core2 | ||||
|   end | ||||
|  | ||||
|   subgraph amd["`**AMD** Latte GPU`"] | ||||
|     mem[Memory & I/O Bridge] | ||||
|     dram[DRAM Controller] | ||||
|     edram[32 MB EDRAM MEM1] | ||||
|     rom[512 B SEEPROM] | ||||
|  | ||||
|     sata[SATA IF] | ||||
|     exi[EXI] | ||||
|  | ||||
|     subgraph gx[GX] | ||||
|       sram[3 MB 1T-SRAM] | ||||
|     end | ||||
|  | ||||
|     radeon[AMD Radeon R7xx GX2] | ||||
|  | ||||
|     mem --- gx | ||||
|     mem --- radeon | ||||
|  | ||||
|     rom --- mem | ||||
|  | ||||
|     mem --- sata | ||||
|     mem --- exi | ||||
|  | ||||
|     dram --- sata | ||||
|     dram --- exi | ||||
|   end | ||||
|  | ||||
|   ddr3[2 GB DDR3 RAM MEM2] | ||||
|  | ||||
|   mem --- ddr3 | ||||
|   dram --- ddr3 | ||||
|   edram --- ddr3 | ||||
|  | ||||
|   core1 --- mem | ||||
|  | ||||
|   exi --- rtc | ||||
|   rtc{{rtc}} | ||||
| </pre | ||||
|     > | ||||
|     <pre id="diagram" class="mermaid2"> | ||||
| %%{init: {"flowchart": {"defaultRenderer": "elk", "htmlLabels": false}} }%% | ||||
| flowchart TB | ||||
|   %% I could not figure out how to use double quotes in labels in Mermaid | ||||
|   subgraph ibm[IBM Espresso CPU] | ||||
|     core0[IBM PowerPC Broadway Core 0] | ||||
|     core1[IBM PowerPC Broadway Core 1] | ||||
|     core2[IBM PowerPC Broadway Core 2] | ||||
|  | ||||
|     rom[16 KB ROM] | ||||
|  | ||||
|     core0 --- core2 | ||||
|  | ||||
|     rom --> core2 | ||||
|   end | ||||
|  | ||||
|   subgraph amd["`**AMD** Latte GPU`"] | ||||
|     mem[Memory & I/O Bridge] | ||||
|     dram[DRAM Controller] | ||||
|     edram[32 MB EDRAM MEM1] | ||||
|     rom[512 B SEEPROM] | ||||
|  | ||||
|     sata[SATA IF] | ||||
|     exi[EXI] | ||||
|  | ||||
|     subgraph gx[GX] | ||||
|       sram[3 MB 1T-SRAM] | ||||
|     end | ||||
|  | ||||
|     radeon[AMD Radeon R7xx GX2] | ||||
|  | ||||
|     mem --- gx | ||||
|     mem --- radeon | ||||
|  | ||||
|     rom --- mem | ||||
|  | ||||
|     mem --- sata | ||||
|     mem --- exi | ||||
|  | ||||
|     dram --- sata | ||||
|     dram --- exi | ||||
|   end | ||||
|  | ||||
|   ddr3[2 GB DDR3 RAM MEM2] | ||||
|  | ||||
|   mem --- ddr3 | ||||
|   dram --- ddr3 | ||||
|   edram --- ddr3 | ||||
|  | ||||
|   core1 --- mem | ||||
|  | ||||
|   exi --- rtc | ||||
|   rtc{{rtc}} | ||||
| </pre | ||||
|     > | ||||
|  | ||||
|     <br /> | ||||
|     <pre id="diagram" class="mermaid2"> | ||||
| flowchart TB | ||||
|   %% I could not figure out how to use double quotes in labels in Mermaid | ||||
|   subgraph ibm[IBM Espresso CPU] | ||||
|     core0[IBM PowerPC Broadway Core 0] | ||||
|     core1[IBM PowerPC Broadway Core 1] | ||||
|     core2[IBM PowerPC Broadway Core 2] | ||||
|  | ||||
|     rom[16 KB ROM] | ||||
|  | ||||
|     core0 --- core2 | ||||
|  | ||||
|     rom --> core2 | ||||
|   end | ||||
|  | ||||
|   subgraph amd[AMD Latte GPU] | ||||
|     mem[Memory & I/O Bridge] | ||||
|     dram[DRAM Controller] | ||||
|     edram[32 MB EDRAM MEM1] | ||||
|     rom[512 B SEEPROM] | ||||
|  | ||||
|     sata[SATA IF] | ||||
|     exi[EXI] | ||||
|  | ||||
|     subgraph gx[GX] | ||||
|       sram[3 MB 1T-SRAM] | ||||
|     end | ||||
|  | ||||
|     radeon[AMD Radeon R7xx GX2] | ||||
|  | ||||
|     mem --- gx | ||||
|     mem --- radeon | ||||
|  | ||||
|     rom --- mem | ||||
|  | ||||
|     mem --- sata | ||||
|     mem --- exi | ||||
|  | ||||
|     dram --- sata | ||||
|     dram --- exi | ||||
|   end | ||||
|  | ||||
|   ddr3[2 GB DDR3 RAM MEM2] | ||||
|  | ||||
|   mem --- ddr3 | ||||
|   dram --- ddr3 | ||||
|   edram --- ddr3 | ||||
|  | ||||
|   core1 --- mem | ||||
|  | ||||
|   exi --- rtc | ||||
|   rtc{{rtc}} | ||||
| </pre | ||||
|     > | ||||
|     <br /> | ||||
|       | ||||
|     <pre id="diagram" class="mermaid2"> | ||||
|       flowchart LR | ||||
|   B1 --be be--x B2 | ||||
|   B1 --bo bo--o B3 | ||||
|   subgraph Ugge | ||||
|       B2 | ||||
|       B3 | ||||
|       subgraph inner | ||||
|           B4 | ||||
|           B5 | ||||
|       end | ||||
|       subgraph inner2 | ||||
|         subgraph deeper | ||||
|           C4 | ||||
|           C5 | ||||
|         end | ||||
|         C6 | ||||
|       end | ||||
|  | ||||
|       B4 --> C4 | ||||
|  | ||||
|       B3 -- X --> B4 | ||||
|       B2 --> inner | ||||
|  | ||||
|       C4 --> C5 | ||||
|   end | ||||
|  | ||||
|   subgraph outer | ||||
|       B6 | ||||
|   end | ||||
|   B6 --> B5 | ||||
|   </pre | ||||
|     > | ||||
|     <pre id="diagram" class="mermaid2"> | ||||
| sequenceDiagram | ||||
|     Customer->>+Stripe: Makes a payment request | ||||
|     Stripe->>+Bank: Forwards the payment request to the bank | ||||
|     Bank->>+Customer: Asks for authorization | ||||
|     Customer->>+Bank: Provides authorization | ||||
|     Bank->>+Stripe: Sends a response with payment details | ||||
|     Stripe->>+Merchant: Sends a notification of payment receipt | ||||
|     Merchant->>+Stripe: Confirms the payment | ||||
|     Stripe->>+Customer: Sends a confirmation of payment | ||||
|     Customer->>+Merchant: Receives goods or services | ||||
|         </pre | ||||
|     > | ||||
|     <pre id="diagram" class="mermaid2"> | ||||
| 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> | ||||
|     <br /> | ||||
|     <pre id="diagram" class="mermaid2"> | ||||
|   example-diagram | ||||
|     </pre> | ||||
|  | ||||
|     <!-- <div id="cy"></div> --> | ||||
|     <!-- <script src="http://localhost:9000/packages/mermaid-mindmap/dist/mermaid-mindmap-detector.js"></script> --> | ||||
|     <!-- <script src="./mermaid-example-diagram-detector.js"></script>    --> | ||||
|     <!-- <script src="//cdn.jsdelivr.net/npm/mermaid@9.1.7/dist/mermaid.min.js"></script> --> | ||||
|     <!-- <script src="./mermaid.js"></script> --> | ||||
|  | ||||
|     <script type="module"> | ||||
|       // import mindmap from '../../packages/mermaid-mindmap/src/detector'; | ||||
|       // import example from '../../packages/mermaid-example-diagram/src/mermaid-example-diagram.core.mjs'; | ||||
|       import mermaid from './mermaid.esm.mjs'; | ||||
|       import layouts from './mermaid-layout-elk.esm.mjs'; | ||||
|       mermaid.registerLayoutLoaders(layouts); | ||||
|       // await mermaid.registerExternalDiagrams([example]); | ||||
|       mermaid.parseError = function (err, hash) { | ||||
|         console.error('Mermaid error: ', err); | ||||
|       }; | ||||
|       window.callback = function () { | ||||
|         alert('A callback was triggered'); | ||||
|         // console.error('Mermaid error: ', err); | ||||
|       }; | ||||
|       // mermaid.initialize({ | ||||
|       //   // theme: 'forest', | ||||
|       //   startOnLoad: true, | ||||
|       //   logLevel: 0, | ||||
|       //   flowchart: { | ||||
|       //     // defaultRenderer: 'elk', | ||||
|       //     useMaxWidth: false, | ||||
|       //     // htmlLabels: false, | ||||
|       //     htmlLabels: true, | ||||
|       //   }, | ||||
|       //   // htmlLabels: false, | ||||
|       //   gantt: { | ||||
|       //     useMaxWidth: false, | ||||
|       //   }, | ||||
|       //   useMaxWidth: false, | ||||
|       // }); | ||||
|       mermaid.initialize({ | ||||
|         // theme: 'base', | ||||
|         // handDrawnSeed: 12, | ||||
|         // look: 'handDrawn', | ||||
|         // 'elk.nodePlacement.strategy': 'NETWORK_SIMPLEX', | ||||
|         // layout: 'dagre', | ||||
|         // layout: 'elk', | ||||
|         // layout: 'fixed', | ||||
|         // htmlLabels: false, | ||||
|         flowchart: { titleTopMargin: 10 }, | ||||
|         // fontFamily: 'Caveat', | ||||
|         // fontFamily: 'Kalam', | ||||
|         // fontFamily: 'courier', | ||||
|         sequence: { | ||||
|           actorFontFamily: 'courier', | ||||
|           noteFontFamily: 'courier', | ||||
|           messageFontFamily: 'courier', | ||||
|         }, | ||||
|         fontSize: 12, | ||||
|         theme: 'dark', | ||||
|         startOnLoad: true, | ||||
|         logLevel: 0, | ||||
|         securityLevel: 'loose', | ||||
|       }); | ||||
|       function callback() { | ||||
|         alert('It worked'); | ||||
| @@ -205,6 +605,10 @@ flowchart LR | ||||
|         console.error('In parse error:'); | ||||
|         console.error(err); | ||||
|       }; | ||||
|       // mermaid.test1('first_slow', 1200).then((r) => console.info(r)); | ||||
|       // mermaid.test1('second_fast', 200).then((r) => console.info(r)); | ||||
|       // mermaid.test1('third_fast', 200).then((r) => console.info(r)); | ||||
|       // mermaid.test1('forth_slow', 1200).then((r) => console.info(r)); | ||||
|     </script> | ||||
|   </body> | ||||
| </html> | ||||
|   | ||||
| @@ -1,4 +1,4 @@ | ||||
| <!doctype html> | ||||
| <!DOCTYPE html> | ||||
| <html lang="en"> | ||||
|   <head> | ||||
|     <meta charset="utf-8" /> | ||||
|   | ||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -1,7 +1,7 @@ | ||||
| import externalExample from './mermaid-example-diagram.esm.mjs'; | ||||
| import layouts from './mermaid-layout-elk.esm.mjs'; | ||||
| import zenUml from './mermaid-zenuml.esm.mjs'; | ||||
| import mermaid from './mermaid.esm.mjs'; | ||||
| import flowchartELK from './mermaid-flowchart-elk.esm.mjs'; | ||||
| import externalExample from './mermaid-example-diagram.esm.mjs'; | ||||
| import zenUml from './mermaid-zenuml.esm.mjs'; | ||||
|  | ||||
| function b64ToUtf8(str) { | ||||
|   return decodeURIComponent(escape(window.atob(str))); | ||||
| @@ -46,9 +46,7 @@ const contentLoaded = async function () { | ||||
|       document.getElementsByTagName('body')[0].appendChild(div); | ||||
|     } | ||||
|  | ||||
|     await mermaid.registerExternalDiagrams([externalExample, zenUml]); | ||||
|  | ||||
|     mermaid.registerLayoutLoaders(layouts); | ||||
|     await mermaid.registerExternalDiagrams([externalExample, zenUml, flowchartELK]); | ||||
|     mermaid.initialize(graphObj.mermaid); | ||||
|     await mermaid.run(); | ||||
|   } | ||||
|   | ||||
| @@ -1,43 +0,0 @@ | ||||
| > **Warning** | ||||
| > | ||||
| > ## THIS IS AN AUTOGENERATED FILE. DO NOT EDIT. | ||||
| > | ||||
| > ## Please edit the corresponding file in [/packages/mermaid/src/docs/config/setup/interfaces/mermaid.LayoutData.md](../../../../packages/mermaid/src/docs/config/setup/interfaces/mermaid.LayoutData.md). | ||||
|  | ||||
| # Interface: LayoutData | ||||
|  | ||||
| [mermaid](../modules/mermaid.md).LayoutData | ||||
|  | ||||
| ## Indexable | ||||
|  | ||||
| ▪ \[key: `string`]: `any` | ||||
|  | ||||
| ## Properties | ||||
|  | ||||
| ### config | ||||
|  | ||||
| • **config**: `MermaidConfig` | ||||
|  | ||||
| #### Defined in | ||||
|  | ||||
| [packages/mermaid/src/rendering-util/types.d.ts:118](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/rendering-util/types.d.ts#L118) | ||||
|  | ||||
| --- | ||||
|  | ||||
| ### edges | ||||
|  | ||||
| • **edges**: `Edge`\[] | ||||
|  | ||||
| #### Defined in | ||||
|  | ||||
| [packages/mermaid/src/rendering-util/types.d.ts:117](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/rendering-util/types.d.ts#L117) | ||||
|  | ||||
| --- | ||||
|  | ||||
| ### nodes | ||||
|  | ||||
| • **nodes**: `Node`\[] | ||||
|  | ||||
| #### Defined in | ||||
|  | ||||
| [packages/mermaid/src/rendering-util/types.d.ts:116](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/rendering-util/types.d.ts#L116) | ||||
| @@ -1,39 +0,0 @@ | ||||
| > **Warning** | ||||
| > | ||||
| > ## THIS IS AN AUTOGENERATED FILE. DO NOT EDIT. | ||||
| > | ||||
| > ## Please edit the corresponding file in [/packages/mermaid/src/docs/config/setup/interfaces/mermaid.LayoutLoaderDefinition.md](../../../../packages/mermaid/src/docs/config/setup/interfaces/mermaid.LayoutLoaderDefinition.md). | ||||
|  | ||||
| # Interface: LayoutLoaderDefinition | ||||
|  | ||||
| [mermaid](../modules/mermaid.md).LayoutLoaderDefinition | ||||
|  | ||||
| ## Properties | ||||
|  | ||||
| ### algorithm | ||||
|  | ||||
| • `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) | ||||
|  | ||||
| --- | ||||
|  | ||||
| ### loader | ||||
|  | ||||
| • **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) | ||||
|  | ||||
| --- | ||||
|  | ||||
| ### name | ||||
|  | ||||
| • **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) | ||||
| @@ -28,7 +28,7 @@ page. | ||||
|  | ||||
| #### Defined in | ||||
|  | ||||
| [packages/mermaid/src/mermaid.ts:435](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/mermaid.ts#L435) | ||||
| [packages/mermaid/src/mermaid.ts:425](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/mermaid.ts#L425) | ||||
|  | ||||
| --- | ||||
|  | ||||
| @@ -59,7 +59,7 @@ A graph definition key | ||||
|  | ||||
| #### Defined in | ||||
|  | ||||
| [packages/mermaid/src/mermaid.ts:437](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/mermaid.ts#L437) | ||||
| [packages/mermaid/src/mermaid.ts:427](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/mermaid.ts#L427) | ||||
|  | ||||
| --- | ||||
|  | ||||
| @@ -89,7 +89,7 @@ Use [initialize](mermaid.Mermaid.md#initialize) and [run](mermaid.Mermaid.md#run | ||||
|  | ||||
| #### Defined in | ||||
|  | ||||
| [packages/mermaid/src/mermaid.ts:430](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/mermaid.ts#L430) | ||||
| [packages/mermaid/src/mermaid.ts:421](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/mermaid.ts#L421) | ||||
|  | ||||
| --- | ||||
|  | ||||
| @@ -116,13 +116,13 @@ This function should be called before the run function. | ||||
|  | ||||
| #### Defined in | ||||
|  | ||||
| [packages/mermaid/src/mermaid.ts:434](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/mermaid.ts#L434) | ||||
| [packages/mermaid/src/mermaid.ts:424](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/mermaid.ts#L424) | ||||
|  | ||||
| --- | ||||
|  | ||||
| ### mermaidAPI | ||||
|  | ||||
| • **mermaidAPI**: `Readonly`<{ `defaultConfig`: [`MermaidConfig`](mermaid.MermaidConfig.md) = configApi.defaultConfig; `getConfig`: () => [`MermaidConfig`](mermaid.MermaidConfig.md) = configApi.getConfig; `getDiagramFromText`: (`text`: `string`, `metadata`: `Pick`<`DiagramMetadata`, `"title"`>) => `Promise`<`Diagram`> ; `getSiteConfig`: () => [`MermaidConfig`](mermaid.MermaidConfig.md) = configApi.getSiteConfig; `globalReset`: () => `void` ; `initialize`: (`userOptions`: [`MermaidConfig`](mermaid.MermaidConfig.md)) => `void` ; `parse`: (`text`: `string`, `parseOptions`: [`ParseOptions`](mermaid.ParseOptions.md) & { `suppressErrors`: `true` }) => `Promise`<[`ParseResult`](mermaid.ParseResult.md) | `false`>(`text`: `string`, `parseOptions?`: [`ParseOptions`](mermaid.ParseOptions.md)) => `Promise`<[`ParseResult`](mermaid.ParseResult.md)> ; `render`: (`id`: `string`, `text`: `string`, `svgContainingElement?`: `Element`) => `Promise`<[`RenderResult`](mermaid.RenderResult.md)> ; `reset`: () => `void` ; `setConfig`: (`conf`: [`MermaidConfig`](mermaid.MermaidConfig.md)) => [`MermaidConfig`](mermaid.MermaidConfig.md) = configApi.setConfig; `updateSiteConfig`: (`conf`: [`MermaidConfig`](mermaid.MermaidConfig.md)) => [`MermaidConfig`](mermaid.MermaidConfig.md) = configApi.updateSiteConfig }> | ||||
| • **mermaidAPI**: `Readonly`<{ `defaultConfig`: [`MermaidConfig`](mermaid.MermaidConfig.md) = configApi.defaultConfig; `getConfig`: () => [`MermaidConfig`](mermaid.MermaidConfig.md) = configApi.getConfig; `getDiagramFromText`: (`text`: `string`, `metadata`: `Pick`<`DiagramMetadata`, `"title"`>) => `Promise`<`Diagram`> ; `getSiteConfig`: () => [`MermaidConfig`](mermaid.MermaidConfig.md) = configApi.getSiteConfig; `globalReset`: () => `void` ; `initialize`: (`options`: [`MermaidConfig`](mermaid.MermaidConfig.md)) => `void` ; `parse`: (`text`: `string`, `parseOptions`: [`ParseOptions`](mermaid.ParseOptions.md) & { `suppressErrors`: `true` }) => `Promise`<[`ParseResult`](mermaid.ParseResult.md) | `false`>(`text`: `string`, `parseOptions?`: [`ParseOptions`](mermaid.ParseOptions.md)) => `Promise`<[`ParseResult`](mermaid.ParseResult.md)> ; `render`: (`id`: `string`, `text`: `string`, `svgContainingElement?`: `Element`) => `Promise`<[`RenderResult`](mermaid.RenderResult.md)> ; `reset`: () => `void` ; `setConfig`: (`conf`: [`MermaidConfig`](mermaid.MermaidConfig.md)) => [`MermaidConfig`](mermaid.MermaidConfig.md) = configApi.setConfig; `updateSiteConfig`: (`conf`: [`MermaidConfig`](mermaid.MermaidConfig.md)) => [`MermaidConfig`](mermaid.MermaidConfig.md) = configApi.updateSiteConfig }> | ||||
|  | ||||
| **`Deprecated`** | ||||
|  | ||||
| @@ -130,7 +130,7 @@ Use [parse](mermaid.Mermaid.md#parse) and [render](mermaid.Mermaid.md#render) in | ||||
|  | ||||
| #### Defined in | ||||
|  | ||||
| [packages/mermaid/src/mermaid.ts:424](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/mermaid.ts#L424) | ||||
| [packages/mermaid/src/mermaid.ts:415](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/mermaid.ts#L415) | ||||
|  | ||||
| --- | ||||
|  | ||||
| @@ -180,7 +180,7 @@ Error if the diagram is invalid and parseOptions.suppressErrors is false or not | ||||
|  | ||||
| #### Defined in | ||||
|  | ||||
| [packages/mermaid/src/mermaid.ts:425](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/mermaid.ts#L425) | ||||
| [packages/mermaid/src/mermaid.ts:416](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/mermaid.ts#L416) | ||||
|  | ||||
| --- | ||||
|  | ||||
| @@ -190,7 +190,7 @@ Error if the diagram is invalid and parseOptions.suppressErrors is false or not | ||||
|  | ||||
| #### Defined in | ||||
|  | ||||
| [packages/mermaid/src/mermaid.ts:419](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/mermaid.ts#L419) | ||||
| [packages/mermaid/src/mermaid.ts:410](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/mermaid.ts#L410) | ||||
|  | ||||
| --- | ||||
|  | ||||
| @@ -218,31 +218,7 @@ Used to register external diagram types. | ||||
|  | ||||
| #### Defined in | ||||
|  | ||||
| [packages/mermaid/src/mermaid.ts:433](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/mermaid.ts#L433) | ||||
|  | ||||
| --- | ||||
|  | ||||
| ### registerLayoutLoaders | ||||
|  | ||||
| • **registerLayoutLoaders**: (`loaders`: [`LayoutLoaderDefinition`](mermaid.LayoutLoaderDefinition.md)\[]) => `void` | ||||
|  | ||||
| #### Type declaration | ||||
|  | ||||
| ▸ (`loaders`): `void` | ||||
|  | ||||
| ##### Parameters | ||||
|  | ||||
| | Name      | Type                                                             | | ||||
| | :-------- | :--------------------------------------------------------------- | | ||||
| | `loaders` | [`LayoutLoaderDefinition`](mermaid.LayoutLoaderDefinition.md)\[] | | ||||
|  | ||||
| ##### Returns | ||||
|  | ||||
| `void` | ||||
|  | ||||
| #### Defined in | ||||
|  | ||||
| [packages/mermaid/src/mermaid.ts:432](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/mermaid.ts#L432) | ||||
| [packages/mermaid/src/mermaid.ts:423](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/mermaid.ts#L423) | ||||
|  | ||||
| --- | ||||
|  | ||||
| @@ -268,7 +244,7 @@ Used to register external diagram types. | ||||
|  | ||||
| #### Defined in | ||||
|  | ||||
| [packages/mermaid/src/mermaid.ts:426](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/mermaid.ts#L426) | ||||
| [packages/mermaid/src/mermaid.ts:417](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/mermaid.ts#L417) | ||||
|  | ||||
| --- | ||||
|  | ||||
| @@ -316,7 +292,7 @@ Renders the mermaid diagrams | ||||
|  | ||||
| #### Defined in | ||||
|  | ||||
| [packages/mermaid/src/mermaid.ts:431](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/mermaid.ts#L431) | ||||
| [packages/mermaid/src/mermaid.ts:422](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/mermaid.ts#L422) | ||||
|  | ||||
| --- | ||||
|  | ||||
| @@ -351,7 +327,7 @@ to it (eg. dart interop wrapper). (Initially there is no parseError member of me | ||||
|  | ||||
| #### Defined in | ||||
|  | ||||
| [packages/mermaid/src/mermaid.ts:436](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/mermaid.ts#L436) | ||||
| [packages/mermaid/src/mermaid.ts:426](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/mermaid.ts#L426) | ||||
|  | ||||
| --- | ||||
|  | ||||
| @@ -361,4 +337,4 @@ to it (eg. dart interop wrapper). (Initially there is no parseError member of me | ||||
|  | ||||
| #### Defined in | ||||
|  | ||||
| [packages/mermaid/src/mermaid.ts:418](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/mermaid.ts#L418) | ||||
| [packages/mermaid/src/mermaid.ts:409](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/mermaid.ts#L409) | ||||
|   | ||||
| @@ -16,7 +16,7 @@ | ||||
|  | ||||
| #### Defined in | ||||
|  | ||||
| [packages/mermaid/src/config.type.ts:112](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L112) | ||||
| [packages/mermaid/src/config.type.ts:85](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L85) | ||||
|  | ||||
| --- | ||||
|  | ||||
| @@ -29,7 +29,7 @@ This matters if you are using base tag settings. | ||||
|  | ||||
| #### Defined in | ||||
|  | ||||
| [packages/mermaid/src/config.type.ts:131](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L131) | ||||
| [packages/mermaid/src/config.type.ts:104](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L104) | ||||
|  | ||||
| --- | ||||
|  | ||||
| @@ -39,7 +39,7 @@ This matters if you are using base tag settings. | ||||
|  | ||||
| #### Defined in | ||||
|  | ||||
| [packages/mermaid/src/config.type.ts:189](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L189) | ||||
| [packages/mermaid/src/config.type.ts:162](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L162) | ||||
|  | ||||
| --- | ||||
|  | ||||
| @@ -49,7 +49,7 @@ This matters if you are using base tag settings. | ||||
|  | ||||
| #### Defined in | ||||
|  | ||||
| [packages/mermaid/src/config.type.ts:186](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L186) | ||||
| [packages/mermaid/src/config.type.ts:159](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L159) | ||||
|  | ||||
| --- | ||||
|  | ||||
| @@ -59,7 +59,7 @@ This matters if you are using base tag settings. | ||||
|  | ||||
| #### Defined in | ||||
|  | ||||
| [packages/mermaid/src/config.type.ts:177](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L177) | ||||
| [packages/mermaid/src/config.type.ts:150](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L150) | ||||
|  | ||||
| --- | ||||
|  | ||||
| @@ -69,7 +69,7 @@ This matters if you are using base tag settings. | ||||
|  | ||||
| #### Defined in | ||||
|  | ||||
| [packages/mermaid/src/config.type.ts:103](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L103) | ||||
| [packages/mermaid/src/config.type.ts:76](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L76) | ||||
|  | ||||
| --- | ||||
|  | ||||
| @@ -83,7 +83,7 @@ You can set this attribute to base the seed on a static string. | ||||
|  | ||||
| #### Defined in | ||||
|  | ||||
| [packages/mermaid/src/config.type.ts:171](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L171) | ||||
| [packages/mermaid/src/config.type.ts:144](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L144) | ||||
|  | ||||
| --- | ||||
|  | ||||
| @@ -101,7 +101,7 @@ should not change unless content is changed. | ||||
|  | ||||
| #### Defined in | ||||
|  | ||||
| [packages/mermaid/src/config.type.ts:164](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L164) | ||||
| [packages/mermaid/src/config.type.ts:137](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L137) | ||||
|  | ||||
| --- | ||||
|  | ||||
| @@ -111,24 +111,7 @@ should not change unless content is changed. | ||||
|  | ||||
| #### Defined in | ||||
|  | ||||
| [packages/mermaid/src/config.type.ts:190](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L190) | ||||
|  | ||||
| --- | ||||
|  | ||||
| ### elk | ||||
|  | ||||
| • `Optional` **elk**: `Object` | ||||
|  | ||||
| #### Type declaration | ||||
|  | ||||
| | Name                     | Type                                                                          | Description                                                                                                                                               | | ||||
| | :----------------------- | :---------------------------------------------------------------------------- | :-------------------------------------------------------------------------------------------------------------------------------------------------------- | | ||||
| | `mergeEdges?`            | `boolean`                                                                     | Elk specific option that allows edges to share path where it convenient. It can make for pretty diagrams but can also make it harder to read the diagram. | | ||||
| | `nodePlacementStrategy?` | `"SIMPLE"` \| `"NETWORK_SIMPLEX"` \| `"LINEAR_SEGMENTS"` \| `"BRANDES_KOEPF"` | Elk specific option affecting how nodes are placed.                                                                                                       | | ||||
|  | ||||
| #### Defined in | ||||
|  | ||||
| [packages/mermaid/src/config.type.ts:91](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L91) | ||||
| [packages/mermaid/src/config.type.ts:163](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L163) | ||||
|  | ||||
| --- | ||||
|  | ||||
| @@ -138,7 +121,7 @@ should not change unless content is changed. | ||||
|  | ||||
| #### Defined in | ||||
|  | ||||
| [packages/mermaid/src/config.type.ts:179](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L179) | ||||
| [packages/mermaid/src/config.type.ts:152](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L152) | ||||
|  | ||||
| --- | ||||
|  | ||||
| @@ -148,7 +131,7 @@ should not change unless content is changed. | ||||
|  | ||||
| #### Defined in | ||||
|  | ||||
| [packages/mermaid/src/config.type.ts:172](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L172) | ||||
| [packages/mermaid/src/config.type.ts:145](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L145) | ||||
|  | ||||
| --- | ||||
|  | ||||
| @@ -162,7 +145,7 @@ See <https://developer.mozilla.org/en-US/docs/Web/CSS/font-family> | ||||
|  | ||||
| #### Defined in | ||||
|  | ||||
| [packages/mermaid/src/config.type.ts:111](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L111) | ||||
| [packages/mermaid/src/config.type.ts:84](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L84) | ||||
|  | ||||
| --- | ||||
|  | ||||
| @@ -172,7 +155,7 @@ See <https://developer.mozilla.org/en-US/docs/Web/CSS/font-family> | ||||
|  | ||||
| #### Defined in | ||||
|  | ||||
| [packages/mermaid/src/config.type.ts:192](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L192) | ||||
| [packages/mermaid/src/config.type.ts:165](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L165) | ||||
|  | ||||
| --- | ||||
|  | ||||
| @@ -186,7 +169,7 @@ If set to true, ignores legacyMathML. | ||||
|  | ||||
| #### Defined in | ||||
|  | ||||
| [packages/mermaid/src/config.type.ts:153](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L153) | ||||
| [packages/mermaid/src/config.type.ts:126](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L126) | ||||
|  | ||||
| --- | ||||
|  | ||||
| @@ -196,7 +179,7 @@ If set to true, ignores legacyMathML. | ||||
|  | ||||
| #### Defined in | ||||
|  | ||||
| [packages/mermaid/src/config.type.ts:174](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L174) | ||||
| [packages/mermaid/src/config.type.ts:147](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L147) | ||||
|  | ||||
| --- | ||||
|  | ||||
| @@ -206,19 +189,7 @@ If set to true, ignores legacyMathML. | ||||
|  | ||||
| #### Defined in | ||||
|  | ||||
| [packages/mermaid/src/config.type.ts:185](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L185) | ||||
|  | ||||
| --- | ||||
|  | ||||
| ### handDrawnSeed | ||||
|  | ||||
| • `Optional` **handDrawnSeed**: `number` | ||||
|  | ||||
| Defines the seed to be used when using handDrawn look. This is important for the automated tests as they will always find differences without the seed. The default value is 0 which gives a random seed. | ||||
|  | ||||
| #### Defined in | ||||
|  | ||||
| [packages/mermaid/src/config.type.ts:76](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L76) | ||||
| [packages/mermaid/src/config.type.ts:158](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L158) | ||||
|  | ||||
| --- | ||||
|  | ||||
| @@ -228,7 +199,7 @@ Defines the seed to be used when using handDrawn look. This is important for the | ||||
|  | ||||
| #### Defined in | ||||
|  | ||||
| [packages/mermaid/src/config.type.ts:104](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L104) | ||||
| [packages/mermaid/src/config.type.ts:77](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L77) | ||||
|  | ||||
| --- | ||||
|  | ||||
| @@ -238,19 +209,7 @@ Defines the seed to be used when using handDrawn look. This is important for the | ||||
|  | ||||
| #### Defined in | ||||
|  | ||||
| [packages/mermaid/src/config.type.ts:175](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L175) | ||||
|  | ||||
| --- | ||||
|  | ||||
| ### layout | ||||
|  | ||||
| • `Optional` **layout**: `string` | ||||
|  | ||||
| Defines which layout algorithm to use for rendering the diagram. | ||||
|  | ||||
| #### Defined in | ||||
|  | ||||
| [packages/mermaid/src/config.type.ts:81](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L81) | ||||
| [packages/mermaid/src/config.type.ts:148](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L148) | ||||
|  | ||||
| --- | ||||
|  | ||||
| @@ -265,7 +224,7 @@ fall back to legacy rendering for KaTeX. | ||||
|  | ||||
| #### Defined in | ||||
|  | ||||
| [packages/mermaid/src/config.type.ts:146](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L146) | ||||
| [packages/mermaid/src/config.type.ts:119](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L119) | ||||
|  | ||||
| --- | ||||
|  | ||||
| @@ -277,19 +236,7 @@ This option decides the amount of logging to be used by mermaid. | ||||
|  | ||||
| #### Defined in | ||||
|  | ||||
| [packages/mermaid/src/config.type.ts:117](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L117) | ||||
|  | ||||
| --- | ||||
|  | ||||
| ### look | ||||
|  | ||||
| • `Optional` **look**: `"classic"` | `"handDrawn"` | ||||
|  | ||||
| Defines which main look to use for the diagram. | ||||
|  | ||||
| #### Defined in | ||||
|  | ||||
| [packages/mermaid/src/config.type.ts:71](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L71) | ||||
| [packages/mermaid/src/config.type.ts:90](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L90) | ||||
|  | ||||
| --- | ||||
|  | ||||
| @@ -299,7 +246,7 @@ Defines which main look to use for the diagram. | ||||
|  | ||||
| #### Defined in | ||||
|  | ||||
| [packages/mermaid/src/config.type.ts:193](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L193) | ||||
| [packages/mermaid/src/config.type.ts:166](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L166) | ||||
|  | ||||
| --- | ||||
|  | ||||
| @@ -311,7 +258,7 @@ Defines the maximum number of edges that can be drawn in a graph. | ||||
|  | ||||
| #### Defined in | ||||
|  | ||||
| [packages/mermaid/src/config.type.ts:90](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L90) | ||||
| [packages/mermaid/src/config.type.ts:75](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L75) | ||||
|  | ||||
| --- | ||||
|  | ||||
| @@ -323,7 +270,7 @@ The maximum allowed size of the users text diagram | ||||
|  | ||||
| #### Defined in | ||||
|  | ||||
| [packages/mermaid/src/config.type.ts:85](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L85) | ||||
| [packages/mermaid/src/config.type.ts:70](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L70) | ||||
|  | ||||
| --- | ||||
|  | ||||
| @@ -333,7 +280,7 @@ The maximum allowed size of the users text diagram | ||||
|  | ||||
| #### Defined in | ||||
|  | ||||
| [packages/mermaid/src/config.type.ts:184](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L184) | ||||
| [packages/mermaid/src/config.type.ts:157](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L157) | ||||
|  | ||||
| --- | ||||
|  | ||||
| @@ -343,7 +290,7 @@ The maximum allowed size of the users text diagram | ||||
|  | ||||
| #### Defined in | ||||
|  | ||||
| [packages/mermaid/src/config.type.ts:188](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L188) | ||||
| [packages/mermaid/src/config.type.ts:161](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L161) | ||||
|  | ||||
| --- | ||||
|  | ||||
| @@ -353,7 +300,7 @@ The maximum allowed size of the users text diagram | ||||
|  | ||||
| #### Defined in | ||||
|  | ||||
| [packages/mermaid/src/config.type.ts:180](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L180) | ||||
| [packages/mermaid/src/config.type.ts:153](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L153) | ||||
|  | ||||
| --- | ||||
|  | ||||
| @@ -363,7 +310,7 @@ The maximum allowed size of the users text diagram | ||||
|  | ||||
| #### Defined in | ||||
|  | ||||
| [packages/mermaid/src/config.type.ts:181](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L181) | ||||
| [packages/mermaid/src/config.type.ts:154](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L154) | ||||
|  | ||||
| --- | ||||
|  | ||||
| @@ -373,7 +320,7 @@ The maximum allowed size of the users text diagram | ||||
|  | ||||
| #### Defined in | ||||
|  | ||||
| [packages/mermaid/src/config.type.ts:183](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L183) | ||||
| [packages/mermaid/src/config.type.ts:156](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L156) | ||||
|  | ||||
| --- | ||||
|  | ||||
| @@ -383,7 +330,7 @@ The maximum allowed size of the users text diagram | ||||
|  | ||||
| #### Defined in | ||||
|  | ||||
| [packages/mermaid/src/config.type.ts:187](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L187) | ||||
| [packages/mermaid/src/config.type.ts:160](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L160) | ||||
|  | ||||
| --- | ||||
|  | ||||
| @@ -397,7 +344,7 @@ This prevents malicious graph directives from overriding a site's default securi | ||||
|  | ||||
| #### Defined in | ||||
|  | ||||
| [packages/mermaid/src/config.type.ts:138](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L138) | ||||
| [packages/mermaid/src/config.type.ts:111](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L111) | ||||
|  | ||||
| --- | ||||
|  | ||||
| @@ -409,7 +356,7 @@ Level of trust for parsed diagram | ||||
|  | ||||
| #### Defined in | ||||
|  | ||||
| [packages/mermaid/src/config.type.ts:121](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L121) | ||||
| [packages/mermaid/src/config.type.ts:94](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L94) | ||||
|  | ||||
| --- | ||||
|  | ||||
| @@ -419,7 +366,7 @@ Level of trust for parsed diagram | ||||
|  | ||||
| #### Defined in | ||||
|  | ||||
| [packages/mermaid/src/config.type.ts:173](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L173) | ||||
| [packages/mermaid/src/config.type.ts:146](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L146) | ||||
|  | ||||
| --- | ||||
|  | ||||
| @@ -431,7 +378,7 @@ Dictates whether mermaid starts on Page load | ||||
|  | ||||
| #### Defined in | ||||
|  | ||||
| [packages/mermaid/src/config.type.ts:125](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L125) | ||||
| [packages/mermaid/src/config.type.ts:98](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L98) | ||||
|  | ||||
| --- | ||||
|  | ||||
| @@ -441,7 +388,7 @@ Dictates whether mermaid starts on Page load | ||||
|  | ||||
| #### Defined in | ||||
|  | ||||
| [packages/mermaid/src/config.type.ts:178](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L178) | ||||
| [packages/mermaid/src/config.type.ts:151](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L151) | ||||
|  | ||||
| --- | ||||
|  | ||||
| @@ -454,13 +401,13 @@ This is useful when you want to control how to handle syntax errors in your appl | ||||
|  | ||||
| #### Defined in | ||||
|  | ||||
| [packages/mermaid/src/config.type.ts:199](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L199) | ||||
| [packages/mermaid/src/config.type.ts:172](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L172) | ||||
|  | ||||
| --- | ||||
|  | ||||
| ### theme | ||||
|  | ||||
| • `Optional` **theme**: `"default"` | `"base"` | `"dark"` | `"forest"` | `"neutral"` | `"null"` | ||||
| • `Optional` **theme**: `"default"` | `"forest"` | `"dark"` | `"neutral"` | `"null"` | ||||
|  | ||||
| Theme, the CSS style sheet. | ||||
| You may also use `themeCSS` to override this value. | ||||
| @@ -497,7 +444,7 @@ You may also use `themeCSS` to override this value. | ||||
|  | ||||
| #### Defined in | ||||
|  | ||||
| [packages/mermaid/src/config.type.ts:176](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L176) | ||||
| [packages/mermaid/src/config.type.ts:149](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L149) | ||||
|  | ||||
| --- | ||||
|  | ||||
| @@ -507,7 +454,7 @@ You may also use `themeCSS` to override this value. | ||||
|  | ||||
| #### Defined in | ||||
|  | ||||
| [packages/mermaid/src/config.type.ts:191](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L191) | ||||
| [packages/mermaid/src/config.type.ts:164](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L164) | ||||
|  | ||||
| --- | ||||
|  | ||||
| @@ -517,4 +464,4 @@ You may also use `themeCSS` to override this value. | ||||
|  | ||||
| #### Defined in | ||||
|  | ||||
| [packages/mermaid/src/config.type.ts:182](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L182) | ||||
| [packages/mermaid/src/config.type.ts:155](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.type.ts#L155) | ||||
|   | ||||
| @@ -1,19 +0,0 @@ | ||||
| > **Warning** | ||||
| > | ||||
| > ## THIS IS AN AUTOGENERATED FILE. DO NOT EDIT. | ||||
| > | ||||
| > ## Please edit the corresponding file in [/packages/mermaid/src/docs/config/setup/interfaces/mermaid.RenderOptions.md](../../../../packages/mermaid/src/docs/config/setup/interfaces/mermaid.RenderOptions.md). | ||||
|  | ||||
| # Interface: RenderOptions | ||||
|  | ||||
| [mermaid](../modules/mermaid.md).RenderOptions | ||||
|  | ||||
| ## Properties | ||||
|  | ||||
| ### algorithm | ||||
|  | ||||
| • `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) | ||||
| @@ -18,7 +18,7 @@ The nodes to render. If this is set, `querySelector` will be ignored. | ||||
|  | ||||
| #### Defined in | ||||
|  | ||||
| [packages/mermaid/src/mermaid.ts:48](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/mermaid.ts#L48) | ||||
| [packages/mermaid/src/mermaid.ts:39](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/mermaid.ts#L39) | ||||
|  | ||||
| --- | ||||
|  | ||||
| @@ -44,7 +44,7 @@ A callback to call after each diagram is rendered. | ||||
|  | ||||
| #### Defined in | ||||
|  | ||||
| [packages/mermaid/src/mermaid.ts:52](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/mermaid.ts#L52) | ||||
| [packages/mermaid/src/mermaid.ts:43](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/mermaid.ts#L43) | ||||
|  | ||||
| --- | ||||
|  | ||||
| @@ -56,7 +56,7 @@ The query selector to use when finding elements to render. Default: `".mermaid"` | ||||
|  | ||||
| #### Defined in | ||||
|  | ||||
| [packages/mermaid/src/mermaid.ts:44](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/mermaid.ts#L44) | ||||
| [packages/mermaid/src/mermaid.ts:35](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/mermaid.ts#L35) | ||||
|  | ||||
| --- | ||||
|  | ||||
| @@ -68,4 +68,4 @@ If `true`, errors will be logged to the console, but not thrown. Default: `false | ||||
|  | ||||
| #### Defined in | ||||
|  | ||||
| [packages/mermaid/src/mermaid.ts:56](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/mermaid.ts#L56) | ||||
| [packages/mermaid/src/mermaid.ts:47](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/mermaid.ts#L47) | ||||
|   | ||||
| @@ -118,7 +118,7 @@ The siteConfig | ||||
|  | ||||
| #### Defined in | ||||
|  | ||||
| [packages/mermaid/src/config.ts:221](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.ts#L221) | ||||
| [packages/mermaid/src/config.ts:218](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/config.ts#L218) | ||||
|  | ||||
| --- | ||||
|  | ||||
|   | ||||
| @@ -14,7 +14,7 @@ | ||||
|  | ||||
| #### Defined in | ||||
|  | ||||
| [packages/mermaid/src/defaultConfig.ts:266](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/defaultConfig.ts#L266) | ||||
| [packages/mermaid/src/defaultConfig.ts:275](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/defaultConfig.ts#L275) | ||||
|  | ||||
| --- | ||||
|  | ||||
|   | ||||
| @@ -14,28 +14,15 @@ | ||||
|  | ||||
| - [DetailedError](../interfaces/mermaid.DetailedError.md) | ||||
| - [ExternalDiagramDefinition](../interfaces/mermaid.ExternalDiagramDefinition.md) | ||||
| - [LayoutData](../interfaces/mermaid.LayoutData.md) | ||||
| - [LayoutLoaderDefinition](../interfaces/mermaid.LayoutLoaderDefinition.md) | ||||
| - [Mermaid](../interfaces/mermaid.Mermaid.md) | ||||
| - [MermaidConfig](../interfaces/mermaid.MermaidConfig.md) | ||||
| - [ParseOptions](../interfaces/mermaid.ParseOptions.md) | ||||
| - [ParseResult](../interfaces/mermaid.ParseResult.md) | ||||
| - [RenderOptions](../interfaces/mermaid.RenderOptions.md) | ||||
| - [RenderResult](../interfaces/mermaid.RenderResult.md) | ||||
| - [RunOptions](../interfaces/mermaid.RunOptions.md) | ||||
|  | ||||
| ## Type Aliases | ||||
|  | ||||
| ### InternalHelpers | ||||
|  | ||||
| Ƭ **InternalHelpers**: typeof `internalHelpers` | ||||
|  | ||||
| #### Defined in | ||||
|  | ||||
| [packages/mermaid/src/internals.ts:33](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/internals.ts#L33) | ||||
|  | ||||
| --- | ||||
|  | ||||
| ### ParseErrorFunction | ||||
|  | ||||
| Ƭ **ParseErrorFunction**: (`err`: `string` | [`DetailedError`](../interfaces/mermaid.DetailedError.md) | `unknown`, `hash?`: `any`) => `void` | ||||
| @@ -59,26 +46,6 @@ | ||||
|  | ||||
| [packages/mermaid/src/Diagram.ts:10](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/Diagram.ts#L10) | ||||
|  | ||||
| --- | ||||
|  | ||||
| ### SVG | ||||
|  | ||||
| Ƭ **SVG**: `d3.Selection`<`SVGSVGElement`, `unknown`, `Element` | `null`, `unknown`> | ||||
|  | ||||
| #### Defined in | ||||
|  | ||||
| [packages/mermaid/src/diagram-api/types.ts:130](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/diagram-api/types.ts#L130) | ||||
|  | ||||
| --- | ||||
|  | ||||
| ### SVGGroup | ||||
|  | ||||
| Ƭ **SVGGroup**: `d3.Selection`<`SVGGElement`, `unknown`, `Element` | `null`, `unknown`> | ||||
|  | ||||
| #### Defined in | ||||
|  | ||||
| [packages/mermaid/src/diagram-api/types.ts:132](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/diagram-api/types.ts#L132) | ||||
|  | ||||
| ## Variables | ||||
|  | ||||
| ### default | ||||
| @@ -87,4 +54,4 @@ | ||||
|  | ||||
| #### Defined in | ||||
|  | ||||
| [packages/mermaid/src/mermaid.ts:440](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/mermaid.ts#L440) | ||||
| [packages/mermaid/src/mermaid.ts:430](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/mermaid.ts#L430) | ||||
|   | ||||
| @@ -56,10 +56,8 @@ To add an integration to this list, see the [Integrations - create page](./integ | ||||
|   - [SVG diagram generator](https://github.com/SimonKenyonShepard/mermaidjs-github-svg-generator) | ||||
| - [GitLab](https://docs.gitlab.com/ee/user/markdown.html#diagrams-and-flowcharts) ✅ | ||||
| - [Mermaid Plugin for JetBrains IDEs](https://plugins.jetbrains.com/plugin/20146-mermaid) | ||||
| - [MonsterWriter](https://www.monsterwriter.com/) ✅ | ||||
| - [Joplin](https://joplinapp.org) ✅ | ||||
| - [LiveBook](https://livebook.dev) ✅ | ||||
| - [Slidev](https://sli.dev) ✅ | ||||
| - [Tuleap](https://docs.tuleap.org/user-guide/writing-in-tuleap.html#graphs) ✅ | ||||
| - [Mermaid Flow Visual Editor](https://www.mermaidflow.app) ✅ | ||||
| - [Mermerd](https://github.com/KarnerTh/mermerd) | ||||
| @@ -135,7 +133,7 @@ Communication tools and platforms | ||||
| ### Wikis | ||||
|  | ||||
| - [DokuWiki](https://dokuwiki.org) | ||||
|   - [ComboStrap](https://combostrap.com/utility/create-diagram-with-mermaid-vh3ab9yj) | ||||
|   - [ComboStrap](https://combostrap.com/mermaid) | ||||
|   - [Mermaid Plugin](https://www.dokuwiki.org/plugin:mermaid) | ||||
| - [Foswiki](https://foswiki.org) | ||||
|   - [Mermaid Plugin](https://foswiki.org/Extensions/MermaidPlugin) | ||||
| @@ -210,7 +208,6 @@ Communication tools and platforms | ||||
|   - [gatsby-remark-mermaid](https://github.com/remcohaszing/gatsby-remark-mermaid) | ||||
| - [JSDoc](https://jsdoc.app/) | ||||
|   - [jsdoc-mermaid](https://github.com/Jellyvision/jsdoc-mermaid) | ||||
| - [Madness](https://madness.dannyb.co/) | ||||
| - [mdBook](https://rust-lang.github.io/mdBook/index.html) | ||||
|   - [mdbook-mermaid](https://github.com/badboy/mdbook-mermaid) | ||||
| - [MkDocs](https://www.mkdocs.org) | ||||
|   | ||||
| @@ -12,7 +12,7 @@ Try the Ultimate AI, Mermaid, and Visual Diagramming Suite by creating an accoun | ||||
|  | ||||
| <br /> | ||||
|  | ||||
| <a href="https://www.producthunt.com/products/mermaid-chart?utm_source=badge-follow&utm_medium=badge&utm_souce=badge-mermaid-chart" target="_blank"><img src="https://api.producthunt.com/widgets/embed-image/v1/follow.svg?product_id=552855&theme=light" alt="Mermaid Chart - A smarter way to create diagrams | Product Hunt" style="width: 250px; height: 54px;" width="250" height="54" /></a> | ||||
| <a href="https://www.producthunt.com/posts/mermaid-chart?utm_source=badge-featured&utm_medium=badge&utm_souce=badge-mermaid-chart" target="_blank"><img src="https://api.producthunt.com/widgets/embed-image/v1/featured.svg?post_id=416671&theme=light" alt="Mermaid Chart - A smarter way to create diagrams | Product Hunt" style="width: 250px; height: 54px;" width="250" height="54" /></a> | ||||
|  | ||||
| ## About | ||||
|  | ||||
|   | ||||
| @@ -83,139 +83,3 @@ Allows for the limited reconfiguration of a diagram just before it is rendered. | ||||
| ### [Theme Manipulation](../config/theming.md) | ||||
|  | ||||
| An application of using Directives to change [Themes](../config/theming.md). `Theme` is a value within Mermaid's configuration that dictates the color scheme for diagrams. | ||||
|  | ||||
| ### Layout and look | ||||
|  | ||||
| We've restructured how Mermaid renders diagrams, enabling new features like selecting layout and look. **Currently, this is supported for flowcharts and state diagrams**, with plans to extend support to all diagram types. | ||||
|  | ||||
| ### Selecting Diagram Looks | ||||
|  | ||||
| Mermaid offers a variety of styles or “looks” for your diagrams, allowing you to tailor the visual appearance to match your specific needs or preferences. Whether you prefer a hand-drawn or classic style, you can easily customize your diagrams. | ||||
|  | ||||
| **Available Looks:** | ||||
|  | ||||
| ``` | ||||
| •	Hand-Drawn Look: For a more personal, creative touch, the hand-drawn look brings a sketch-like quality to your diagrams. This style is perfect for informal settings or when you want to add a bit of personality to your diagrams. | ||||
| •	Classic Look: If you prefer the traditional Mermaid style, the classic look maintains the original appearance that many users are familiar with. It’s great for consistency across projects or when you want to keep the familiar aesthetic. | ||||
| ``` | ||||
|  | ||||
| **How to Select a Look:** | ||||
|  | ||||
| You can select a look by adding the look parameter in the metadata section of your Mermaid diagram code. Here’s an example: | ||||
|  | ||||
| ```mermaid-example | ||||
| --- | ||||
| config: | ||||
|   look: handDrawn | ||||
|   theme: neutral | ||||
| --- | ||||
| flowchart LR | ||||
|   A[Start] --> B{Decision} | ||||
|   B -->|Yes| C[Continue] | ||||
|   B -->|No| D[Stop] | ||||
| ``` | ||||
|  | ||||
| ```mermaid | ||||
| --- | ||||
| config: | ||||
|   look: handDrawn | ||||
|   theme: neutral | ||||
| --- | ||||
| flowchart LR | ||||
|   A[Start] --> B{Decision} | ||||
|   B -->|Yes| C[Continue] | ||||
|   B -->|No| D[Stop] | ||||
| ``` | ||||
|  | ||||
| #### Selecting Layout Algorithms | ||||
|  | ||||
| In addition to customizing the look of your diagrams, Mermaid Chart now allows you to choose different layout algorithms to better organize and present your diagrams, especially when dealing with more complex structures. The layout algorithm dictates how nodes and edges are arranged on the page. | ||||
|  | ||||
| #### Supported Layout Algorithms: | ||||
|  | ||||
| ``` | ||||
| •	Dagre (default): This is the classic layout algorithm that has been used in Mermaid for a long time. It provides a good balance of simplicity and visual clarity, making it ideal for most diagrams. | ||||
| •	ELK: For those who need more sophisticated layout capabilities, especially when working with large or intricate diagrams, the ELK (Eclipse Layout Kernel) layout offers advanced options. It provides a more optimized arrangement, potentially reducing overlapping and improving readability. This is not included out the box but needs to be added when integrating mermaid for sites/applications that want to have elk support. | ||||
| ``` | ||||
|  | ||||
| #### How to Select a Layout Algorithm: | ||||
|  | ||||
| You can specify the layout algorithm directly in the metadata section of your Mermaid diagram code. Here’s an example: | ||||
|  | ||||
| ```mermaid-example | ||||
| --- | ||||
| config: | ||||
|   layout: elk | ||||
|   look: handDrawn | ||||
|   theme: dark | ||||
| --- | ||||
| flowchart TB | ||||
|   A[Start] --> B{Decision} | ||||
|   B -->|Yes| C[Continue] | ||||
|   B -->|No| D[Stop] | ||||
| ``` | ||||
|  | ||||
| ```mermaid | ||||
| --- | ||||
| config: | ||||
|   layout: elk | ||||
|   look: handDrawn | ||||
|   theme: dark | ||||
| --- | ||||
| flowchart TB | ||||
|   A[Start] --> B{Decision} | ||||
|   B -->|Yes| C[Continue] | ||||
|   B -->|No| D[Stop] | ||||
| ``` | ||||
|  | ||||
| In this example, the `layout: elk` line configures the diagram to use the ELK layout algorithm, along with the hand drawn look and forest theme. | ||||
|  | ||||
| #### Customizing ELK Layout: | ||||
|  | ||||
| When using the ELK layout, you can further refine the diagram’s configuration, such as how nodes are placed and whether parallel edges should be combined: | ||||
|  | ||||
| - To combine parallel edges, use mergeEdges: true | false. | ||||
| - To configure node placement, use nodePlacementStrategy with the following options: | ||||
|   - SIMPLE | ||||
|   - NETWORK_SIMPLEX | ||||
|   - LINEAR_SEGMENTS | ||||
|   - BRANDES_KOEPF (default) | ||||
|  | ||||
| **Example configuration:** | ||||
|  | ||||
| ``` | ||||
| --- | ||||
| config: | ||||
|   layout: elk | ||||
|   elk: | ||||
|     mergeEdges: true | ||||
|     nodePlacementStrategy: LINEAR_SEGMENTS | ||||
| --- | ||||
| flowchart LR | ||||
|   A[Start] --> B{Choose Path} | ||||
|   B -->|Option 1| C[Path 1] | ||||
|   B -->|Option 2| D[Path 2] | ||||
|  | ||||
| #### Using Dagre Layout with Classic Look: | ||||
| ``` | ||||
|  | ||||
| Another example: | ||||
|  | ||||
| ``` | ||||
| --- | ||||
| config: | ||||
|   layout: dagre | ||||
|   look: classic | ||||
|   theme: default | ||||
| --- | ||||
|  | ||||
| flowchart LR | ||||
| A[Start] --> B{Choose Path} | ||||
| B -->|Option 1| C[Path 1] | ||||
| B -->|Option 2| D[Path 2] | ||||
|  | ||||
| ``` | ||||
|  | ||||
| These options give you the flexibility to create diagrams that not only look great but are also arranged to best suit your data’s structure and flow. | ||||
|  | ||||
| When integrating Mermaid, you can include look and layout configuration with the initialize call. This is also where you add the loading of elk. | ||||
|   | ||||
| @@ -6,48 +6,6 @@ | ||||
|  | ||||
| # Blog | ||||
|  | ||||
| ## [Mermaid v11 is out!](https://www.mermaidchart.com/blog/posts/mermaid-v11/) | ||||
|  | ||||
| 23 August 2024 · 2 mins | ||||
|  | ||||
| Mermaid v11 introduces advanced layout options, new diagram types, and enhanced customization features, thanks to the incredible contributions from our community. | ||||
|  | ||||
| ## [Mermaid Innovation - Introducing New Looks for Mermaid Diagrams](https://www.mermaidchart.com/blog/posts/mermaid-innovation-introducing-new-looks-for-mermaid-diagrams/) | ||||
|  | ||||
| 6 August 2024 ·3 mins | ||||
|  | ||||
| Discover the fresh new and unique Neo and Hand-Drawn looks for Mermaid Diagrams, while still offering the classic look you love. | ||||
|  | ||||
| ## [The Mermaid Chart Plugin for Jira: A How-To User Guide](https://www.mermaidchart.com/blog/posts/the-mermaid-chart-plugin-for-jira-a-how-to-user-guide/) | ||||
|  | ||||
| 31 July 2024 · 5 mins | ||||
|  | ||||
| The Mermaid Chart plugin for Jira has arrived! | ||||
|  | ||||
| ## [Mermaid AI Is Here to Change the Game For Diagram Creation](https://www.mermaidchart.com/blog/posts/mermaid-ai-is-here-to-change-the-game-for-diagram-creation/) | ||||
|  | ||||
| 22 July 2024 · 5 mins | ||||
|  | ||||
| The Mermaid AI chat interface | ||||
|  | ||||
| ## [How to Make a Sequence Diagram with Mermaid Chart](https://www.mermaidchart.com/blog/posts/how-to-make-a-sequence-diagram-in-mermaid-chart-step-by-step-guide/) | ||||
|  | ||||
| 8 July 2024 · 6 mins | ||||
|  | ||||
| Sequence diagrams are important for communicating complex systems in a clear and concise manner. | ||||
|  | ||||
| ## [How to Use the New “Comments” Feature in Mermaid Chart](https://www.mermaidchart.com/blog/posts/how-to-use-the-new-comments-feature-in-mermaid-chart/) | ||||
|  | ||||
| 2 July 2024 · 3 mins | ||||
|  | ||||
| How to Use the New Comments Feature in Mermaid Chart | ||||
|  | ||||
| ## [How to Use the official Mermaid Chart for Confluence app](https://www.mermaidchart.com/blog/posts/how-to-use-the-official-mermaid-chart-for-confluence-app/) | ||||
|  | ||||
| 21 May 2024 · 4 mins | ||||
|  | ||||
| It doesn’t matter if you’re a data enthusiast, software engineer, or visual storyteller; our Confluence app can allow you to embed Mermaid Chart diagrams — and dynamically edit them — within your Confluence pages. | ||||
|  | ||||
| ## [How to Choose the Right Documentation Software](https://www.mermaidchart.com/blog/posts/how-to-choose-the-right-documentation-software/) | ||||
|  | ||||
| 7 May 2024 · 5 mins | ||||
|   | ||||
| @@ -136,7 +136,7 @@ Cardinality is a property that describes how many elements of another entity can | ||||
| |      1+      |      1+       | One or more  | | ||||
| | zero or more | zero or more  | Zero or more | | ||||
| | zero or many | zero or many  | Zero or more | | ||||
| |   many(0)    |    many(0)    | Zero or more | | ||||
| |   many(0)    |    many(1)    | Zero or more | | ||||
| |      0+      |      0+       | Zero or more | | ||||
| |   only one   |   only one    | Exactly one  | | ||||
| |      1       |       1       | Exactly one  | | ||||
|   | ||||
| @@ -172,7 +172,7 @@ The `title` is an _optional_ string to be displayed at the top of the Gantt char | ||||
| The `excludes` is an _optional_ attribute that accepts specific dates in YYYY-MM-DD format, days of the week ("sunday") or "weekends", but not the word "weekdays". | ||||
| These date will be marked on the graph, and be excluded from the duration calculation of tasks. Meaning that if there are excluded dates during a task interval, the number of 'skipped' days will be added to the end of the task to ensure the duration is as specified in the code. | ||||
|  | ||||
| #### Weekend (v\11.0.0+) | ||||
| #### Weekend (v\<MERMAID_RELEASE_VERSION>+) | ||||
|  | ||||
| When excluding weekends, it is possible to configure the weekends to be either Friday and Saturday or Saturday and Sunday. By default weekends are Saturday and Sunday. | ||||
| To define the weekend start day, there is an _optional_ attribute `weekend` that can be added in a new line followed by either `friday` or `saturday`. | ||||
|   | ||||
| @@ -918,7 +918,7 @@ Usage example: | ||||
|        commit | ||||
| ``` | ||||
|  | ||||
| ### Bottom to Top (`BT:`) (v11.0.0+) | ||||
| ### Bottom to Top (`BT:`) (v\<MERMAID_RELEASE_VERSION>+) | ||||
|  | ||||
| In `BT` (**Bottom-to-Top**) orientation, the commits run from bottom to top of the graph and branches are arranged side-by-side. | ||||
|  | ||||
|   | ||||
| @@ -4,7 +4,7 @@ | ||||
| > | ||||
| > ## Please edit the corresponding file in [/packages/mermaid/src/docs/syntax/packet.md](../../packages/mermaid/src/docs/syntax/packet.md). | ||||
|  | ||||
| # Packet Diagram (v11.0.0+) | ||||
| # Packet Diagram (v\<MERMAID_RELEASE_VERSION>+) | ||||
|  | ||||
| ## Introduction | ||||
|  | ||||
| @@ -12,7 +12,7 @@ A packet diagram is a visual representation used to illustrate the structure and | ||||
|  | ||||
| ## Usage | ||||
|  | ||||
| This diagram type is particularly useful for developers, network engineers, educators, and students who require a clear and concise way to represent the structure of network packets. | ||||
| This diagram type is particularly useful for network engineers, educators, and students who require a clear and concise way to represent the structure of network packets. | ||||
|  | ||||
| ## Syntax | ||||
|  | ||||
|   | ||||
| @@ -208,18 +208,18 @@ Messages can be of two displayed either solid or with a dotted line. | ||||
|  | ||||
| There are ten types of arrows currently supported: | ||||
|  | ||||
| | Type     | Description                                          | | ||||
| | -------- | ---------------------------------------------------- | | ||||
| | `->`     | Solid line without arrow                             | | ||||
| | `-->`    | Dotted line without arrow                            | | ||||
| | `->>`    | Solid line with arrowhead                            | | ||||
| | `-->>`   | Dotted line with arrowhead                           | | ||||
| | `<<->>`  | Solid line with bidirectional arrowheads (v11.0.0+)  | | ||||
| | `<<-->>` | Dotted line with bidirectional arrowheads (v11.0.0+) | | ||||
| | `-x`     | Solid line with a cross at the end                   | | ||||
| | `--x`    | Dotted line with a cross at the end.                 | | ||||
| | `-)`     | Solid line with an open arrow at the end (async)     | | ||||
| | `--)`    | Dotted line with a open arrow at the end (async)     | | ||||
| | Type     | Description                                                              | | ||||
| | -------- | ------------------------------------------------------------------------ | | ||||
| | `->`     | Solid line without arrow                                                 | | ||||
| | `-->`    | Dotted line without arrow                                                | | ||||
| | `->>`    | Solid line with arrowhead                                                | | ||||
| | `-->>`   | Dotted line with arrowhead                                               | | ||||
| | `<<->>`  | Solid line with bidirectional arrowheads (v\<MERMAID_RELEASE_VERSION>+)  | | ||||
| | `<<-->>` | Dotted line with bidirectional arrowheads (v\<MERMAID_RELEASE_VERSION>+) | | ||||
| | `-x`     | Solid line with a cross at the end                                       | | ||||
| | `--x`    | Dotted line with a cross at the end.                                     | | ||||
| | `-)`     | Solid line with an open arrow at the end (async)                         | | ||||
| | `--)`    | Dotted line with a open arrow at the end (async)                         | | ||||
|  | ||||
| ## Activations | ||||
|  | ||||
|   | ||||
| @@ -483,8 +483,8 @@ a _[valid CSS property name](https://www.w3.org/TR/CSS/#properties)_ followed by | ||||
|  | ||||
| Here is an example of a classDef with just one property-value pair: | ||||
|  | ||||
| ```txt | ||||
| classDef movement font-style:italic; | ||||
| ``` | ||||
|     classDef movement font-style:italic; | ||||
| ``` | ||||
|  | ||||
| where | ||||
| @@ -496,8 +496,8 @@ If you want to have more than one _property-value pair_ then you put a comma (`, | ||||
|  | ||||
| Here is an example with three property-value pairs: | ||||
|  | ||||
| ```txt | ||||
| classDef badBadEvent fill:#f00,color:white,font-weight:bold,stroke-width:2px,stroke:yellow | ||||
| ``` | ||||
|     classDef badBadEvent fill:#f00,color:white,font-weight:bold,stroke-width:2px,stroke:yellow | ||||
| ``` | ||||
|  | ||||
| where | ||||
| @@ -522,7 +522,7 @@ There are two ways to apply a `classDef` style to a state: | ||||
| A `class` statement tells Mermaid to apply the named classDef to one or more classes. The form is: | ||||
|  | ||||
| ```txt | ||||
| class [one or more state names, separated by commas] [name of a style defined with classDef] | ||||
|     class [one or more state names, separated by commas] [name of a style defined with classDef] | ||||
| ``` | ||||
|  | ||||
| Here is an example applying the `badBadEvent` style to a state named `Crash`: | ||||
|   | ||||
| @@ -109,7 +109,6 @@ export default tseslint.config( | ||||
|       '@typescript-eslint/no-unsafe-member-access': 'off', | ||||
|       '@typescript-eslint/no-unsafe-return': 'off', | ||||
|       '@typescript-eslint/only-throw-error': 'warn', | ||||
|       '@typescript-eslint/prefer-nullish-coalescing': 'warn', | ||||
|       '@typescript-eslint/prefer-promise-reject-errors': 'warn', | ||||
|       // END | ||||
|       'json/*': ['error', 'allowComments'], | ||||
|   | ||||
							
								
								
									
										16
									
								
								package.json
									
									
									
									
									
								
							
							
						
						
									
										16
									
								
								package.json
									
									
									
									
									
								
							| @@ -4,7 +4,7 @@ | ||||
|   "version": "10.2.4", | ||||
|   "description": "Markdownish syntax for generating flowcharts, sequence diagrams, class diagrams, gantt charts and git graphs.", | ||||
|   "type": "module", | ||||
|   "packageManager": "pnpm@9.7.1+sha512.faf344af2d6ca65c4c5c8c2224ea77a81a5e8859cbc4e06b1511ddce2f0151512431dd19e6aff31f2c6a8f5f2aced9bd2273e1fed7dd4de1868984059d2c4247", | ||||
|   "packageManager": "pnpm@9.4.0+sha512.f549b8a52c9d2b8536762f99c0722205efc5af913e77835dbccc3b0b0b2ca9e7dc8022b78062c17291c48e88749c70ce88eb5a74f1fa8c4bf5e18bb46c8bd83a", | ||||
|   "keywords": [ | ||||
|     "diagram", | ||||
|     "markdown", | ||||
| @@ -24,9 +24,7 @@ | ||||
|     "dev": "tsx .esbuild/server.ts", | ||||
|     "dev:vite": "tsx .vite/server.ts", | ||||
|     "dev:coverage": "pnpm coverage:cypress:clean && VITE_COVERAGE=true pnpm dev:vite", | ||||
|     "copy-readme": "cpy './README.*' ./packages/mermaid/ --cwd=.", | ||||
|     "changeset:version": "changeset version && pnpm build && pnpm --filter mermaid run docs:release-version && pnpm --filter mermaid run docs:build && git add --all", | ||||
|     "changeset:publish": "pnpm copy-readme && changeset publish", | ||||
|     "release": "pnpm build", | ||||
|     "lint": "eslint --quiet --stats --cache --cache-strategy content . && pnpm lint:jison && prettier --cache --check .", | ||||
|     "lint:fix": "eslint --cache --cache-strategy content --fix . && prettier --write . && tsx scripts/fixCSpell.ts", | ||||
|     "lint:jison": "tsx ./scripts/jison/lint.mts", | ||||
| @@ -42,6 +40,7 @@ | ||||
|     "test": "pnpm lint && vitest run", | ||||
|     "test:watch": "vitest --watch", | ||||
|     "test:coverage": "vitest --coverage", | ||||
|     "prepublishOnly": "pnpm build && pnpm test", | ||||
|     "prepare": "husky install && pnpm build", | ||||
|     "pre-commit": "lint-staged" | ||||
|   }, | ||||
| @@ -62,10 +61,8 @@ | ||||
|     ] | ||||
|   }, | ||||
|   "devDependencies": { | ||||
|     "@applitools/eyes-cypress": "^3.44.4", | ||||
|     "@applitools/eyes-cypress": "^3.42.3", | ||||
|     "@argos-ci/cypress": "^2.1.0", | ||||
|     "@changesets/changelog-github": "^0.5.0", | ||||
|     "@changesets/cli": "^2.27.7", | ||||
|     "@cspell/eslint-plugin": "^8.8.4", | ||||
|     "@cypress/code-coverage": "^3.12.30", | ||||
|     "@eslint/js": "^9.4.0", | ||||
| @@ -85,10 +82,9 @@ | ||||
|     "chokidar": "^3.6.0", | ||||
|     "concurrently": "^8.2.2", | ||||
|     "cors": "^2.8.5", | ||||
|     "cpy-cli": "^5.0.0", | ||||
|     "cross-env": "^7.0.3", | ||||
|     "cspell": "^8.6.0", | ||||
|     "cypress": "^13.11.0", | ||||
|     "cypress": "^13.7.1", | ||||
|     "cypress-image-snapshot": "^4.0.1", | ||||
|     "esbuild": "^0.21.5", | ||||
|     "eslint": "^9.4.0", | ||||
| @@ -102,7 +98,7 @@ | ||||
|     "eslint-plugin-markdown": "^5.0.0", | ||||
|     "eslint-plugin-no-only-tests": "^3.1.0", | ||||
|     "eslint-plugin-tsdoc": "^0.3.0", | ||||
|     "eslint-plugin-unicorn": "^55.0.0", | ||||
|     "eslint-plugin-unicorn": "^54.0.0", | ||||
|     "express": "^4.19.1", | ||||
|     "globals": "^15.4.0", | ||||
|     "globby": "^14.0.1", | ||||
|   | ||||
| @@ -1,7 +1,6 @@ | ||||
| { | ||||
|   "name": "@mermaid-js/mermaid-example-diagram", | ||||
|   "version": "9.3.0", | ||||
|   "private": true, | ||||
|   "description": "Example of external diagram module for MermaidJS.", | ||||
|   "module": "dist/mermaid-example-diagram.core.mjs", | ||||
|   "types": "dist/detector.d.ts", | ||||
| @@ -19,7 +18,9 @@ | ||||
|     "example", | ||||
|     "mermaid" | ||||
|   ], | ||||
|   "scripts": {}, | ||||
|   "scripts": { | ||||
|     "prepublishOnly": "pnpm -w run build" | ||||
|   }, | ||||
|   "repository": { | ||||
|     "type": "git", | ||||
|     "url": "https://github.com/mermaid-js/mermaid" | ||||
|   | ||||
							
								
								
									
										45
									
								
								packages/mermaid-flowchart-elk/package.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										45
									
								
								packages/mermaid-flowchart-elk/package.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,45 @@ | ||||
| { | ||||
|   "name": "@mermaid-js/flowchart-elk", | ||||
|   "version": "1.0.0-rc.1", | ||||
|   "description": "Flowchart plugin for mermaid with ELK layout", | ||||
|   "module": "dist/mermaid-flowchart-elk.core.mjs", | ||||
|   "types": "dist/packages/mermaid-flowchart-elk/src/detector.d.ts", | ||||
|   "type": "module", | ||||
|   "exports": { | ||||
|     ".": { | ||||
|       "import": "./dist/mermaid-flowchart-elk.core.mjs", | ||||
|       "types": "./dist/packages/mermaid-flowchart-elk/src/detector.d.ts" | ||||
|     }, | ||||
|     "./*": "./*" | ||||
|   }, | ||||
|   "keywords": [ | ||||
|     "diagram", | ||||
|     "markdown", | ||||
|     "flowchart", | ||||
|     "elk", | ||||
|     "mermaid" | ||||
|   ], | ||||
|   "scripts": { | ||||
|     "prepublishOnly": "pnpm -w run build" | ||||
|   }, | ||||
|   "repository": { | ||||
|     "type": "git", | ||||
|     "url": "https://github.com/mermaid-js/mermaid" | ||||
|   }, | ||||
|   "author": "Knut Sveidqvist", | ||||
|   "license": "MIT", | ||||
|   "dependencies": { | ||||
|     "d3": "^7.9.0", | ||||
|     "dagre-d3-es": "7.0.10", | ||||
|     "elkjs": "^0.9.2", | ||||
|     "khroma": "^2.1.0" | ||||
|   }, | ||||
|   "devDependencies": { | ||||
|     "concurrently": "^8.2.2", | ||||
|     "mermaid": "workspace:^", | ||||
|     "rimraf": "^5.0.5" | ||||
|   }, | ||||
|   "files": [ | ||||
|     "dist" | ||||
|   ] | ||||
| } | ||||
							
								
								
									
										75
									
								
								packages/mermaid-flowchart-elk/src/detector.spec.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										75
									
								
								packages/mermaid-flowchart-elk/src/detector.spec.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,75 @@ | ||||
| import plugin from './detector.js'; | ||||
| import { describe, it } from 'vitest'; | ||||
|  | ||||
| const { detector } = plugin; | ||||
|  | ||||
| describe('flowchart-elk detector', () => { | ||||
|   it('should fail for dagre-d3', () => { | ||||
|     expect( | ||||
|       detector('flowchart', { | ||||
|         flowchart: { | ||||
|           defaultRenderer: 'dagre-d3', | ||||
|         }, | ||||
|       }) | ||||
|     ).toBe(false); | ||||
|   }); | ||||
|   it('should fail for dagre-wrapper', () => { | ||||
|     expect( | ||||
|       detector('flowchart', { | ||||
|         flowchart: { | ||||
|           defaultRenderer: 'dagre-wrapper', | ||||
|         }, | ||||
|       }) | ||||
|     ).toBe(false); | ||||
|   }); | ||||
|   it('should succeed for elk', () => { | ||||
|     expect( | ||||
|       detector('flowchart', { | ||||
|         flowchart: { | ||||
|           defaultRenderer: 'elk', | ||||
|         }, | ||||
|       }) | ||||
|     ).toBe(true); | ||||
|     expect( | ||||
|       detector('graph', { | ||||
|         flowchart: { | ||||
|           defaultRenderer: 'elk', | ||||
|         }, | ||||
|       }) | ||||
|     ).toBe(true); | ||||
|   }); | ||||
|  | ||||
|   // The error from the issue was reproduced with mindmap, so this is just an example | ||||
|   // what matters is the keyword somewhere inside graph definition | ||||
|   it('should check only the beginning of the line in search of keywords', () => { | ||||
|     expect( | ||||
|       detector('mindmap ["Descendant node in flowchart"]', { | ||||
|         flowchart: { | ||||
|           defaultRenderer: 'elk', | ||||
|         }, | ||||
|       }) | ||||
|     ).toBe(false); | ||||
|  | ||||
|     expect( | ||||
|       detector('mindmap ["Descendant node in graph"]', { | ||||
|         flowchart: { | ||||
|           defaultRenderer: 'elk', | ||||
|         }, | ||||
|       }) | ||||
|     ).toBe(false); | ||||
|   }); | ||||
|  | ||||
|   it('should detect flowchart-elk', () => { | ||||
|     expect(detector('flowchart-elk')).toBe(true); | ||||
|   }); | ||||
|  | ||||
|   it('should not detect class with defaultRenderer set to elk', () => { | ||||
|     expect( | ||||
|       detector('class', { | ||||
|         flowchart: { | ||||
|           defaultRenderer: 'elk', | ||||
|         }, | ||||
|       }) | ||||
|     ).toBe(false); | ||||
|   }); | ||||
| }); | ||||
							
								
								
									
										32
									
								
								packages/mermaid-flowchart-elk/src/detector.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										32
									
								
								packages/mermaid-flowchart-elk/src/detector.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,32 @@ | ||||
| import type { | ||||
|   ExternalDiagramDefinition, | ||||
|   DiagramDetector, | ||||
|   DiagramLoader, | ||||
| } from '../../mermaid/src/diagram-api/types.js'; | ||||
|  | ||||
| const id = 'flowchart-elk'; | ||||
|  | ||||
| const detector: DiagramDetector = (txt, config): boolean => { | ||||
|   if ( | ||||
|     // If diagram explicitly states flowchart-elk | ||||
|     /^\s*flowchart-elk/.test(txt) || | ||||
|     // If a flowchart/graph diagram has their default renderer set to elk | ||||
|     (/^\s*(flowchart|graph)/.test(txt) && config?.flowchart?.defaultRenderer === 'elk') | ||||
|   ) { | ||||
|     return true; | ||||
|   } | ||||
|   return false; | ||||
| }; | ||||
|  | ||||
| const loader: DiagramLoader = async () => { | ||||
|   const { diagram } = await import('./diagram-definition.js'); | ||||
|   return { id, diagram }; | ||||
| }; | ||||
|  | ||||
| const plugin: ExternalDiagramDefinition = { | ||||
|   id, | ||||
|   detector, | ||||
|   loader, | ||||
| }; | ||||
|  | ||||
| export default plugin; | ||||
							
								
								
									
										12
									
								
								packages/mermaid-flowchart-elk/src/diagram-definition.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								packages/mermaid-flowchart-elk/src/diagram-definition.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,12 @@ | ||||
| // @ts-ignore: JISON typing missing | ||||
| import parser from '../../mermaid/src/diagrams/flowchart/parser/flow.jison'; | ||||
| import db from '../../mermaid/src/diagrams/flowchart/flowDb.js'; | ||||
| import styles from '../../mermaid/src/diagrams/flowchart/styles.js'; | ||||
| import renderer from './flowRenderer-elk.js'; | ||||
|  | ||||
| export const diagram = { | ||||
|   db, | ||||
|   renderer, | ||||
|   parser, | ||||
|   styles, | ||||
| }; | ||||
							
								
								
									
										889
									
								
								packages/mermaid-flowchart-elk/src/flowRenderer-elk.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										889
									
								
								packages/mermaid-flowchart-elk/src/flowRenderer-elk.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,889 @@ | ||||
| import { select, line, curveLinear } from 'd3'; | ||||
| import { insertNode } from '../../mermaid/src/dagre-wrapper/nodes.js'; | ||||
| import insertMarkers from '../../mermaid/src/dagre-wrapper/markers.js'; | ||||
| import { insertEdgeLabel } from '../../mermaid/src/dagre-wrapper/edges.js'; | ||||
| import { findCommonAncestor } from './render-utils.js'; | ||||
| import { labelHelper } from '../../mermaid/src/dagre-wrapper/shapes/util.js'; | ||||
| import { getConfig } from '../../mermaid/src/config.js'; | ||||
| import { log } from '../../mermaid/src/logger.js'; | ||||
| import utils from '../../mermaid/src/utils.js'; | ||||
| import { setupGraphViewbox } from '../../mermaid/src/setupGraphViewbox.js'; | ||||
| import common from '../../mermaid/src/diagrams/common/common.js'; | ||||
| import { interpolateToCurve, getStylesFromArray } from '../../mermaid/src/utils.js'; | ||||
| import ELK from 'elkjs/lib/elk.bundled.js'; | ||||
| import { getLineFunctionsWithOffset } from '../../mermaid/src/utils/lineWithOffset.js'; | ||||
| import { addEdgeMarkers } from '../../mermaid/src/dagre-wrapper/edgeMarker.js'; | ||||
|  | ||||
| const elk = new ELK(); | ||||
|  | ||||
| let portPos = {}; | ||||
|  | ||||
| const conf = {}; | ||||
| export const setConf = function (cnf) { | ||||
|   const keys = Object.keys(cnf); | ||||
|   for (const key of keys) { | ||||
|     conf[key] = cnf[key]; | ||||
|   } | ||||
| }; | ||||
|  | ||||
| let nodeDb = {}; | ||||
|  | ||||
| // /** | ||||
| //  * Function that adds the vertices found during parsing to the graph to be rendered. | ||||
| //  * | ||||
| //  * @param vert Object containing the vertices. | ||||
| //  * @param g The graph that is to be drawn. | ||||
| //  * @param svgId | ||||
| //  * @param root | ||||
| //  * @param doc | ||||
| //  * @param diagObj | ||||
| //  */ | ||||
| export const addVertices = async function (vert, svgId, root, doc, diagObj, parentLookupDb, graph) { | ||||
|   const svg = root.select(`[id="${svgId}"]`); | ||||
|   const nodes = svg.insert('g').attr('class', 'nodes'); | ||||
|   const keys = [...vert.keys()]; | ||||
|  | ||||
|   // Iterate through each item in the vertex object (containing all the vertices found) in the graph definition | ||||
|   await Promise.all( | ||||
|     keys.map(async function (id) { | ||||
|       const vertex = vert.get(id); | ||||
|  | ||||
|       /** | ||||
|        * Variable for storing the classes for the vertex | ||||
|        * | ||||
|        * @type {string} | ||||
|        */ | ||||
|       let classStr = 'default'; | ||||
|       if (vertex.classes.length > 0) { | ||||
|         classStr = vertex.classes.join(' '); | ||||
|       } | ||||
|       classStr = classStr + ' flowchart-label'; | ||||
|       const styles = getStylesFromArray(vertex.styles); | ||||
|  | ||||
|       // Use vertex id as text in the box if no text is provided by the graph definition | ||||
|       let vertexText = vertex.text !== undefined ? vertex.text : vertex.id; | ||||
|  | ||||
|       // We create a SVG label, either by delegating to addHtmlLabel or manually | ||||
|       const labelData = { width: 0, height: 0 }; | ||||
|  | ||||
|       const ports = [ | ||||
|         { | ||||
|           id: vertex.id + '-west', | ||||
|           layoutOptions: { | ||||
|             'port.side': 'WEST', | ||||
|           }, | ||||
|         }, | ||||
|         { | ||||
|           id: vertex.id + '-east', | ||||
|           layoutOptions: { | ||||
|             'port.side': 'EAST', | ||||
|           }, | ||||
|         }, | ||||
|         { | ||||
|           id: vertex.id + '-south', | ||||
|           layoutOptions: { | ||||
|             'port.side': 'SOUTH', | ||||
|           }, | ||||
|         }, | ||||
|         { | ||||
|           id: vertex.id + '-north', | ||||
|           layoutOptions: { | ||||
|             'port.side': 'NORTH', | ||||
|           }, | ||||
|         }, | ||||
|       ]; | ||||
|  | ||||
|       let radius = 0; | ||||
|       let _shape = ''; | ||||
|       let layoutOptions = {}; | ||||
|       // Set the shape based parameters | ||||
|       switch (vertex.type) { | ||||
|         case 'round': | ||||
|           radius = 5; | ||||
|           _shape = 'rect'; | ||||
|           break; | ||||
|         case 'square': | ||||
|           _shape = 'rect'; | ||||
|           break; | ||||
|         case 'diamond': | ||||
|           _shape = 'question'; | ||||
|           layoutOptions = { | ||||
|             portConstraints: 'FIXED_SIDE', | ||||
|           }; | ||||
|           break; | ||||
|         case 'hexagon': | ||||
|           _shape = 'hexagon'; | ||||
|           break; | ||||
|         case 'odd': | ||||
|           _shape = 'rect_left_inv_arrow'; | ||||
|           break; | ||||
|         case 'lean_right': | ||||
|           _shape = 'lean_right'; | ||||
|           break; | ||||
|         case 'lean_left': | ||||
|           _shape = 'lean_left'; | ||||
|           break; | ||||
|         case 'trapezoid': | ||||
|           _shape = 'trapezoid'; | ||||
|           break; | ||||
|         case 'inv_trapezoid': | ||||
|           _shape = 'inv_trapezoid'; | ||||
|           break; | ||||
|         case 'odd_right': | ||||
|           _shape = 'rect_left_inv_arrow'; | ||||
|           break; | ||||
|         case 'circle': | ||||
|           _shape = 'circle'; | ||||
|           break; | ||||
|         case 'ellipse': | ||||
|           _shape = 'ellipse'; | ||||
|           break; | ||||
|         case 'stadium': | ||||
|           _shape = 'stadium'; | ||||
|           break; | ||||
|         case 'subroutine': | ||||
|           _shape = 'subroutine'; | ||||
|           break; | ||||
|         case 'cylinder': | ||||
|           _shape = 'cylinder'; | ||||
|           break; | ||||
|         case 'group': | ||||
|           _shape = 'rect'; | ||||
|           break; | ||||
|         case 'doublecircle': | ||||
|           _shape = 'doublecircle'; | ||||
|           break; | ||||
|         default: | ||||
|           _shape = 'rect'; | ||||
|       } | ||||
|  | ||||
|       // Add the node | ||||
|       const node = { | ||||
|         labelStyle: styles.labelStyle, | ||||
|         shape: _shape, | ||||
|         labelText: vertexText, | ||||
|         labelType: vertex.labelType, | ||||
|         rx: radius, | ||||
|         ry: radius, | ||||
|         class: classStr, | ||||
|         style: styles.style, | ||||
|         id: vertex.id, | ||||
|         link: vertex.link, | ||||
|         linkTarget: vertex.linkTarget, | ||||
|         tooltip: diagObj.db.getTooltip(vertex.id) || '', | ||||
|         domId: diagObj.db.lookUpDomId(vertex.id), | ||||
|         haveCallback: vertex.haveCallback, | ||||
|         width: vertex.type === 'group' ? 500 : undefined, | ||||
|         dir: vertex.dir, | ||||
|         type: vertex.type, | ||||
|         props: vertex.props, | ||||
|         padding: getConfig().flowchart.padding, | ||||
|       }; | ||||
|       let boundingBox; | ||||
|       let nodeEl; | ||||
|  | ||||
|       // Add the element to the DOM | ||||
|       if (node.type !== 'group') { | ||||
|         nodeEl = await insertNode(nodes, node, vertex.dir); | ||||
|         boundingBox = nodeEl.node().getBBox(); | ||||
|       } else { | ||||
|         const { shapeSvg, bbox } = await labelHelper(nodes, node, undefined, true); | ||||
|         labelData.width = bbox.width; | ||||
|         labelData.wrappingWidth = getConfig().flowchart.wrappingWidth; | ||||
|         labelData.height = bbox.height; | ||||
|         labelData.labelNode = shapeSvg.node(); | ||||
|         node.labelData = labelData; | ||||
|       } | ||||
|       // const { shapeSvg, bbox } = await labelHelper(svg, node, undefined, true); | ||||
|  | ||||
|       const data = { | ||||
|         id: vertex.id, | ||||
|         ports: vertex.type === 'diamond' ? ports : [], | ||||
|         // labelStyle: styles.labelStyle, | ||||
|         // shape: _shape, | ||||
|         layoutOptions, | ||||
|         labelText: vertexText, | ||||
|         labelData, | ||||
|         // labels: [{ text: vertexText }], | ||||
|         // rx: radius, | ||||
|         // ry: radius, | ||||
|         // class: classStr, | ||||
|         // style: styles.style, | ||||
|         // link: vertex.link, | ||||
|         // linkTarget: vertex.linkTarget, | ||||
|         // tooltip: diagObj.db.getTooltip(vertex.id) || '', | ||||
|         domId: diagObj.db.lookUpDomId(vertex.id), | ||||
|         // haveCallback: vertex.haveCallback, | ||||
|         width: boundingBox?.width, | ||||
|         height: boundingBox?.height, | ||||
|         // dir: vertex.dir, | ||||
|         type: vertex.type, | ||||
|         // props: vertex.props, | ||||
|         // padding: getConfig().flowchart.padding, | ||||
|         // boundingBox, | ||||
|         el: nodeEl, | ||||
|         parent: parentLookupDb.parentById[vertex.id], | ||||
|       }; | ||||
|       // if (!Object.keys(parentLookupDb.childrenById).includes(vertex.id)) { | ||||
|       // graph.children.push({ | ||||
|       //   ...data, | ||||
|       // }); | ||||
|       // } | ||||
|       nodeDb[node.id] = data; | ||||
|       // log.trace('setNode', { | ||||
|       //   labelStyle: styles.labelStyle, | ||||
|       //   shape: _shape, | ||||
|       //   labelText: vertexText, | ||||
|       //   rx: radius, | ||||
|       //   ry: radius, | ||||
|       //   class: classStr, | ||||
|       //   style: styles.style, | ||||
|       //   id: vertex.id, | ||||
|       //   domId: diagObj.db.lookUpDomId(vertex.id), | ||||
|       //   width: vertex.type === 'group' ? 500 : undefined, | ||||
|       //   type: vertex.type, | ||||
|       //   dir: vertex.dir, | ||||
|       //   props: vertex.props, | ||||
|       //   padding: getConfig().flowchart.padding, | ||||
|       //   parent: parentLookupDb.parentById[vertex.id], | ||||
|       // }); | ||||
|     }) | ||||
|   ); | ||||
|   return graph; | ||||
| }; | ||||
|  | ||||
| const getNextPosition = (position, edgeDirection, graphDirection) => { | ||||
|   const portPos = { | ||||
|     TB: { | ||||
|       in: { | ||||
|         north: 'north', | ||||
|       }, | ||||
|       out: { | ||||
|         south: 'west', | ||||
|         west: 'east', | ||||
|         east: 'south', | ||||
|       }, | ||||
|     }, | ||||
|     LR: { | ||||
|       in: { | ||||
|         west: 'west', | ||||
|       }, | ||||
|       out: { | ||||
|         east: 'south', | ||||
|         south: 'north', | ||||
|         north: 'east', | ||||
|       }, | ||||
|     }, | ||||
|     RL: { | ||||
|       in: { | ||||
|         east: 'east', | ||||
|       }, | ||||
|       out: { | ||||
|         west: 'north', | ||||
|         north: 'south', | ||||
|         south: 'west', | ||||
|       }, | ||||
|     }, | ||||
|     BT: { | ||||
|       in: { | ||||
|         south: 'south', | ||||
|       }, | ||||
|       out: { | ||||
|         north: 'east', | ||||
|         east: 'west', | ||||
|         west: 'north', | ||||
|       }, | ||||
|     }, | ||||
|   }; | ||||
|   portPos.TD = portPos.TB; | ||||
|   return portPos[graphDirection][edgeDirection][position]; | ||||
|   // return 'south'; | ||||
| }; | ||||
|  | ||||
| const getNextPort = (node, edgeDirection, graphDirection) => { | ||||
|   log.info('getNextPort', { node, edgeDirection, graphDirection }); | ||||
|   if (!portPos[node]) { | ||||
|     switch (graphDirection) { | ||||
|       case 'TB': | ||||
|       case 'TD': | ||||
|         portPos[node] = { | ||||
|           inPosition: 'north', | ||||
|           outPosition: 'south', | ||||
|         }; | ||||
|         break; | ||||
|       case 'BT': | ||||
|         portPos[node] = { | ||||
|           inPosition: 'south', | ||||
|           outPosition: 'north', | ||||
|         }; | ||||
|         break; | ||||
|       case 'RL': | ||||
|         portPos[node] = { | ||||
|           inPosition: 'east', | ||||
|           outPosition: 'west', | ||||
|         }; | ||||
|         break; | ||||
|       case 'LR': | ||||
|         portPos[node] = { | ||||
|           inPosition: 'west', | ||||
|           outPosition: 'east', | ||||
|         }; | ||||
|         break; | ||||
|     } | ||||
|   } | ||||
|   const result = edgeDirection === 'in' ? portPos[node].inPosition : portPos[node].outPosition; | ||||
|  | ||||
|   if (edgeDirection === 'in') { | ||||
|     portPos[node].inPosition = getNextPosition( | ||||
|       portPos[node].inPosition, | ||||
|       edgeDirection, | ||||
|       graphDirection | ||||
|     ); | ||||
|   } else { | ||||
|     portPos[node].outPosition = getNextPosition( | ||||
|       portPos[node].outPosition, | ||||
|       edgeDirection, | ||||
|       graphDirection | ||||
|     ); | ||||
|   } | ||||
|   return result; | ||||
| }; | ||||
|  | ||||
| const getEdgeStartEndPoint = (edge, dir) => { | ||||
|   let source = edge.start; | ||||
|   let target = edge.end; | ||||
|  | ||||
|   // Save the original source and target | ||||
|   const sourceId = source; | ||||
|   const targetId = target; | ||||
|  | ||||
|   const startNode = nodeDb[source]; | ||||
|   const endNode = nodeDb[target]; | ||||
|  | ||||
|   if (!startNode || !endNode) { | ||||
|     return { source, target }; | ||||
|   } | ||||
|  | ||||
|   if (startNode.type === 'diamond') { | ||||
|     source = `${source}-${getNextPort(source, 'out', dir)}`; | ||||
|   } | ||||
|  | ||||
|   if (endNode.type === 'diamond') { | ||||
|     target = `${target}-${getNextPort(target, 'in', dir)}`; | ||||
|   } | ||||
|  | ||||
|   // Add the edge to the graph | ||||
|   return { source, target, sourceId, targetId }; | ||||
| }; | ||||
|  | ||||
| /** | ||||
|  * Add edges to graph based on parsed graph definition | ||||
|  * | ||||
|  * @param {object} edges The edges to add to the graph | ||||
|  * @param {object} g The graph object | ||||
|  * @param cy | ||||
|  * @param diagObj | ||||
|  * @param graph | ||||
|  * @param svg | ||||
|  */ | ||||
| export const addEdges = function (edges, diagObj, graph, svg) { | ||||
|   log.info('abc78 edges = ', edges); | ||||
|   const labelsEl = svg.insert('g').attr('class', 'edgeLabels'); | ||||
|   let linkIdCnt = {}; | ||||
|   let dir = diagObj.db.getDirection(); | ||||
|   let defaultStyle; | ||||
|   let defaultLabelStyle; | ||||
|  | ||||
|   if (edges.defaultStyle !== undefined) { | ||||
|     const defaultStyles = getStylesFromArray(edges.defaultStyle); | ||||
|     defaultStyle = defaultStyles.style; | ||||
|     defaultLabelStyle = defaultStyles.labelStyle; | ||||
|   } | ||||
|  | ||||
|   edges.forEach(function (edge) { | ||||
|     // Identify Link | ||||
|     const linkIdBase = 'L-' + edge.start + '-' + edge.end; | ||||
|     // count the links from+to the same node to give unique id | ||||
|     if (linkIdCnt[linkIdBase] === undefined) { | ||||
|       linkIdCnt[linkIdBase] = 0; | ||||
|       log.info('abc78 new entry', linkIdBase, linkIdCnt[linkIdBase]); | ||||
|     } else { | ||||
|       linkIdCnt[linkIdBase]++; | ||||
|       log.info('abc78 new entry', linkIdBase, linkIdCnt[linkIdBase]); | ||||
|     } | ||||
|     let linkId = linkIdBase + '-' + linkIdCnt[linkIdBase]; | ||||
|     log.info('abc78 new link id to be used is', linkIdBase, linkId, linkIdCnt[linkIdBase]); | ||||
|     const linkNameStart = 'LS-' + edge.start; | ||||
|     const linkNameEnd = 'LE-' + edge.end; | ||||
|  | ||||
|     const edgeData = { style: '', labelStyle: '' }; | ||||
|     edgeData.minlen = edge.length || 1; | ||||
|     //edgeData.id = 'id' + cnt; | ||||
|  | ||||
|     // Set link type for rendering | ||||
|     if (edge.type === 'arrow_open') { | ||||
|       edgeData.arrowhead = 'none'; | ||||
|     } else { | ||||
|       edgeData.arrowhead = 'normal'; | ||||
|     } | ||||
|  | ||||
|     // Check of arrow types, placed here in order not to break old rendering | ||||
|     edgeData.arrowTypeStart = 'arrow_open'; | ||||
|     edgeData.arrowTypeEnd = 'arrow_open'; | ||||
|  | ||||
|     /* eslint-disable no-fallthrough */ | ||||
|     switch (edge.type) { | ||||
|       case 'double_arrow_cross': | ||||
|         edgeData.arrowTypeStart = 'arrow_cross'; | ||||
|       case 'arrow_cross': | ||||
|         edgeData.arrowTypeEnd = 'arrow_cross'; | ||||
|         break; | ||||
|       case 'double_arrow_point': | ||||
|         edgeData.arrowTypeStart = 'arrow_point'; | ||||
|       case 'arrow_point': | ||||
|         edgeData.arrowTypeEnd = 'arrow_point'; | ||||
|         break; | ||||
|       case 'double_arrow_circle': | ||||
|         edgeData.arrowTypeStart = 'arrow_circle'; | ||||
|       case 'arrow_circle': | ||||
|         edgeData.arrowTypeEnd = 'arrow_circle'; | ||||
|         break; | ||||
|     } | ||||
|  | ||||
|     let style = ''; | ||||
|     let labelStyle = ''; | ||||
|  | ||||
|     switch (edge.stroke) { | ||||
|       case 'normal': | ||||
|         style = 'fill:none;'; | ||||
|         if (defaultStyle !== undefined) { | ||||
|           style = defaultStyle; | ||||
|         } | ||||
|         if (defaultLabelStyle !== undefined) { | ||||
|           labelStyle = defaultLabelStyle; | ||||
|         } | ||||
|         edgeData.thickness = 'normal'; | ||||
|         edgeData.pattern = 'solid'; | ||||
|         break; | ||||
|       case 'dotted': | ||||
|         edgeData.thickness = 'normal'; | ||||
|         edgeData.pattern = 'dotted'; | ||||
|         edgeData.style = 'fill:none;stroke-width:2px;stroke-dasharray:3;'; | ||||
|         break; | ||||
|       case 'thick': | ||||
|         edgeData.thickness = 'thick'; | ||||
|         edgeData.pattern = 'solid'; | ||||
|         edgeData.style = 'stroke-width: 3.5px;fill:none;'; | ||||
|         break; | ||||
|     } | ||||
|     if (edge.style !== undefined) { | ||||
|       const styles = getStylesFromArray(edge.style); | ||||
|       style = styles.style; | ||||
|       labelStyle = styles.labelStyle; | ||||
|     } | ||||
|  | ||||
|     edgeData.style = edgeData.style += style; | ||||
|     edgeData.labelStyle = edgeData.labelStyle += labelStyle; | ||||
|  | ||||
|     if (edge.interpolate !== undefined) { | ||||
|       edgeData.curve = interpolateToCurve(edge.interpolate, curveLinear); | ||||
|     } else if (edges.defaultInterpolate !== undefined) { | ||||
|       edgeData.curve = interpolateToCurve(edges.defaultInterpolate, curveLinear); | ||||
|     } else { | ||||
|       edgeData.curve = interpolateToCurve(conf.curve, curveLinear); | ||||
|     } | ||||
|  | ||||
|     if (edge.text === undefined) { | ||||
|       if (edge.style !== undefined) { | ||||
|         edgeData.arrowheadStyle = 'fill: #333'; | ||||
|       } | ||||
|     } else { | ||||
|       edgeData.arrowheadStyle = 'fill: #333'; | ||||
|       edgeData.labelpos = 'c'; | ||||
|     } | ||||
|  | ||||
|     edgeData.labelType = edge.labelType; | ||||
|     edgeData.label = edge.text.replace(common.lineBreakRegex, '\n'); | ||||
|  | ||||
|     if (edge.style === undefined) { | ||||
|       edgeData.style = edgeData.style || 'stroke: #333; stroke-width: 1.5px;fill:none;'; | ||||
|     } | ||||
|  | ||||
|     edgeData.labelStyle = edgeData.labelStyle.replace('color:', 'fill:'); | ||||
|  | ||||
|     edgeData.id = linkId; | ||||
|     edgeData.classes = 'flowchart-link ' + linkNameStart + ' ' + linkNameEnd; | ||||
|  | ||||
|     const labelEl = insertEdgeLabel(labelsEl, edgeData); | ||||
|  | ||||
|     // calculate start and end points of the edge, note that the source and target | ||||
|     // can be modified for shapes that have ports | ||||
|     const { source, target, sourceId, targetId } = getEdgeStartEndPoint(edge, dir); | ||||
|     log.debug('abc78 source and target', source, target); | ||||
|     // Add the edge to the graph | ||||
|     graph.edges.push({ | ||||
|       id: 'e' + edge.start + edge.end, | ||||
|       sources: [source], | ||||
|       targets: [target], | ||||
|       sourceId, | ||||
|       targetId, | ||||
|       labelEl: labelEl, | ||||
|       labels: [ | ||||
|         { | ||||
|           width: edgeData.width, | ||||
|           height: edgeData.height, | ||||
|           orgWidth: edgeData.width, | ||||
|           orgHeight: edgeData.height, | ||||
|           text: edgeData.label, | ||||
|           layoutOptions: { | ||||
|             'edgeLabels.inline': 'true', | ||||
|             'edgeLabels.placement': 'CENTER', | ||||
|           }, | ||||
|         }, | ||||
|       ], | ||||
|       edgeData, | ||||
|     }); | ||||
|   }); | ||||
|   return graph; | ||||
| }; | ||||
|  | ||||
| // TODO: break out and share with dagre wrapper. The current code in dagre wrapper also adds | ||||
| // adds the line to the graph, but we don't need that here. This is why we can't use the dagre | ||||
| // wrapper directly for this | ||||
| /** | ||||
|  * Add the markers to the edge depending on the type of arrow is | ||||
|  * @param svgPath | ||||
|  * @param edgeData | ||||
|  * @param diagramType | ||||
|  * @param arrowMarkerAbsolute | ||||
|  * @param id | ||||
|  */ | ||||
| const addMarkersToEdge = function (svgPath, edgeData, diagramType, arrowMarkerAbsolute, id) { | ||||
|   let url = ''; | ||||
|   // Check configuration for absolute path | ||||
|   if (arrowMarkerAbsolute) { | ||||
|     url = | ||||
|       window.location.protocol + | ||||
|       '//' + | ||||
|       window.location.host + | ||||
|       window.location.pathname + | ||||
|       window.location.search; | ||||
|     url = url.replace(/\(/g, '\\('); | ||||
|     url = url.replace(/\)/g, '\\)'); | ||||
|   } | ||||
|  | ||||
|   // look in edge data and decide which marker to use | ||||
|   addEdgeMarkers(svgPath, edgeData, url, id, diagramType); | ||||
| }; | ||||
|  | ||||
| /** | ||||
|  * Returns the all the styles from classDef statements in the graph definition. | ||||
|  * | ||||
|  * @param text | ||||
|  * @param diagObj | ||||
|  * @returns {Map<string, import('../../mermaid/src/diagram-api/types.js').DiagramStyleClassDef>} ClassDef styles | ||||
|  */ | ||||
| export const getClasses = function (text, diagObj) { | ||||
|   log.info('Extracting classes'); | ||||
|   return diagObj.db.getClasses(); | ||||
| }; | ||||
|  | ||||
| const addSubGraphs = function (db) { | ||||
|   const parentLookupDb = { parentById: {}, childrenById: {} }; | ||||
|   const subgraphs = db.getSubGraphs(); | ||||
|   log.info('Subgraphs - ', subgraphs); | ||||
|   subgraphs.forEach(function (subgraph) { | ||||
|     subgraph.nodes.forEach(function (node) { | ||||
|       parentLookupDb.parentById[node] = subgraph.id; | ||||
|       if (parentLookupDb.childrenById[subgraph.id] === undefined) { | ||||
|         parentLookupDb.childrenById[subgraph.id] = []; | ||||
|       } | ||||
|       parentLookupDb.childrenById[subgraph.id].push(node); | ||||
|     }); | ||||
|   }); | ||||
|  | ||||
|   subgraphs.forEach(function (subgraph) { | ||||
|     const data = { id: subgraph.id }; | ||||
|     if (parentLookupDb.parentById[subgraph.id] !== undefined) { | ||||
|       data.parent = parentLookupDb.parentById[subgraph.id]; | ||||
|     } | ||||
|   }); | ||||
|   return parentLookupDb; | ||||
| }; | ||||
|  | ||||
| const calcOffset = function (src, dest, parentLookupDb) { | ||||
|   const ancestor = findCommonAncestor(src, dest, parentLookupDb); | ||||
|   if (ancestor === undefined || ancestor === 'root') { | ||||
|     return { x: 0, y: 0 }; | ||||
|   } | ||||
|  | ||||
|   const ancestorOffset = nodeDb[ancestor].offset; | ||||
|   return { x: ancestorOffset.posX, y: ancestorOffset.posY }; | ||||
| }; | ||||
|  | ||||
| const insertEdge = function (edgesEl, edge, edgeData, diagObj, parentLookupDb, id) { | ||||
|   const offset = calcOffset(edge.sourceId, edge.targetId, parentLookupDb); | ||||
|  | ||||
|   const src = edge.sections[0].startPoint; | ||||
|   const dest = edge.sections[0].endPoint; | ||||
|   const segments = edge.sections[0].bendPoints ? edge.sections[0].bendPoints : []; | ||||
|  | ||||
|   const segPoints = segments.map((segment) => [segment.x + offset.x, segment.y + offset.y]); | ||||
|   const points = [ | ||||
|     [src.x + offset.x, src.y + offset.y], | ||||
|     ...segPoints, | ||||
|     [dest.x + offset.x, dest.y + offset.y], | ||||
|   ]; | ||||
|  | ||||
|   const { x, y } = getLineFunctionsWithOffset(edge.edgeData); | ||||
|   const curve = line().x(x).y(y).curve(curveLinear); | ||||
|   const edgePath = edgesEl | ||||
|     .insert('path') | ||||
|     .attr('d', curve(points)) | ||||
|     .attr('class', 'path ' + edgeData.classes) | ||||
|     .attr('fill', 'none'); | ||||
|   Object.entries(edgeData).forEach(([key, value]) => { | ||||
|     if (key !== 'classes') { | ||||
|       edgePath.attr(key, value); | ||||
|     } | ||||
|   }); | ||||
|   const edgeG = edgesEl.insert('g').attr('class', 'edgeLabel'); | ||||
|   const edgeWithLabel = select(edgeG.node().appendChild(edge.labelEl)); | ||||
|   const box = edgeWithLabel.node().firstChild.getBoundingClientRect(); | ||||
|   edgeWithLabel.attr('width', box.width); | ||||
|   edgeWithLabel.attr('height', box.height); | ||||
|  | ||||
|   edgeG.attr( | ||||
|     'transform', | ||||
|     `translate(${edge.labels[0].x + offset.x}, ${edge.labels[0].y + offset.y})` | ||||
|   ); | ||||
|   addMarkersToEdge(edgePath, edgeData, diagObj.type, diagObj.arrowMarkerAbsolute, id); | ||||
| }; | ||||
|  | ||||
| /** | ||||
|  * Recursive function that iterates over an array of nodes and inserts the children of each node. | ||||
|  * It also recursively populates the inserts the children of the children and so on. | ||||
|  * @param {*} graph | ||||
|  * @param nodeArray | ||||
|  * @param parentLookupDb | ||||
|  */ | ||||
| const insertChildren = (nodeArray, parentLookupDb) => { | ||||
|   nodeArray.forEach((node) => { | ||||
|     // Check if we have reached the end of the tree | ||||
|     if (!node.children) { | ||||
|       node.children = []; | ||||
|     } | ||||
|     // Check if the node has children | ||||
|     const childIds = parentLookupDb.childrenById[node.id]; | ||||
|     // If the node has children, add them to the node | ||||
|     if (childIds) { | ||||
|       childIds.forEach((childId) => { | ||||
|         node.children.push(nodeDb[childId]); | ||||
|       }); | ||||
|     } | ||||
|     // Recursive call | ||||
|     insertChildren(node.children, parentLookupDb); | ||||
|   }); | ||||
| }; | ||||
|  | ||||
| /** | ||||
|  * Draws a flowchart in the tag with id: id based on the graph definition in text. | ||||
|  * | ||||
|  * @param text | ||||
|  * @param id | ||||
|  */ | ||||
|  | ||||
| export const draw = async function (text, id, _version, diagObj) { | ||||
|   const { securityLevel, flowchart: conf } = getConfig(); | ||||
|   nodeDb = {}; | ||||
|   portPos = {}; | ||||
|   const renderEl = select('body').append('div').attr('style', 'height:400px').attr('id', 'cy'); | ||||
|   let graph = { | ||||
|     id: 'root', | ||||
|     layoutOptions: { | ||||
|       'elk.hierarchyHandling': 'INCLUDE_CHILDREN', | ||||
|       'elk.layered.spacing.edgeNodeBetweenLayers': conf?.nodeSpacing ? `${conf.nodeSpacing}` : '30', | ||||
|       // 'elk.layered.mergeEdges': 'true', | ||||
|       'elk.direction': 'DOWN', | ||||
|       // 'elk.ports.sameLayerEdges': true, | ||||
|       // 'nodePlacement.strategy': 'SIMPLE', | ||||
|     }, | ||||
|     children: [], | ||||
|     edges: [], | ||||
|   }; | ||||
|   log.info('Drawing flowchart using v3 renderer', elk); | ||||
|  | ||||
|   // Set the direction, | ||||
|   // Fetch the default direction, use TD if none was found | ||||
|   let dir = diagObj.db.getDirection(); | ||||
|   switch (dir) { | ||||
|     case 'BT': | ||||
|       graph.layoutOptions['elk.direction'] = 'UP'; | ||||
|       break; | ||||
|     case 'TB': | ||||
|       graph.layoutOptions['elk.direction'] = 'DOWN'; | ||||
|       break; | ||||
|     case 'LR': | ||||
|       graph.layoutOptions['elk.direction'] = 'RIGHT'; | ||||
|       break; | ||||
|     case 'RL': | ||||
|       graph.layoutOptions['elk.direction'] = 'LEFT'; | ||||
|       break; | ||||
|   } | ||||
|  | ||||
|   // Find the root dom node to ne used in rendering | ||||
|   // Handle root and document for when rendering in sandbox mode | ||||
|   let sandboxElement; | ||||
|   if (securityLevel === 'sandbox') { | ||||
|     sandboxElement = select('#i' + id); | ||||
|   } | ||||
|   const root = | ||||
|     securityLevel === 'sandbox' | ||||
|       ? select(sandboxElement.nodes()[0].contentDocument.body) | ||||
|       : select('body'); | ||||
|   const doc = securityLevel === 'sandbox' ? sandboxElement.nodes()[0].contentDocument : document; | ||||
|  | ||||
|   const svg = root.select(`[id="${id}"]`); | ||||
|   // Define the supported markers for the diagram | ||||
|   const markers = ['point', 'circle', 'cross']; | ||||
|  | ||||
|   // Add the marker definitions to the svg as marker tags | ||||
|   insertMarkers(svg, markers, diagObj.type, id); | ||||
|  | ||||
|   // Fetch the vertices/nodes and edges/links from the parsed graph definition | ||||
|   const vert = diagObj.db.getVertices(); | ||||
|  | ||||
|   // Setup nodes from the subgraphs with type group, these will be used | ||||
|   // as nodes with children in the subgraph | ||||
|   let subG; | ||||
|   const subGraphs = diagObj.db.getSubGraphs(); | ||||
|   log.info('Subgraphs - ', subGraphs); | ||||
|   for (let i = subGraphs.length - 1; i >= 0; i--) { | ||||
|     subG = subGraphs[i]; | ||||
|     diagObj.db.addVertex( | ||||
|       subG.id, | ||||
|       { text: subG.title, type: subG.labelType }, | ||||
|       'group', | ||||
|       undefined, | ||||
|       subG.classes, | ||||
|       subG.dir | ||||
|     ); | ||||
|   } | ||||
|  | ||||
|   // debugger; | ||||
|   // Add an element in the svg to be used to hold the subgraphs container | ||||
|   // elements | ||||
|   const subGraphsEl = svg.insert('g').attr('class', 'subgraphs'); | ||||
|  | ||||
|   // Create the lookup db for the subgraphs and their children to used when creating | ||||
|   // the tree structured graph | ||||
|   const parentLookupDb = addSubGraphs(diagObj.db); | ||||
|  | ||||
|   // Add the nodes to the graph, this will entail creating the actual nodes | ||||
|   // in order to get the size of the node. You can't get the size of a node | ||||
|   // that is not in the dom so we need to add it to the dom, get the size | ||||
|   // we will position the nodes when we get the layout from elkjs | ||||
|   graph = await addVertices(vert, id, root, doc, diagObj, parentLookupDb, graph); | ||||
|  | ||||
|   // Time for the edges, we start with adding an element in the node to hold the edges | ||||
|   const edgesEl = svg.insert('g').attr('class', 'edges edgePath'); | ||||
|   // Fetch the edges form the parsed graph definition | ||||
|   const edges = diagObj.db.getEdges(); | ||||
|  | ||||
|   // Add the edges to the graph, this will entail creating the actual edges | ||||
|   graph = addEdges(edges, diagObj, graph, svg); | ||||
|  | ||||
|   // Iterate through all nodes and add the top level nodes to the graph | ||||
|   const nodes = Object.keys(nodeDb); | ||||
|   nodes.forEach((nodeId) => { | ||||
|     const node = nodeDb[nodeId]; | ||||
|     if (!node.parent) { | ||||
|       graph.children.push(node); | ||||
|     } | ||||
|     // Subgraph | ||||
|     if (parentLookupDb.childrenById[nodeId] !== undefined) { | ||||
|       node.labels = [ | ||||
|         { | ||||
|           text: node.labelText, | ||||
|           layoutOptions: { | ||||
|             'nodeLabels.placement': '[H_CENTER, V_TOP, INSIDE]', | ||||
|           }, | ||||
|           width: node.labelData.width, | ||||
|           height: node.labelData.height, | ||||
|           // width: 100, | ||||
|           // height: 100, | ||||
|         }, | ||||
|       ]; | ||||
|       delete node.x; | ||||
|       delete node.y; | ||||
|       delete node.width; | ||||
|       delete node.height; | ||||
|     } | ||||
|   }); | ||||
|  | ||||
|   insertChildren(graph.children, parentLookupDb); | ||||
|   log.info('after layout', JSON.stringify(graph, null, 2)); | ||||
|   const g = await elk.layout(graph); | ||||
|   drawNodes(0, 0, g.children, svg, subGraphsEl, diagObj, 0); | ||||
|   utils.insertTitle(svg, 'flowchartTitleText', conf.titleTopMargin, diagObj.db.getDiagramTitle()); | ||||
|   log.info('after layout', g); | ||||
|   g.edges?.map((edge) => { | ||||
|     insertEdge(edgesEl, edge, edge.edgeData, diagObj, parentLookupDb, id); | ||||
|   }); | ||||
|   setupGraphViewbox({}, svg, conf.diagramPadding, conf.useMaxWidth); | ||||
|   // Remove element after layout | ||||
|   renderEl.remove(); | ||||
| }; | ||||
|  | ||||
| const drawNodes = (relX, relY, nodeArray, svg, subgraphsEl, diagObj, depth) => { | ||||
|   nodeArray.forEach(function (node) { | ||||
|     if (node) { | ||||
|       nodeDb[node.id].offset = { | ||||
|         posX: node.x + relX, | ||||
|         posY: node.y + relY, | ||||
|         x: relX, | ||||
|         y: relY, | ||||
|         depth, | ||||
|         width: node.width, | ||||
|         height: node.height, | ||||
|       }; | ||||
|       if (node.type === 'group') { | ||||
|         const subgraphEl = subgraphsEl.insert('g').attr('class', 'subgraph'); | ||||
|         subgraphEl | ||||
|           .insert('rect') | ||||
|           .attr('class', 'subgraph subgraph-lvl-' + (depth % 5) + ' node') | ||||
|           .attr('x', node.x + relX) | ||||
|           .attr('y', node.y + relY) | ||||
|           .attr('width', node.width) | ||||
|           .attr('height', node.height); | ||||
|         const label = subgraphEl.insert('g').attr('class', 'label'); | ||||
|         const labelCentering = getConfig().flowchart.htmlLabels ? node.labelData.width / 2 : 0; | ||||
|         label.attr( | ||||
|           'transform', | ||||
|           `translate(${node.labels[0].x + relX + node.x + labelCentering}, ${ | ||||
|             node.labels[0].y + relY + node.y + 3 | ||||
|           })` | ||||
|         ); | ||||
|         label.node().appendChild(node.labelData.labelNode); | ||||
|  | ||||
|         log.info('Id (UGH)= ', node.type, node.labels); | ||||
|       } else { | ||||
|         log.info('Id (UGH)= ', node.id); | ||||
|         node.el.attr( | ||||
|           'transform', | ||||
|           `translate(${node.x + relX + node.width / 2}, ${node.y + relY + node.height / 2})` | ||||
|         ); | ||||
|       } | ||||
|     } | ||||
|   }); | ||||
|   nodeArray.forEach(function (node) { | ||||
|     if (node && node.type === 'group') { | ||||
|       drawNodes(relX + node.x, relY + node.y, node.children, svg, subgraphsEl, diagObj, depth + 1); | ||||
|     } | ||||
|   }); | ||||
| }; | ||||
|  | ||||
| export default { | ||||
|   getClasses, | ||||
|   draw, | ||||
| }; | ||||
							
								
								
									
										41
									
								
								packages/mermaid-flowchart-elk/src/render-utils.spec.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										41
									
								
								packages/mermaid-flowchart-elk/src/render-utils.spec.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,41 @@ | ||||
| import type { TreeData } from './render-utils.js'; | ||||
| import { findCommonAncestor } from './render-utils.js'; | ||||
| describe('when rendering a flowchart using elk ', () => { | ||||
|   let lookupDb: TreeData; | ||||
|   beforeEach(() => { | ||||
|     lookupDb = { | ||||
|       parentById: { | ||||
|         B4: 'inner', | ||||
|         B5: 'inner', | ||||
|         C4: 'inner2', | ||||
|         C5: 'inner2', | ||||
|         B2: 'Ugge', | ||||
|         B3: 'Ugge', | ||||
|         inner: 'Ugge', | ||||
|         inner2: 'Ugge', | ||||
|         B6: 'outer', | ||||
|       }, | ||||
|       childrenById: { | ||||
|         inner: ['B4', 'B5'], | ||||
|         inner2: ['C4', 'C5'], | ||||
|         Ugge: ['B2', 'B3', 'inner', 'inner2'], | ||||
|         outer: ['B6'], | ||||
|       }, | ||||
|     }; | ||||
|   }); | ||||
|   it('to find parent of siblings in a subgraph', () => { | ||||
|     expect(findCommonAncestor('B4', 'B5', lookupDb)).toBe('inner'); | ||||
|   }); | ||||
|   it('to find an uncle', () => { | ||||
|     expect(findCommonAncestor('B4', 'B2', lookupDb)).toBe('Ugge'); | ||||
|   }); | ||||
|   it('to find a cousin', () => { | ||||
|     expect(findCommonAncestor('B4', 'C4', lookupDb)).toBe('Ugge'); | ||||
|   }); | ||||
|   it('to find a grandparent', () => { | ||||
|     expect(findCommonAncestor('B4', 'B6', lookupDb)).toBe('root'); | ||||
|   }); | ||||
|   it('to find ancestor of siblings in the root', () => { | ||||
|     expect(findCommonAncestor('B1', 'outer', lookupDb)).toBe('root'); | ||||
|   }); | ||||
| }); | ||||
| @@ -3,15 +3,10 @@ export interface TreeData { | ||||
|   childrenById: Record<string, string[]>; | ||||
| } | ||||
| 
 | ||||
| export const findCommonAncestor = (id1: string, id2: string, { parentById }: TreeData) => { | ||||
| export const findCommonAncestor = (id1: string, id2: string, treeData: TreeData) => { | ||||
|   const { parentById } = treeData; | ||||
|   const visited = new Set(); | ||||
|   let currentId = id1; | ||||
| 
 | ||||
|   // Edge case with self edges
 | ||||
|   if (id1 === id2) { | ||||
|     return parentById[id1] || 'root'; | ||||
|   } | ||||
| 
 | ||||
|   while (currentId) { | ||||
|     visited.add(currentId); | ||||
|     if (currentId === id2) { | ||||
| @@ -19,7 +14,6 @@ export const findCommonAncestor = (id1: string, id2: string, { parentById }: Tre | ||||
|     } | ||||
|     currentId = parentById[currentId]; | ||||
|   } | ||||
| 
 | ||||
|   currentId = id2; | ||||
|   while (currentId) { | ||||
|     if (visited.has(currentId)) { | ||||
| @@ -27,6 +21,5 @@ export const findCommonAncestor = (id1: string, id2: string, { parentById }: Tre | ||||
|     } | ||||
|     currentId = parentById[currentId]; | ||||
|   } | ||||
| 
 | ||||
|   return 'root'; | ||||
| }; | ||||
							
								
								
									
										143
									
								
								packages/mermaid-flowchart-elk/src/styles.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										143
									
								
								packages/mermaid-flowchart-elk/src/styles.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,143 @@ | ||||
| /** Returns the styles given options */ | ||||
| export interface FlowChartStyleOptions { | ||||
|   arrowheadColor: string; | ||||
|   border2: string; | ||||
|   clusterBkg: string; | ||||
|   clusterBorder: string; | ||||
|   edgeLabelBackground: string; | ||||
|   fontFamily: string; | ||||
|   lineColor: string; | ||||
|   mainBkg: string; | ||||
|   nodeBorder: string; | ||||
|   nodeTextColor: string; | ||||
|   tertiaryColor: string; | ||||
|   textColor: string; | ||||
|   titleColor: string; | ||||
|   [key: string]: string; | ||||
| } | ||||
|  | ||||
| const genSections = (options: FlowChartStyleOptions) => { | ||||
|   let sections = ''; | ||||
|  | ||||
|   for (let i = 0; i < 5; i++) { | ||||
|     sections += ` | ||||
|       .subgraph-lvl-${i} { | ||||
|         fill: ${options[`surface${i}`]}; | ||||
|         stroke: ${options[`surfacePeer${i}`]}; | ||||
|       } | ||||
|     `; | ||||
|   } | ||||
|   return sections; | ||||
| }; | ||||
|  | ||||
| const getStyles = (options: FlowChartStyleOptions) => | ||||
|   `.label { | ||||
|     font-family: ${options.fontFamily}; | ||||
|     color: ${options.nodeTextColor || options.textColor}; | ||||
|   } | ||||
|   .cluster-label text { | ||||
|     fill: ${options.titleColor}; | ||||
|   } | ||||
|   .cluster-label span { | ||||
|     color: ${options.titleColor}; | ||||
|   } | ||||
|  | ||||
|   .label text,span { | ||||
|     fill: ${options.nodeTextColor || options.textColor}; | ||||
|     color: ${options.nodeTextColor || options.textColor}; | ||||
|   } | ||||
|  | ||||
|   .node rect, | ||||
|   .node circle, | ||||
|   .node ellipse, | ||||
|   .node polygon, | ||||
|   .node path { | ||||
|     fill: ${options.mainBkg}; | ||||
|     stroke: ${options.nodeBorder}; | ||||
|     stroke-width: 1px; | ||||
|   } | ||||
|  | ||||
|   .node .label { | ||||
|     text-align: center; | ||||
|   } | ||||
|   .node.clickable { | ||||
|     cursor: pointer; | ||||
|   } | ||||
|  | ||||
|   .arrowheadPath { | ||||
|     fill: ${options.arrowheadColor}; | ||||
|   } | ||||
|  | ||||
|   .edgePath .path { | ||||
|     stroke: ${options.lineColor}; | ||||
|     stroke-width: 2.0px; | ||||
|   } | ||||
|  | ||||
|   .flowchart-link { | ||||
|     stroke: ${options.lineColor}; | ||||
|     fill: none; | ||||
|   } | ||||
|  | ||||
|   .edgeLabel { | ||||
|     background-color: ${options.edgeLabelBackground}; | ||||
|     rect { | ||||
|       opacity: 0.85; | ||||
|       background-color: ${options.edgeLabelBackground}; | ||||
|       fill: ${options.edgeLabelBackground}; | ||||
|     } | ||||
|     text-align: center; | ||||
|   } | ||||
|  | ||||
|   .cluster rect { | ||||
|     fill: ${options.clusterBkg}; | ||||
|     stroke: ${options.clusterBorder}; | ||||
|     stroke-width: 1px; | ||||
|   } | ||||
|  | ||||
|   .cluster text { | ||||
|     fill: ${options.titleColor}; | ||||
|   } | ||||
|  | ||||
|   .cluster span { | ||||
|     color: ${options.titleColor}; | ||||
|   } | ||||
|   /* .cluster div { | ||||
|     color: ${options.titleColor}; | ||||
|   } */ | ||||
|  | ||||
|   div.mermaidTooltip { | ||||
|     position: absolute; | ||||
|     text-align: center; | ||||
|     max-width: 200px; | ||||
|     padding: 2px; | ||||
|     font-family: ${options.fontFamily}; | ||||
|     font-size: 12px; | ||||
|     background: ${options.tertiaryColor}; | ||||
|     border: 1px solid ${options.border2}; | ||||
|     border-radius: 2px; | ||||
|     pointer-events: none; | ||||
|     z-index: 100; | ||||
|   } | ||||
|  | ||||
|   .flowchartTitleText { | ||||
|     text-anchor: middle; | ||||
|     font-size: 18px; | ||||
|     fill: ${options.textColor}; | ||||
|   } | ||||
|   .subgraph { | ||||
|     stroke-width:2; | ||||
|     rx:3; | ||||
|   } | ||||
|   // .subgraph-lvl-1 { | ||||
|   //   fill:#ccc; | ||||
|   //   // stroke:black; | ||||
|   // } | ||||
|  | ||||
|   .flowchart-label text { | ||||
|     text-anchor: middle; | ||||
|   } | ||||
|  | ||||
|   ${genSections(options)} | ||||
| `; | ||||
|  | ||||
| export default getStyles; | ||||
| @@ -1,7 +1,7 @@ | ||||
| { | ||||
|   "extends": "../../tsconfig.json", | ||||
|   "compilerOptions": { | ||||
|     "rootDir": "./src", | ||||
|     "rootDir": "../..", | ||||
|     "outDir": "./dist", | ||||
|     "types": ["vitest/importMeta", "vitest/globals"] | ||||
|   }, | ||||
| @@ -1,16 +0,0 @@ | ||||
| # @mermaid-js/layout-elk | ||||
|  | ||||
| ## 0.1.2 | ||||
|  | ||||
| ### Patch Changes | ||||
|  | ||||
| - [#5761](https://github.com/mermaid-js/mermaid/pull/5761) [`b34dfe8`](https://github.com/mermaid-js/mermaid/commit/b34dfe8f45eded31da10965ced7ea40fde1ca76c) Thanks [@sidharthv96](https://github.com/sidharthv96)! - Fix type file path | ||||
|  | ||||
| ## 0.1.1 | ||||
|  | ||||
| ### Patch Changes | ||||
|  | ||||
| - [#5758](https://github.com/mermaid-js/mermaid/pull/5758) [`501a55d`](https://github.com/mermaid-js/mermaid/commit/501a55d8f225901ba345c498dec4298490a0196e) Thanks [@sidharthv96](https://github.com/sidharthv96)! - fix: Types path | ||||
|  | ||||
| - Updated dependencies [[`5deaef4`](https://github.com/mermaid-js/mermaid/commit/5deaef456e74d796866431c26f69360e4e74dbff)]: | ||||
|   - mermaid@11.0.2 | ||||
| @@ -1,72 +0,0 @@ | ||||
| # @mermaid-js/layout-elk | ||||
|  | ||||
| This package provides a layout engine for Mermaid based on the [ELK](https://www.eclipse.org/elk/) layout engine. | ||||
|  | ||||
| > [!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. | ||||
|  | ||||
| ## Usage | ||||
|  | ||||
| ``` | ||||
| flowchart-elk TD | ||||
|   A --> B | ||||
|   A --> C | ||||
| ``` | ||||
|  | ||||
| ``` | ||||
| --- | ||||
| config: | ||||
|   layout: elk | ||||
| --- | ||||
|  | ||||
| flowchart TD | ||||
|   A --> B | ||||
|   A --> C | ||||
| ``` | ||||
|  | ||||
| ``` | ||||
| --- | ||||
| config: | ||||
|   layout: elk.stress | ||||
| --- | ||||
|  | ||||
| flowchart TD | ||||
|   A --> B | ||||
|   A --> C | ||||
| ``` | ||||
|  | ||||
| ### With bundlers | ||||
|  | ||||
| ```sh | ||||
| npm install @mermaid-js/layout-elk | ||||
| ``` | ||||
|  | ||||
| ```ts | ||||
| import mermaid from 'mermaid'; | ||||
| import elkLayouts from '@mermaid-js/layout-elk'; | ||||
|  | ||||
| mermaid.registerLayoutLoaders(elkLayouts); | ||||
| ``` | ||||
|  | ||||
| ### With CDN | ||||
|  | ||||
| ```html | ||||
| <script type="module"> | ||||
|   import mermaid from 'https://cdn.jsdelivr.net/npm/mermaid@11/dist/mermaid.esm.min.mjs'; | ||||
|   import elkLayouts from 'https://cdn.jsdelivr.net/npm/@mermaid-js/layout-elk@11/dist/mermaid-layout-elk.esm.min.mjs'; | ||||
|  | ||||
|   mermaid.registerLayoutLoaders(elkLayouts); | ||||
| </script> | ||||
| ``` | ||||
|  | ||||
| ## Supported layouts | ||||
|  | ||||
| - `elk`: The default layout, which is `elk.layered`. | ||||
| - `elk.layered`: Layered layout | ||||
| - `elk.stress`: Stress layout | ||||
| - `elk.force`: Force layout | ||||
| - `elk.mrtree`: Multi-root tree layout | ||||
| - `elk.sporeOverlap`: Spore overlap layout | ||||
|  | ||||
| <!-- TODO: Add images for these layouts, as GitHub doesn't support natively --> | ||||
| @@ -1,45 +0,0 @@ | ||||
| { | ||||
|   "name": "@mermaid-js/layout-elk", | ||||
|   "version": "0.1.2", | ||||
|   "description": "ELK layout engine for mermaid", | ||||
|   "module": "dist/mermaid-layout-elk.core.mjs", | ||||
|   "types": "dist/layouts.d.ts", | ||||
|   "type": "module", | ||||
|   "exports": { | ||||
|     ".": { | ||||
|       "import": "./dist/mermaid-layout-elk.core.mjs", | ||||
|       "types": "./dist/layouts.d.ts" | ||||
|     }, | ||||
|     "./*": "./*" | ||||
|   }, | ||||
|   "keywords": [ | ||||
|     "diagram", | ||||
|     "markdown", | ||||
|     "elk", | ||||
|     "mermaid" | ||||
|   ], | ||||
|   "scripts": {}, | ||||
|   "repository": { | ||||
|     "type": "git", | ||||
|     "url": "https://github.com/mermaid-js/mermaid" | ||||
|   }, | ||||
|   "contributors": [ | ||||
|     "Knut Sveidqvist", | ||||
|     "Sidharth Vinod" | ||||
|   ], | ||||
|   "license": "MIT", | ||||
|   "dependencies": { | ||||
|     "d3": "^7.9.0", | ||||
|     "elkjs": "^0.9.3" | ||||
|   }, | ||||
|   "devDependencies": { | ||||
|     "@types/d3": "^7.4.3", | ||||
|     "mermaid": "workspace:^" | ||||
|   }, | ||||
|   "peerDependencies": { | ||||
|     "mermaid": "^11.0.0" | ||||
|   }, | ||||
|   "files": [ | ||||
|     "dist" | ||||
|   ] | ||||
| } | ||||
| @@ -1,19 +0,0 @@ | ||||
| import type { LayoutLoaderDefinition } from 'mermaid'; | ||||
|  | ||||
| const loader = async () => await import(`./render.js`); | ||||
| const algos = ['elk.stress', 'elk.force', 'elk.mrtree', 'elk.sporeOverlap']; | ||||
|  | ||||
| const layouts: LayoutLoaderDefinition[] = [ | ||||
|   { | ||||
|     name: 'elk', | ||||
|     loader, | ||||
|     algorithm: 'elk.layered', | ||||
|   }, | ||||
|   ...algos.map((algo) => ({ | ||||
|     name: algo, | ||||
|     loader, | ||||
|     algorithm: algo, | ||||
|   })), | ||||
| ]; | ||||
|  | ||||
| export default layouts; | ||||
| @@ -1,992 +0,0 @@ | ||||
| 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'; | ||||
|  | ||||
| export const render = async ( | ||||
|   data4Layout: LayoutData, | ||||
|   svg: SVG, | ||||
|   { | ||||
|     common, | ||||
|     getConfig, | ||||
|     insertCluster, | ||||
|     insertEdge, | ||||
|     insertEdgeLabel, | ||||
|     insertMarkers, | ||||
|     insertNode, | ||||
|     interpolateToCurve, | ||||
|     labelHelper, | ||||
|     log, | ||||
|     positionEdgeLabel, | ||||
|   }: InternalHelpers, | ||||
|   { algorithm }: RenderOptions | ||||
| ) => { | ||||
|   const nodeDb: Record<string, any> = {}; | ||||
|   const clusterDb: Record<string, any> = {}; | ||||
|  | ||||
|   const addVertex = async (nodeEl: any, graph: { children: any[] }, nodeArr: any, node: any) => { | ||||
|     const labelData: any = { width: 0, height: 0 }; | ||||
|  | ||||
|     let boundingBox; | ||||
|     const child = { | ||||
|       ...node, | ||||
|     }; | ||||
|     graph.children.push(child); | ||||
|     nodeDb[node.id] = child; | ||||
|  | ||||
|     // Add the element to the DOM | ||||
|     if (!node.isGroup) { | ||||
|       const childNodeEl = await insertNode(nodeEl, node, node.dir); | ||||
|       boundingBox = childNodeEl.node().getBBox(); | ||||
|       child.domId = childNodeEl; | ||||
|       child.width = boundingBox.width; | ||||
|       child.height = boundingBox.height; | ||||
|     } else { | ||||
|       // A subgraph | ||||
|       child.children = []; | ||||
|       await addVertices(nodeEl, nodeArr, child, node.id); | ||||
|  | ||||
|       if (node.label) { | ||||
|         // @ts-ignore TODO: fix this | ||||
|         const { shapeSvg, bbox } = await labelHelper(nodeEl, node, undefined, true); | ||||
|         labelData.width = bbox.width; | ||||
|         labelData.wrappingWidth = getConfig().flowchart!.wrappingWidth; | ||||
|         // Give some padding for elk | ||||
|         labelData.height = bbox.height - 2; | ||||
|         labelData.labelNode = shapeSvg.node(); | ||||
|         // We need the label hight to be able to size the subgraph; | ||||
|         shapeSvg.remove(); | ||||
|       } else { | ||||
|         // Subgraph without label | ||||
|         labelData.width = 0; | ||||
|         labelData.height = 0; | ||||
|       } | ||||
|       child.labelData = labelData; | ||||
|       child.domId = nodeEl; | ||||
|     } | ||||
|   }; | ||||
|  | ||||
|   const addVertices = async function ( | ||||
|     nodeEl: any, | ||||
|     nodeArr: any[], | ||||
|     graph: { | ||||
|       id: string; | ||||
|       layoutOptions: { | ||||
|         'elk.hierarchyHandling': string; | ||||
|         'elk.algorithm': any; | ||||
|         'nodePlacement.strategy': any; | ||||
|         'elk.layered.mergeEdges': any; | ||||
|         'elk.direction': string; | ||||
|         'spacing.baseValue': number; | ||||
|       }; | ||||
|       children: never[]; | ||||
|       edges: never[]; | ||||
|     }, | ||||
|     parentId?: undefined | ||||
|   ) { | ||||
|     const siblings = nodeArr.filter((node: { parentId: any }) => node.parentId === parentId); | ||||
|     log.info('addVertices APA12', siblings, parentId); | ||||
|     // Iterate through each item in the vertex object (containing all the vertices found) in the graph definition | ||||
|     await Promise.all( | ||||
|       siblings.map(async (node: any) => { | ||||
|         await addVertex(nodeEl, graph, nodeArr, node); | ||||
|       }) | ||||
|     ); | ||||
|     return graph; | ||||
|   }; | ||||
|  | ||||
|   const drawNodes = async ( | ||||
|     relX: number, | ||||
|     relY: number, | ||||
|     nodeArray: any[], | ||||
|     svg: any, | ||||
|     subgraphsEl: SVGGroup, | ||||
|     depth: number | ||||
|   ) => { | ||||
|     await Promise.all( | ||||
|       nodeArray.map(async function (node: { | ||||
|         id: string | number; | ||||
|         x: any; | ||||
|         y: any; | ||||
|         width: number; | ||||
|         labels: { width: any }[]; | ||||
|         height: number; | ||||
|         isGroup: any; | ||||
|         labelData: any; | ||||
|         offset: { posX: number; posY: number }; | ||||
|         shape: any; | ||||
|         domId: { node: () => any; attr: (arg0: string, arg1: string) => void }; | ||||
|       }) { | ||||
|         if (node) { | ||||
|           nodeDb[node.id] = node; | ||||
|           nodeDb[node.id].offset = { | ||||
|             posX: node.x + relX, | ||||
|             posY: node.y + relY, | ||||
|             x: relX, | ||||
|             y: relY, | ||||
|             depth, | ||||
|             width: Math.max(node.width, node.labels ? node.labels[0]?.width || 0 : 0), | ||||
|             height: node.height, | ||||
|           }; | ||||
|           if (node.isGroup) { | ||||
|             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)); | ||||
|             clusterNode.x = node.offset.posX + node.width / 2; | ||||
|             clusterNode.y = node.offset.posY + node.height / 2; | ||||
|             await insertCluster(subgraphEl, clusterNode); | ||||
|  | ||||
|             log.debug('Id (UIO)= ', node.id, node.width, node.shape, node.labels); | ||||
|           } else { | ||||
|             log.info( | ||||
|               'Id NODE = ', | ||||
|               node.id, | ||||
|               node.x, | ||||
|               node.y, | ||||
|               relX, | ||||
|               relY, | ||||
|               node.domId.node(), | ||||
|               `translate(${node.x + relX + node.width / 2}, ${node.y + relY + node.height / 2})` | ||||
|             ); | ||||
|             node.domId.attr( | ||||
|               'transform', | ||||
|               `translate(${node.x + relX + node.width / 2}, ${node.y + relY + node.height / 2})` | ||||
|             ); | ||||
|           } | ||||
|         } | ||||
|       }) | ||||
|     ); | ||||
|  | ||||
|     await Promise.all( | ||||
|       nodeArray.map(async function (node: { isGroup: any; x: any; y: any; children: any }) { | ||||
|         if (node?.isGroup) { | ||||
|           await drawNodes(relX + node.x, relY + node.y, node.children, svg, subgraphsEl, depth + 1); | ||||
|         } | ||||
|       }) | ||||
|     ); | ||||
|   }; | ||||
|  | ||||
|   const addSubGraphs = (nodeArr: any[]): TreeData => { | ||||
|     const parentLookupDb: TreeData = { parentById: {}, childrenById: {} }; | ||||
|     const subgraphs = nodeArr.filter((node: { isGroup: any }) => node.isGroup); | ||||
|     log.info('Subgraphs - ', subgraphs); | ||||
|     subgraphs.forEach((subgraph: { id: string }) => { | ||||
|       const children = nodeArr.filter((node: { parentId: any }) => node.parentId === subgraph.id); | ||||
|       children.forEach((node: any) => { | ||||
|         parentLookupDb.parentById[node.id] = subgraph.id; | ||||
|         if (parentLookupDb.childrenById[subgraph.id] === undefined) { | ||||
|           parentLookupDb.childrenById[subgraph.id] = []; | ||||
|         } | ||||
|         parentLookupDb.childrenById[subgraph.id].push(node); | ||||
|       }); | ||||
|     }); | ||||
|  | ||||
|     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; | ||||
|  | ||||
|     // Save the original source and target | ||||
|     const sourceId = source; | ||||
|     const targetId = target; | ||||
|  | ||||
|     const startNode = nodeDb[edge.start.id]; | ||||
|     const endNode = nodeDb[edge.end.id]; | ||||
|  | ||||
|     if (!startNode || !endNode) { | ||||
|       return { source, target }; | ||||
|     } | ||||
|  | ||||
|     // Add the edge to the graph | ||||
|     return { source, target, sourceId, targetId }; | ||||
|   }; | ||||
|  | ||||
|   const calcOffset = function (src: string, dest: string, parentLookupDb: TreeData) { | ||||
|     const ancestor = findCommonAncestor(src, dest, parentLookupDb); | ||||
|     if (ancestor === undefined || ancestor === 'root') { | ||||
|       return { x: 0, y: 0 }; | ||||
|     } | ||||
|  | ||||
|     const ancestorOffset = nodeDb[ancestor].offset; | ||||
|     return { x: ancestorOffset.posX, y: ancestorOffset.posY }; | ||||
|   }; | ||||
|  | ||||
|   /** | ||||
|    * Add edges to graph based on parsed graph definition | ||||
|    */ | ||||
|   const addEdges = async function ( | ||||
|     dataForLayout: { edges: any; direction: string }, | ||||
|     graph: { | ||||
|       id?: string; | ||||
|       layoutOptions?: { | ||||
|         'elk.hierarchyHandling': string; | ||||
|         'elk.algorithm': any; | ||||
|         'nodePlacement.strategy': any; | ||||
|         'elk.layered.mergeEdges': any; | ||||
|         'elk.direction': string; | ||||
|         'spacing.baseValue': number; | ||||
|       }; | ||||
|       children?: never[]; | ||||
|       edges: any; | ||||
|     }, | ||||
|     svg: SVG | ||||
|   ) { | ||||
|     log.info('abc78 DAGA edges = ', dataForLayout); | ||||
|     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; | ||||
|  | ||||
|     await Promise.all( | ||||
|       edges.map(async function (edge: { | ||||
|         id: string; | ||||
|         start: string; | ||||
|         end: string; | ||||
|         length: number; | ||||
|         text: undefined; | ||||
|         label: any; | ||||
|         type: string; | ||||
|         stroke: any; | ||||
|         interpolate: undefined; | ||||
|         style: undefined; | ||||
|         labelType: any; | ||||
|       }) { | ||||
|         // Identify Link | ||||
|         const linkIdBase = edge.id; // 'L-' + edge.start + '-' + edge.end; | ||||
|         // count the links from+to the same node to give unique id | ||||
|         if (linkIdCnt[linkIdBase] === undefined) { | ||||
|           linkIdCnt[linkIdBase] = 0; | ||||
|           log.info('abc78 new entry', linkIdBase, linkIdCnt[linkIdBase]); | ||||
|         } else { | ||||
|           linkIdCnt[linkIdBase]++; | ||||
|           log.info('abc78 new entry', 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; | ||||
|         const linkNameEnd = 'LE_' + edge.end; | ||||
|  | ||||
|         const edgeData: any = { style: '', labelStyle: '' }; | ||||
|         edgeData.minlen = edge.length || 1; | ||||
|         edge.text = edge.label; | ||||
|         // Set link type for rendering | ||||
|         if (edge.type === 'arrow_open') { | ||||
|           edgeData.arrowhead = 'none'; | ||||
|         } else { | ||||
|           edgeData.arrowhead = 'normal'; | ||||
|         } | ||||
|  | ||||
|         // Check of arrow types, placed here in order not to break old rendering | ||||
|         edgeData.arrowTypeStart = 'arrow_open'; | ||||
|         edgeData.arrowTypeEnd = 'arrow_open'; | ||||
|  | ||||
|         /* eslint-disable no-fallthrough */ | ||||
|         switch (edge.type) { | ||||
|           case 'double_arrow_cross': | ||||
|             edgeData.arrowTypeStart = 'arrow_cross'; | ||||
|           case 'arrow_cross': | ||||
|             edgeData.arrowTypeEnd = 'arrow_cross'; | ||||
|             break; | ||||
|           case 'double_arrow_point': | ||||
|             edgeData.arrowTypeStart = 'arrow_point'; | ||||
|           case 'arrow_point': | ||||
|             edgeData.arrowTypeEnd = 'arrow_point'; | ||||
|             break; | ||||
|           case 'double_arrow_circle': | ||||
|             edgeData.arrowTypeStart = 'arrow_circle'; | ||||
|           case 'arrow_circle': | ||||
|             edgeData.arrowTypeEnd = 'arrow_circle'; | ||||
|             break; | ||||
|         } | ||||
|  | ||||
|         let style = ''; | ||||
|         let labelStyle = ''; | ||||
|  | ||||
|         switch (edge.stroke) { | ||||
|           case 'normal': | ||||
|             style = 'fill:none;'; | ||||
|             if (defaultStyle !== undefined) { | ||||
|               style = defaultStyle; | ||||
|             } | ||||
|             if (defaultLabelStyle !== undefined) { | ||||
|               labelStyle = defaultLabelStyle; | ||||
|             } | ||||
|             edgeData.thickness = 'normal'; | ||||
|             edgeData.pattern = 'solid'; | ||||
|             break; | ||||
|           case 'dotted': | ||||
|             edgeData.thickness = 'normal'; | ||||
|             edgeData.pattern = 'dotted'; | ||||
|             edgeData.style = 'fill:none;stroke-width:2px;stroke-dasharray:3;'; | ||||
|             break; | ||||
|           case 'thick': | ||||
|             edgeData.thickness = 'thick'; | ||||
|             edgeData.pattern = 'solid'; | ||||
|             edgeData.style = 'stroke-width: 3.5px;fill:none;'; | ||||
|             break; | ||||
|         } | ||||
|  | ||||
|         edgeData.style = edgeData.style += style; | ||||
|         edgeData.labelStyle = edgeData.labelStyle += labelStyle; | ||||
|  | ||||
|         const conf = getConfig(); | ||||
|         if (edge.interpolate !== undefined) { | ||||
|           edgeData.curve = interpolateToCurve(edge.interpolate, curveLinear); | ||||
|         } else if (edges.defaultInterpolate !== undefined) { | ||||
|           edgeData.curve = interpolateToCurve(edges.defaultInterpolate, curveLinear); | ||||
|         } else { | ||||
|           // @ts-ignore TODO: fix this | ||||
|           edgeData.curve = interpolateToCurve(conf.curve, curveLinear); | ||||
|         } | ||||
|  | ||||
|         if (edge.text === undefined) { | ||||
|           if (edge.style !== undefined) { | ||||
|             edgeData.arrowheadStyle = 'fill: #333'; | ||||
|           } | ||||
|         } else { | ||||
|           edgeData.arrowheadStyle = 'fill: #333'; | ||||
|           edgeData.labelpos = 'c'; | ||||
|         } | ||||
|  | ||||
|         edgeData.labelType = edge.labelType; | ||||
|         edgeData.label = (edge?.text || '').replace(common.lineBreakRegex, '\n'); | ||||
|  | ||||
|         if (edge.style === undefined) { | ||||
|           edgeData.style = edgeData.style || 'stroke: #333; stroke-width: 1.5px;fill:none;'; | ||||
|         } | ||||
|  | ||||
|         edgeData.labelStyle = edgeData.labelStyle.replace('color:', 'fill:'); | ||||
|  | ||||
|         edgeData.id = linkId; | ||||
|         edgeData.classes = 'flowchart-link ' + linkNameStart + ' ' + linkNameEnd; | ||||
|  | ||||
|         const labelEl = await insertEdgeLabel(labelsEl, edgeData); | ||||
|  | ||||
|         // 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); | ||||
|         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], | ||||
|           sourceId, | ||||
|           targetId, | ||||
|           labelEl: labelEl, | ||||
|           labels: [ | ||||
|             { | ||||
|               width: edgeData.width, | ||||
|               height: edgeData.height, | ||||
|               orgWidth: edgeData.width, | ||||
|               orgHeight: edgeData.height, | ||||
|               text: edgeData.label, | ||||
|               layoutOptions: { | ||||
|                 'edgeLabels.inline': 'true', | ||||
|                 'edgeLabels.placement': 'CENTER', | ||||
|               }, | ||||
|             }, | ||||
|           ], | ||||
|           edgeData, | ||||
|         }); | ||||
|       }) | ||||
|     ); | ||||
|     return graph; | ||||
|   }; | ||||
|  | ||||
|   function dir2ElkDirection(dir: any) { | ||||
|     switch (dir) { | ||||
|       case 'LR': | ||||
|         return 'RIGHT'; | ||||
|       case 'RL': | ||||
|         return 'LEFT'; | ||||
|       case 'TB': | ||||
|         return 'DOWN'; | ||||
|       case 'BT': | ||||
|         return 'UP'; | ||||
|       default: | ||||
|         return 'DOWN'; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   function setIncludeChildrenPolicy(nodeId: string, ancestorId: string) { | ||||
|     const node = nodeDb[nodeId]; | ||||
|  | ||||
|     if (!node) { | ||||
|       return; | ||||
|     } | ||||
|     if (node?.layoutOptions === undefined) { | ||||
|       node.layoutOptions = {}; | ||||
|     } | ||||
|     node.layoutOptions['elk.hierarchyHandling'] = 'INCLUDE_CHILDREN'; | ||||
|     if (node.id !== ancestorId) { | ||||
|       setIncludeChildrenPolicy(node.parentId, ancestorId); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   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; | ||||
|  | ||||
|     // 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 (r1 !== 0 && r2 !== 0 && 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( | ||||
|       `UIO diamondIntersection calc abc89: | ||||
|   outsidePoint: ${JSON.stringify(outsidePoint)} | ||||
|   insidePoint : ${JSON.stringify(insidePoint)} | ||||
|   node        : x:${bounds.x} y:${bounds.y} w:${bounds.width} h:${bounds.height}`, | ||||
|       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; | ||||
|     // const top = y1 + h / 2; | ||||
|  | ||||
|     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: p1.x, y: p1.y }, | ||||
|         { x: p2.x, y: 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 }, | ||||
|     insidePoint: { x: number; y: number } | ||||
|   ) => { | ||||
|     log.debug(`intersection calc abc89: | ||||
|   outsidePoint: ${JSON.stringify(outsidePoint)} | ||||
|   insidePoint : ${JSON.stringify(insidePoint)} | ||||
|   node        : x:${node.x} y:${node.y} w:${node.width} h:${node.height}`); | ||||
|     const x = node.x; | ||||
|     const y = node.y; | ||||
|  | ||||
|     const dx = Math.abs(x - insidePoint.x); | ||||
|     // const dy = Math.abs(y - insidePoint.y); | ||||
|     const w = node.width / 2; | ||||
|     let r = insidePoint.x < outsidePoint.x ? w - dx : w + dx; | ||||
|     const h = node.height / 2; | ||||
|  | ||||
|     const Q = Math.abs(outsidePoint.y - insidePoint.y); | ||||
|     const R = Math.abs(outsidePoint.x - insidePoint.x); | ||||
|  | ||||
|     if (Math.abs(y - outsidePoint.y) * w > Math.abs(x - outsidePoint.x) * h) { | ||||
|       // Intersection is top or bottom of rect. | ||||
|       const q = insidePoint.y < outsidePoint.y ? outsidePoint.y - h - y : y - h - outsidePoint.y; | ||||
|       r = (R * q) / Q; | ||||
|       const res = { | ||||
|         x: insidePoint.x < outsidePoint.x ? insidePoint.x + r : insidePoint.x - R + r, | ||||
|         y: insidePoint.y < outsidePoint.y ? insidePoint.y + Q - q : insidePoint.y - Q + q, | ||||
|       }; | ||||
|  | ||||
|       if (r === 0) { | ||||
|         res.x = outsidePoint.x; | ||||
|         res.y = outsidePoint.y; | ||||
|       } | ||||
|       if (R === 0) { | ||||
|         res.x = outsidePoint.x; | ||||
|       } | ||||
|       if (Q === 0) { | ||||
|         res.y = outsidePoint.y; | ||||
|       } | ||||
|  | ||||
|       log.debug(`abc89 topp/bott calc, Q ${Q}, q ${q}, R ${R}, r ${r}`, res); // cspell: disable-line | ||||
|  | ||||
|       return res; | ||||
|     } else { | ||||
|       // Intersection onn sides of rect | ||||
|       if (insidePoint.x < outsidePoint.x) { | ||||
|         r = outsidePoint.x - w - x; | ||||
|       } else { | ||||
|         // r = outsidePoint.x - w - x; | ||||
|         r = x - w - outsidePoint.x; | ||||
|       } | ||||
|       const q = (Q * r) / R; | ||||
|       //  OK let _x = insidePoint.x < outsidePoint.x ? insidePoint.x + R - r : insidePoint.x + dx - w; | ||||
|       // OK let _x = insidePoint.x < outsidePoint.x ? insidePoint.x + R - r : outsidePoint.x + r; | ||||
|       let _x = insidePoint.x < outsidePoint.x ? insidePoint.x + R - r : insidePoint.x - R + r; | ||||
|       // let _x = insidePoint.x < outsidePoint.x ? insidePoint.x + R - r : outsidePoint.x + r; | ||||
|       let _y = insidePoint.y < outsidePoint.y ? insidePoint.y + q : insidePoint.y - q; | ||||
|       log.debug(`sides calc abc89, Q ${Q}, q ${q}, R ${R}, r ${r}`, { _x, _y }); | ||||
|       if (r === 0) { | ||||
|         _x = outsidePoint.x; | ||||
|         _y = outsidePoint.y; | ||||
|       } | ||||
|       if (R === 0) { | ||||
|         _x = outsidePoint.x; | ||||
|       } | ||||
|       if (Q === 0) { | ||||
|         _y = outsidePoint.y; | ||||
|       } | ||||
|  | ||||
|       return { x: _x, y: _y }; | ||||
|     } | ||||
|   }; | ||||
|   const outsideNode = ( | ||||
|     node: { x: any; y: any; width: number; height: number }, | ||||
|     point: { x: number; y: number } | ||||
|   ) => { | ||||
|     const x = node.x; | ||||
|     const y = node.y; | ||||
|     const dx = Math.abs(point.x - x); | ||||
|     const dy = Math.abs(point.y - y); | ||||
|     const w = node.width / 2; | ||||
|     const h = node.height / 2; | ||||
|     if (dx >= w || dy >= h) { | ||||
|       return true; | ||||
|     } | ||||
|     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('UIO cutPathAtIntersect Points:', _points, 'node:', bounds, 'isDiamond', isDiamond); | ||||
|     const points: any[] = []; | ||||
|     let lastPointOutside = _points[0]; | ||||
|     let isInside = false; | ||||
|     _points.forEach((point: any) => { | ||||
|       // const node = clusterDb[edge.toCluster].node; | ||||
|       log.debug(' checking point', point, bounds); | ||||
|  | ||||
|       // 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 anf 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); | ||||
|         } | ||||
|  | ||||
|         // 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); | ||||
|         } | ||||
|       } | ||||
|     }); | ||||
|     log.debug('returning points', points); | ||||
|     return points; | ||||
|   }; | ||||
|  | ||||
|   // @ts-ignore - ELK is not typed | ||||
|   const elk = new ELK(); | ||||
|   const element = svg.select('g'); | ||||
|   // Add the arrowheads to the svg | ||||
|   insertMarkers(element, data4Layout.markers, data4Layout.type, data4Layout.diagramId); | ||||
|  | ||||
|   // Setup the graph with the layout options and the data for the layout | ||||
|   let elkGraph: any = { | ||||
|     id: 'root', | ||||
|     layoutOptions: { | ||||
|       'elk.hierarchyHandling': 'INCLUDE_CHILDREN', | ||||
|       'elk.algorithm': algorithm, | ||||
|       'nodePlacement.strategy': data4Layout.config.elk.nodePlacementStrategy, | ||||
|       'elk.layered.mergeEdges': data4Layout.config.elk.mergeEdges, | ||||
|       'elk.direction': 'DOWN', | ||||
|       'spacing.baseValue': 30, | ||||
|       // 'spacing.nodeNode': 40, | ||||
|       // 'spacing.nodeNodeBetweenLayers': 45, | ||||
|       // 'spacing.edgeNode': 40, | ||||
|       // 'spacing.edgeNodeBetweenLayers': 30, | ||||
|       // 'spacing.edgeEdge': 30, | ||||
|       // 'spacing.edgeEdgeBetweenLayers': 40, | ||||
|       // 'spacing.nodeSelfLoop': 50, | ||||
|     }, | ||||
|     children: [], | ||||
|     edges: [], | ||||
|   }; | ||||
|  | ||||
|   log.info('Drawing flowchart using v4 renderer', elk); | ||||
|  | ||||
|   // Set the direction of the graph based on the parsed information | ||||
|   const dir = data4Layout.direction || 'DOWN'; | ||||
|   elkGraph.layoutOptions['elk.direction'] = dir2ElkDirection(dir); | ||||
|  | ||||
|   // Create the lookup db for the subgraphs and their children to used when creating | ||||
|   // the tree structured graph | ||||
|   const parentLookupDb: any = addSubGraphs(data4Layout.nodes); | ||||
|  | ||||
|   // Add elements in the svg to be used to hold the subgraphs container | ||||
|   // elements and the nodes | ||||
|   const subGraphsEl = svg.insert('g').attr('class', 'subgraphs'); | ||||
|  | ||||
|   const nodeEl = svg.insert('g').attr('class', 'nodes'); | ||||
|  | ||||
|   // Add the nodes to the graph, this will entail creating the actual nodes | ||||
|   // in order to get the size of the node. You can't get the size of a node | ||||
|   // that is not in the dom so we need to add it to the dom, get the size | ||||
|   // we will position the nodes when we get the layout from elkjs | ||||
|   elkGraph = await addVertices(nodeEl, data4Layout.nodes, elkGraph); | ||||
|   // Time for the edges, we start with adding an element in the node to hold the edges | ||||
|   const edgesEl = svg.insert('g').attr('class', 'edges edgePaths'); | ||||
|  | ||||
|   // Add the edges to the elk graph, this will entail creating the actual edges | ||||
|   elkGraph = await addEdges(data4Layout, elkGraph, svg); | ||||
|  | ||||
|   // Iterate through all nodes and add the top level nodes to the graph | ||||
|   const nodes = data4Layout.nodes; | ||||
|   nodes.forEach((n: { id: string | number }) => { | ||||
|     const node = nodeDb[n.id]; | ||||
|  | ||||
|     // Subgraph | ||||
|     if (parentLookupDb.childrenById[node.id] !== undefined) { | ||||
|       node.labels = [ | ||||
|         { | ||||
|           text: node.label, | ||||
|           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.layoutOptions = { | ||||
|         'spacing.baseValue': 30, | ||||
|         'nodeLabels.placement': '[H_CENTER V_TOP, INSIDE]', | ||||
|       }; | ||||
|       if (node.dir) { | ||||
|         node.layoutOptions = { | ||||
|           ...node.layoutOptions, | ||||
|           'elk.algorithm': algorithm, | ||||
|           'elk.direction': dir2ElkDirection(node.dir), | ||||
|           'nodePlacement.strategy': data4Layout.config['elk.nodePlacement.strategy'], | ||||
|           'elk.layered.mergeEdges': data4Layout.config['elk.mergeEdges'], | ||||
|           'elk.hierarchyHandling': 'SEPARATE_CHILDREN', | ||||
|         }; | ||||
|       } | ||||
|       delete node.x; | ||||
|       delete node.y; | ||||
|       delete node.width; | ||||
|       delete node.height; | ||||
|     } | ||||
|   }); | ||||
|   elkGraph.edges.forEach((edge: any) => { | ||||
|     const source = edge.sources[0]; | ||||
|     const target = edge.targets[0]; | ||||
|  | ||||
|     if (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); | ||||
|       setIncludeChildrenPolicy(target, ancestorId); | ||||
|     } | ||||
|   }); | ||||
|  | ||||
|   const g = await elk.layout(elkGraph); | ||||
|  | ||||
|   // debugger; | ||||
|   await drawNodes(0, 0, g.children, svg, subGraphsEl, 0); | ||||
|   g.edges?.map( | ||||
|     (edge: { | ||||
|       sources: (string | number)[]; | ||||
|       targets: (string | number)[]; | ||||
|       start: any; | ||||
|       end: any; | ||||
|       sections: { startPoint: any; endPoint: any; bendPoints: any }[]; | ||||
|       points: any[]; | ||||
|       x: any; | ||||
|       labels: { height: number; width: number; x: number; y: number }[]; | ||||
|       y: any; | ||||
|     }) => { | ||||
|       // (elem, edge, clusterDb, diagramType, graph, id) | ||||
|       const startNode = nodeDb[edge.sources[0]]; | ||||
|       const startCluster = parentLookupDb[edge.sources[0]]; | ||||
|       const endNode = nodeDb[edge.targets[0]]; | ||||
|       const sourceId = edge.start; | ||||
|       const targetId = edge.end; | ||||
|  | ||||
|       const offset = calcOffset(sourceId, targetId, parentLookupDb); | ||||
|       log.debug( | ||||
|         'offset', | ||||
|         offset, | ||||
|         sourceId, | ||||
|         ' ==> ', | ||||
|         targetId, | ||||
|         'edge:', | ||||
|         edge, | ||||
|         'cluster:', | ||||
|         startCluster, | ||||
|         startNode | ||||
|       ); | ||||
|       if (edge.sections) { | ||||
|         const src = edge.sections[0].startPoint; | ||||
|         const dest = edge.sections[0].endPoint; | ||||
|         const segments = edge.sections[0].bendPoints ? edge.sections[0].bendPoints : []; | ||||
|  | ||||
|         const segPoints = segments.map((segment: { x: any; y: any }) => { | ||||
|           return { x: segment.x + offset.x, y: segment.y + offset.y }; | ||||
|         }); | ||||
|         edge.points = [ | ||||
|           { x: src.x + offset.x, y: src.y + offset.y }, | ||||
|           ...segPoints, | ||||
|           { x: dest.x + offset.x, y: dest.y + offset.y }, | ||||
|         ]; | ||||
|  | ||||
|         let sw = startNode.width; | ||||
|         let ew = endNode.width; | ||||
|         if (startNode.isGroup) { | ||||
|           const bbox = startNode.domId.node().getBBox(); | ||||
|           // sw = Math.max(bbox.width, startNode.width, startNode.labels[0].width); | ||||
|           sw = Math.max(startNode.width, startNode.labels[0].width + startNode.padding); | ||||
|           // sw = startNode.width; | ||||
|           log.debug( | ||||
|             'UIO width', | ||||
|             startNode.id, | ||||
|             startNode.with, | ||||
|             'bbox.width=', | ||||
|             bbox.width, | ||||
|             'lw=', | ||||
|             startNode.labels[0].width, | ||||
|             'node:', | ||||
|             startNode.width, | ||||
|             'SW = ', | ||||
|             sw | ||||
|             // 'HTML:', | ||||
|             // startNode.domId.node().innerHTML | ||||
|           ); | ||||
|         } | ||||
|         if (endNode.isGroup) { | ||||
|           const bbox = endNode.domId.node().getBBox(); | ||||
|           ew = Math.max(endNode.width, endNode.labels[0].width + endNode.padding); | ||||
|  | ||||
|           log.debug( | ||||
|             'UIO width', | ||||
|             startNode.id, | ||||
|             startNode.with, | ||||
|             bbox.width, | ||||
|             'EW = ', | ||||
|             ew, | ||||
|             'HTML:', | ||||
|             startNode.innerHTML | ||||
|           ); | ||||
|         } | ||||
|         if (startNode.shape === 'diamond') { | ||||
|           edge.points.unshift({ | ||||
|             x: startNode.x + startNode.width / 2 + offset.x, | ||||
|             y: startNode.y + startNode.height / 2 + offset.y, | ||||
|           }); | ||||
|         } | ||||
|         if (endNode.shape === 'diamond') { | ||||
|           const x = endNode.x + endNode.width / 2 + offset.x; | ||||
|           // Add a point at the center of the diamond | ||||
|           if ( | ||||
|             Math.abs(edge.points[edge.points.length - 1].y - endNode.y - offset.y) > 0.001 || | ||||
|             Math.abs(edge.points[edge.points.length - 1].x - x) > 0.001 | ||||
|           ) { | ||||
|             edge.points.push({ | ||||
|               x: endNode.x + endNode.width / 2 + offset.x, | ||||
|               y: endNode.y + endNode.height / 2 + offset.y, | ||||
|             }); | ||||
|           } | ||||
|         } | ||||
|  | ||||
|         edge.points = cutPathAtIntersect( | ||||
|           edge.points.reverse(), | ||||
|           { | ||||
|             x: startNode.x + startNode.width / 2 + offset.x, | ||||
|             y: startNode.y + startNode.height / 2 + offset.y, | ||||
|             width: sw, | ||||
|             height: startNode.height, | ||||
|             padding: startNode.padding, | ||||
|           }, | ||||
|           startNode.shape === 'diamond' | ||||
|         ).reverse(); | ||||
|  | ||||
|         edge.points = cutPathAtIntersect( | ||||
|           edge.points, | ||||
|           { | ||||
|             x: endNode.x + ew / 2 + endNode.offset.x, | ||||
|             y: endNode.y + endNode.height / 2 + endNode.offset.y, | ||||
|             width: ew, | ||||
|             height: endNode.height, | ||||
|             padding: endNode.padding, | ||||
|           }, | ||||
|           endNode.shape === 'diamond' | ||||
|         ); | ||||
|  | ||||
|         const paths = insertEdge( | ||||
|           edgesEl, | ||||
|           edge, | ||||
|           clusterDb, | ||||
|           data4Layout.type, | ||||
|           startNode, | ||||
|           endNode, | ||||
|           data4Layout.diagramId | ||||
|         ); | ||||
|         log.info('APA12 edge points after insert', JSON.stringify(edge.points)); | ||||
|  | ||||
|         edge.x = edge.labels[0].x + offset.x + edge.labels[0].width / 2; | ||||
|         edge.y = edge.labels[0].y + offset.y + edge.labels[0].height / 2; | ||||
|         positionEdgeLabel(edge, paths); | ||||
|       } | ||||
|     } | ||||
|   ); | ||||
| }; | ||||
| @@ -19,7 +19,8 @@ | ||||
|     "mermaid" | ||||
|   ], | ||||
|   "scripts": { | ||||
|     "clean": "rimraf dist" | ||||
|     "clean": "rimraf dist", | ||||
|     "prepublishOnly": "pnpm -w run build" | ||||
|   }, | ||||
|   "repository": { | ||||
|     "type": "git", | ||||
| @@ -39,7 +40,7 @@ | ||||
|     "mermaid": "workspace:^" | ||||
|   }, | ||||
|   "peerDependencies": { | ||||
|     "mermaid": "^10 || ^11" | ||||
|     "mermaid": "workspace:>=10.0.0" | ||||
|   }, | ||||
|   "files": [ | ||||
|     "dist" | ||||
|   | ||||
| @@ -1,32 +0,0 @@ | ||||
| # mermaid | ||||
|  | ||||
| ## 11.0.2 | ||||
|  | ||||
| ### Patch Changes | ||||
|  | ||||
| - [#5664](https://github.com/mermaid-js/mermaid/pull/5664) [`5deaef4`](https://github.com/mermaid-js/mermaid/commit/5deaef456e74d796866431c26f69360e4e74dbff) Thanks [@Austin-Fulbright](https://github.com/Austin-Fulbright)! - chore: Migrate git graph to langium, use typescript for internals | ||||
|  | ||||
| - Updated dependencies [[`5deaef4`](https://github.com/mermaid-js/mermaid/commit/5deaef456e74d796866431c26f69360e4e74dbff)]: | ||||
|   - @mermaid-js/parser@0.2.0 | ||||
|  | ||||
| ## 11.0.1 | ||||
|  | ||||
| ### Patch Changes | ||||
|  | ||||
| - [#2](https://github.com/calvinvette/mermaid/pull/2) [`bf05d87`](https://github.com/mermaid-js/mermaid/commit/bf05d8781edacb580fdb053da167e968b7570117) Thanks [@calvinvette](https://github.com/calvinvette)! - test changeset | ||||
|  | ||||
| ## 11.0.2 | ||||
|  | ||||
| ### Patch Changes | ||||
|  | ||||
| - Updated dependencies [[`83926c9`](https://github.com/mermaid-js/mermaid/commit/83926c9707b09c34e300888186250191ee8ae30a)]: | ||||
|   - @mermaid-js/parser@0.1.1 | ||||
|  | ||||
| ## 11.0.1 | ||||
|  | ||||
| ### Patch Changes | ||||
|  | ||||
| - [#5744](https://github.com/mermaid-js/mermaid/pull/5744) [`5013484`](https://github.com/mermaid-js/mermaid/commit/50134849246141ec400e33e08c12c10539b84de9) Thanks [@sidharthv96](https://github.com/sidharthv96)! - Release parser, test changesets | ||||
|  | ||||
| - Updated dependencies [[`5013484`](https://github.com/mermaid-js/mermaid/commit/50134849246141ec400e33e08c12c10539b84de9)]: | ||||
|   - @mermaid-js/parser@0.1.0 | ||||
| @@ -1,6 +1,6 @@ | ||||
| { | ||||
|   "name": "mermaid", | ||||
|   "version": "11.0.2", | ||||
|   "version": "11.0.0-alpha.7", | ||||
|   "description": "Markdown-ish syntax for generating flowcharts, mindmaps, sequence diagrams, class diagrams, gantt charts, git graphs and more.", | ||||
|   "type": "module", | ||||
|   "module": "./dist/mermaid.core.mjs", | ||||
| @@ -48,7 +48,8 @@ | ||||
|     "types:build-config": "tsx scripts/create-types-from-json-schema.mts", | ||||
|     "types:verify-config": "tsx scripts/create-types-from-json-schema.mts --verify", | ||||
|     "checkCircle": "npx madge --circular ./src", | ||||
|     "prepublishOnly": "pnpm docs:verify-version" | ||||
|     "release": "pnpm build", | ||||
|     "prepublishOnly": "cpy '../../README.*' ./ --cwd=. && pnpm docs:release-version && pnpm -w run build" | ||||
|   }, | ||||
|   "repository": { | ||||
|     "type": "git", | ||||
| @@ -76,11 +77,11 @@ | ||||
|     "dagre-d3-es": "7.0.10", | ||||
|     "dayjs": "^1.11.10", | ||||
|     "dompurify": "^3.0.11", | ||||
|     "elkjs": "^0.9.2", | ||||
|     "katex": "^0.16.9", | ||||
|     "khroma": "^2.1.0", | ||||
|     "lodash-es": "^4.17.21", | ||||
|     "marked": "^13.0.2", | ||||
|     "roughjs": "^4.6.6", | ||||
|     "mdast-util-from-markdown": "^2.0.0", | ||||
|     "stylis": "^4.3.1", | ||||
|     "ts-dedent": "^2.2.0", | ||||
|     "uuid": "^9.0.1" | ||||
| @@ -105,6 +106,7 @@ | ||||
|     "ajv": "^8.12.0", | ||||
|     "chokidar": "^3.6.0", | ||||
|     "concurrently": "^8.2.2", | ||||
|     "cpy-cli": "^5.0.0", | ||||
|     "csstree-validator": "^3.0.0", | ||||
|     "globby": "^14.0.1", | ||||
|     "jison": "^0.4.18", | ||||
| @@ -132,6 +134,7 @@ | ||||
|     "dist/", | ||||
|     "README.md" | ||||
|   ], | ||||
|   "sideEffects": false, | ||||
|   "publishConfig": { | ||||
|     "access": "public" | ||||
|   } | ||||
|   | ||||
| @@ -190,10 +190,7 @@ export const addDirective = (directive: MermaidConfig) => { | ||||
|  | ||||
|   // If the directive has a fontFamily, but no themeVariables, add the fontFamily to the themeVariables | ||||
|   if (directive.fontFamily && !directive.themeVariables?.fontFamily) { | ||||
|     directive.themeVariables = { | ||||
|       ...directive.themeVariables, | ||||
|       fontFamily: directive.fontFamily, | ||||
|     }; | ||||
|     directive.themeVariables = { fontFamily: directive.fontFamily }; | ||||
|   } | ||||
|  | ||||
|   directives.push(directive); | ||||
|   | ||||
| @@ -61,24 +61,9 @@ export interface MermaidConfig { | ||||
|    * You may also use `themeCSS` to override this value. | ||||
|    * | ||||
|    */ | ||||
|   theme?: 'default' | 'base' | 'dark' | 'forest' | 'neutral' | 'null'; | ||||
|   theme?: 'default' | 'forest' | 'dark' | 'neutral' | 'null'; | ||||
|   themeVariables?: any; | ||||
|   themeCSS?: string; | ||||
|   /** | ||||
|    * Defines which main look to use for the diagram. | ||||
|    * | ||||
|    */ | ||||
|   look?: 'classic' | 'handDrawn'; | ||||
|   /** | ||||
|    * Defines the seed to be used when using handDrawn look. This is important for the automated tests as they will always find differences without the seed. The default value is 0 which gives a random seed. | ||||
|    * | ||||
|    */ | ||||
|   handDrawnSeed?: number; | ||||
|   /** | ||||
|    * Defines which layout algorithm to use for rendering the diagram. | ||||
|    * | ||||
|    */ | ||||
|   layout?: string; | ||||
|   /** | ||||
|    * The maximum allowed size of the users text diagram | ||||
|    */ | ||||
| @@ -88,18 +73,6 @@ export interface MermaidConfig { | ||||
|    * | ||||
|    */ | ||||
|   maxEdges?: number; | ||||
|   elk?: { | ||||
|     /** | ||||
|      * Elk specific option that allows edges to share path where it convenient. It can make for pretty diagrams but can also make it harder to read the diagram. | ||||
|      * | ||||
|      */ | ||||
|     mergeEdges?: boolean; | ||||
|     /** | ||||
|      * Elk specific option affecting how nodes are placed. | ||||
|      * | ||||
|      */ | ||||
|     nodePlacementStrategy?: 'SIMPLE' | 'NETWORK_SIMPLEX' | 'LINEAR_SEGMENTS' | 'BRANDES_KOEPF'; | ||||
|   }; | ||||
|   darkMode?: boolean; | ||||
|   htmlLabels?: boolean; | ||||
|   /** | ||||
| @@ -724,8 +697,6 @@ export interface StateDiagramConfig extends BaseDiagramConfig { | ||||
|   textHeight?: number; | ||||
|   titleShift?: number; | ||||
|   noteMargin?: number; | ||||
|   nodeSpacing?: number; | ||||
|   rankSpacing?: number; | ||||
|   forkWidth?: number; | ||||
|   forkHeight?: number; | ||||
|   miniPadding?: number; | ||||
|   | ||||
| @@ -97,26 +97,25 @@ const recursiveRender = async (_elem, graph, diagramType, id, parentCluster, sit | ||||
|   // Also figure out which edges point to/from clusters and adjust them accordingly | ||||
|   // Edges from/to clusters really points to the first child in the cluster. | ||||
|   // TODO: pick optimal child in the cluster to us as link anchor | ||||
|   graph.edges().forEach(async function (e) { | ||||
|   graph.edges().forEach(function (e) { | ||||
|     const edge = graph.edge(e.v, e.w, e.name); | ||||
|     log.info('Edge ' + e.v + ' -> ' + e.w + ': ' + JSON.stringify(e)); | ||||
|     log.info('Edge ' + e.v + ' -> ' + e.w + ': ', e, ' ', JSON.stringify(graph.edge(e))); | ||||
|  | ||||
|     // Check if link is either from or to a cluster | ||||
|     log.info('Fix', clusterDb, 'ids:', e.v, e.w, 'Translating: ', clusterDb[e.v], clusterDb[e.w]); | ||||
|     await insertEdgeLabel(edgeLabels, edge); | ||||
|     insertEdgeLabel(edgeLabels, edge); | ||||
|   }); | ||||
|  | ||||
|   graph.edges().forEach(function (e) { | ||||
|     log.info('Edge ' + e.v + ' -> ' + e.w + ': ' + JSON.stringify(e)); | ||||
|   }); | ||||
|   log.info('Graph before layout:', JSON.stringify(graphlibJson.write(graph))); | ||||
|   log.info('#############################################'); | ||||
|   log.info('###                Layout                 ###'); | ||||
|   log.info('#############################################'); | ||||
|   log.info(graph); | ||||
|   dagreLayout(graph); | ||||
|   log.info('Graph after layout:', JSON.stringify(graphlibJson.write(graph))); | ||||
|   log.info('Graph after layout:', graphlibJson.write(graph)); | ||||
|   // Move the nodes to the correct place | ||||
|   let diff = 0; | ||||
|   const { subGraphTitleTotalMargin } = getSubGraphTitleMargins(siteConfig); | ||||
|   | ||||
| @@ -1,12 +1,12 @@ | ||||
| import { select } from 'd3'; | ||||
| import { getConfig } from '../diagram-api/diagramAPI.js'; | ||||
| import { evaluate } from '../diagrams/common/common.js'; | ||||
| import { log } from '../logger.js'; | ||||
| import { getArrowPoints } from './blockArrowHelper.js'; | ||||
| import createLabel from './createLabel.js'; | ||||
| import { labelHelper, updateNodeBounds, insertPolygonShape } from './shapes/util.js'; | ||||
| import { getConfig } from '../diagram-api/diagramAPI.js'; | ||||
| import intersect from './intersect/index.js'; | ||||
| import createLabel from './createLabel.js'; | ||||
| import note from './shapes/note.js'; | ||||
| import { insertPolygonShape, labelHelper, updateNodeBounds } from './shapes/util.js'; | ||||
| import { evaluate } from '../diagrams/common/common.js'; | ||||
| import { getArrowPoints } from './blockArrowHelper.js'; | ||||
|  | ||||
| const formatClass = (str) => { | ||||
|   if (str) { | ||||
| @@ -395,7 +395,6 @@ const rect = async (parent, node) => { | ||||
|   // add the rect | ||||
|   const rect = shapeSvg.insert('rect', ':first-child'); | ||||
|  | ||||
|   // console.log('Rect node:', node, 'bbox:', bbox, 'halfPadding:', halfPadding, 'node.padding:', node.padding); | ||||
|   // const totalWidth = bbox.width + node.padding * 2; | ||||
|   // const totalHeight = bbox.height + node.padding * 2; | ||||
|   const totalWidth = node.positioned ? node.width : bbox.width + node.padding; | ||||
| @@ -1155,6 +1154,9 @@ export const insertNode = async (elem, node, dir) => { | ||||
|   if (node.class) { | ||||
|     el.attr('class', 'node default ' + node.class); | ||||
|   } | ||||
|   // MC Special | ||||
|   newEl.attr('data-node', 'true'); | ||||
|   newEl.attr('data-id', node.id); | ||||
|  | ||||
|   nodeElems[node.id] = newEl; | ||||
|  | ||||
|   | ||||
| @@ -20,10 +20,6 @@ const config: RequiredDeep<MermaidConfig> = { | ||||
|   // Set, even though they're `undefined` so that `configKeys` finds these keys | ||||
|   // TODO: Should we replace these with `null` so that they can go in the JSON Schema? | ||||
|   deterministicIDSeed: undefined, | ||||
|   elk: { | ||||
|     mergeEdges: false, | ||||
|     nodePlacementStrategy: 'SIMPLE', | ||||
|   }, | ||||
|   themeCSS: undefined, | ||||
|  | ||||
|   // add non-JSON default config values | ||||
| @@ -248,6 +244,19 @@ const config: RequiredDeep<MermaidConfig> = { | ||||
|     ...defaultConfigJson.requirement, | ||||
|     useWidth: undefined, | ||||
|   }, | ||||
|   gitGraph: { | ||||
|     ...defaultConfigJson.gitGraph, | ||||
|     // TODO: This is a temporary override for `gitGraph`, since every other | ||||
|     //       diagram does have `useMaxWidth`, but instead sets it to `true`. | ||||
|     //       Should we set this to `true` instead? | ||||
|     useMaxWidth: false, | ||||
|   }, | ||||
|   sankey: { | ||||
|     ...defaultConfigJson.sankey, | ||||
|     // this is false, unlike every other diagram (other than gitGraph) | ||||
|     // TODO: can we make this default to `true` instead? | ||||
|     useMaxWidth: false, | ||||
|   }, | ||||
|   packet: { | ||||
|     ...defaultConfigJson.packet, | ||||
|   }, | ||||
|   | ||||
| @@ -129,6 +129,6 @@ export type HTML = d3.Selection<HTMLIFrameElement, unknown, Element | null, unkn | ||||
|  | ||||
| export type SVG = d3.Selection<SVGSVGElement, unknown, Element | null, unknown>; | ||||
|  | ||||
| export type SVGGroup = d3.Selection<SVGGElement, unknown, Element | null, unknown>; | ||||
| export type Group = d3.Selection<SVGGElement, unknown, Element | null, unknown>; | ||||
|  | ||||
| export type DiagramStylesProvider = (options?: any) => string; | ||||
|   | ||||
| @@ -8,6 +8,13 @@ import type { BlockDB } from './blockDB.js'; | ||||
| import { layout } from './layout.js'; | ||||
| import { calculateBlockSizes, insertBlocks, insertEdges } from './renderHelpers.js'; | ||||
|  | ||||
| /** | ||||
|  * Returns the all the styles from classDef statements in the graph definition. | ||||
|  * | ||||
|  * @param text - The text with the classes | ||||
|  * @param diagObj - The diagram object | ||||
|  * @returns ClassDef - The styles | ||||
|  */ | ||||
| export const getClasses = function (text: any, diagObj: any) { | ||||
|   return diagObj.db.getClasses(); | ||||
| }; | ||||
| @@ -38,6 +45,8 @@ export const draw = async function ( | ||||
|   const markers = ['point', 'circle', 'cross']; | ||||
|  | ||||
|   // Add the marker definitions to the svg as marker tags | ||||
|   // insertMarkers(svg, markers, diagObj.type, diagObj.arrowMarkerAbsolute); | ||||
|   // insertMarkers(svg, markers, diagObj.type, true); | ||||
|   insertMarkers(svg, markers, diagObj.type, id); | ||||
|  | ||||
|   const bl = db.getBlocks(); | ||||
| @@ -50,7 +59,11 @@ export const draw = async function ( | ||||
|   await insertBlocks(nodes, bl, db); | ||||
|   await insertEdges(nodes, edges, blArr, db, id); | ||||
|  | ||||
|   // log.debug('Here', bl); | ||||
|  | ||||
|   // Establish svg dimensions and get width and height | ||||
|   // | ||||
|   // const bounds2 = nodes.node().getBoundingClientRect(); | ||||
|   // Why, oh why ???? | ||||
|   if (bounds) { | ||||
|     const bounds2 = bounds; | ||||
|   | ||||
| @@ -1,6 +1,5 @@ | ||||
| import { sanitizeUrl } from '@braintree/sanitize-url'; | ||||
| import type { SVG, SVGGroup } from '../../diagram-api/types.js'; | ||||
| import { lineBreakRegex } from './common.js'; | ||||
| import type { Group, SVG } from '../../diagram-api/types.js'; | ||||
| import type { | ||||
|   Bound, | ||||
|   D3ImageElement, | ||||
| @@ -12,8 +11,9 @@ import type { | ||||
|   TextData, | ||||
|   TextObject, | ||||
| } from './commonTypes.js'; | ||||
| import { lineBreakRegex } from './common.js'; | ||||
|  | ||||
| export const drawRect = (element: SVG | SVGGroup, rectData: RectData): D3RectElement => { | ||||
| export const drawRect = (element: SVG | Group, rectData: RectData): D3RectElement => { | ||||
|   const rectElement: D3RectElement = element.append('rect'); | ||||
|   rectElement.attr('x', rectData.x); | ||||
|   rectElement.attr('y', rectData.y); | ||||
| @@ -50,7 +50,7 @@ export const drawRect = (element: SVG | SVGGroup, rectData: RectData): D3RectEle | ||||
|  * @param element - Diagram (reference for bounds) | ||||
|  * @param bounds - Shape of the rectangle | ||||
|  */ | ||||
| export const drawBackgroundRect = (element: SVG | SVGGroup, bounds: Bound): void => { | ||||
| export const drawBackgroundRect = (element: SVG | Group, bounds: Bound): void => { | ||||
|   const rectData: RectData = { | ||||
|     x: bounds.startx, | ||||
|     y: bounds.starty, | ||||
| @@ -64,7 +64,7 @@ export const drawBackgroundRect = (element: SVG | SVGGroup, bounds: Bound): void | ||||
|   rectElement.lower(); | ||||
| }; | ||||
|  | ||||
| export const drawText = (element: SVG | SVGGroup, textData: TextData): D3TextElement => { | ||||
| export const drawText = (element: SVG | Group, textData: TextData): D3TextElement => { | ||||
|   const nText: string = textData.text.replace(lineBreakRegex, ' '); | ||||
|  | ||||
|   const textElem: D3TextElement = element.append('text'); | ||||
| @@ -84,7 +84,7 @@ export const drawText = (element: SVG | SVGGroup, textData: TextData): D3TextEle | ||||
|   return textElem; | ||||
| }; | ||||
|  | ||||
| export const drawImage = (elem: SVG | SVGGroup, x: number, y: number, link: string): void => { | ||||
| export const drawImage = (elem: SVG | Group, x: number, y: number, link: string): void => { | ||||
|   const imageElement: D3ImageElement = elem.append('image'); | ||||
|   imageElement.attr('x', x); | ||||
|   imageElement.attr('y', y); | ||||
| @@ -93,7 +93,7 @@ export const drawImage = (elem: SVG | SVGGroup, x: number, y: number, link: stri | ||||
| }; | ||||
|  | ||||
| export const drawEmbeddedImage = ( | ||||
|   element: SVG | SVGGroup, | ||||
|   element: SVG | Group, | ||||
|   x: number, | ||||
|   y: number, | ||||
|   link: string | ||||
|   | ||||
| @@ -1,5 +1,5 @@ | ||||
| import type { SVG, SVGGroup } from '../../diagram-api/types.js'; | ||||
| import { log } from '../../logger.js'; | ||||
| import type { Group, SVG } from '../../diagram-api/types.js'; | ||||
| import { selectSvgElement } from '../../rendering-util/selectSvgElement.js'; | ||||
| import { configureSvgSize } from '../../setupGraphViewbox.js'; | ||||
|  | ||||
| @@ -13,7 +13,7 @@ import { configureSvgSize } from '../../setupGraphViewbox.js'; | ||||
| export const draw = (_text: string, id: string, version: string) => { | ||||
|   log.debug('rendering svg for syntax error\n'); | ||||
|   const svg: SVG = selectSvgElement(id); | ||||
|   const g: SVGGroup = svg.append('g'); | ||||
|   const g: Group = svg.append('g'); | ||||
|  | ||||
|   svg.attr('viewBox', '0 0 2412 512'); | ||||
|   configureSvgSize(svg, 100, 512, true); | ||||
|   | ||||
| @@ -1,26 +1,34 @@ | ||||
| import type { | ||||
|   ExternalDiagramDefinition, | ||||
|   DiagramDetector, | ||||
|   DiagramLoader, | ||||
|   ExternalDiagramDefinition, | ||||
| } from '../../../diagram-api/types.js'; | ||||
| import { log } from '../../../logger.js'; | ||||
|  | ||||
| const id = 'flowchart-elk'; | ||||
|  | ||||
| const detector: DiagramDetector = (txt, config = {}): boolean => { | ||||
| const detector: DiagramDetector = (txt, config): boolean => { | ||||
|   if ( | ||||
|     // If diagram explicitly states flowchart-elk | ||||
|     /^\s*flowchart-elk/.test(txt) || | ||||
|     // If a flowchart/graph diagram has their default renderer set to elk | ||||
|     (/^\s*flowchart|graph/.test(txt) && config?.flowchart?.defaultRenderer === 'elk') | ||||
|   ) { | ||||
|     config.layout = 'elk'; | ||||
|     // This will log at the end, hopefully. | ||||
|     setTimeout( | ||||
|       () => | ||||
|         log.warn( | ||||
|           'flowchart-elk was moved to an external package in Mermaid v11. Please refer [release notes](link) for more details. This diagram will be rendered using `dagre` layout as a fallback.' | ||||
|         ), | ||||
|       500 | ||||
|     ); | ||||
|     return true; | ||||
|   } | ||||
|   return false; | ||||
| }; | ||||
|  | ||||
| const loader: DiagramLoader = async () => { | ||||
|   const { diagram } = await import('../flowDiagram.js'); | ||||
|   const { diagram } = await import('../flowDiagram-v2.js'); | ||||
|   return { id, diagram }; | ||||
| }; | ||||
|  | ||||
|   | ||||
| @@ -1,8 +1,7 @@ | ||||
| import { select } from 'd3'; | ||||
| import utils, { getEdgeId } from '../../utils.js'; | ||||
| import utils from '../../utils.js'; | ||||
| import { getConfig, defaultConfig } from '../../diagram-api/diagramAPI.js'; | ||||
| import common from '../common/common.js'; | ||||
| import type { Node, Edge } from '../../rendering-util/types.js'; | ||||
| import { log } from '../../logger.js'; | ||||
| import { | ||||
|   setAccTitle, | ||||
| @@ -201,17 +200,10 @@ export const updateLink = function (positions: ('default' | number)[], style: st | ||||
|     if (pos === 'default') { | ||||
|       edges.defaultStyle = style; | ||||
|     } else { | ||||
|       // if (utils.isSubstringInArray('fill', style) === -1) { | ||||
|       //   style.push('fill:none'); | ||||
|       // } | ||||
|       edges[pos].style = style; | ||||
|       // if edges[pos].style does have fill not set, set it to none | ||||
|       if ( | ||||
|         (edges[pos]?.style?.length ?? 0) > 0 && | ||||
|         !edges[pos]?.style?.some((s) => s?.startsWith('fill')) | ||||
|       ) { | ||||
|         edges[pos]?.style?.push('fill:none'); | ||||
|       if (utils.isSubstringInArray('fill', style) === -1) { | ||||
|         style.push('fill:none'); | ||||
|       } | ||||
|       edges[pos].style = style; | ||||
|     } | ||||
|   }); | ||||
| }; | ||||
| @@ -227,7 +219,7 @@ export const addClass = function (ids: string, style: string[]) { | ||||
|     if (style !== undefined && style !== null) { | ||||
|       style.forEach(function (s) { | ||||
|         if (/color/.exec(s)) { | ||||
|           const newStyle = s.replace('fill', 'bgFill'); // .replace('color', 'fill'); | ||||
|           const newStyle = s.replace('fill', 'bgFill').replace('color', 'fill'); | ||||
|           classNode.textStyles.push(newStyle); | ||||
|         } | ||||
|         classNode.styles.push(s); | ||||
| @@ -736,12 +728,14 @@ export const destructLink = (_str: string, _startStr: string) => { | ||||
|  | ||||
| // Todo optimizer this by caching existing nodes | ||||
| const exists = (allSgs: FlowSubGraph[], _id: string) => { | ||||
|   for (const sg of allSgs) { | ||||
|     if (sg.nodes.includes(_id)) { | ||||
|       return true; | ||||
|   let res = false; | ||||
|   allSgs.forEach((sg) => { | ||||
|     const pos = sg.nodes.indexOf(_id); | ||||
|     if (pos >= 0) { | ||||
|       res = true; | ||||
|     } | ||||
|   } | ||||
|   return false; | ||||
|   }); | ||||
|   return res; | ||||
| }; | ||||
| /** | ||||
|  * Deletes an id from all subgraphs | ||||
| @@ -761,174 +755,11 @@ export const lex = { | ||||
|   firstGraph, | ||||
| }; | ||||
|  | ||||
| const getTypeFromVertex = (vertex: FlowVertex) => { | ||||
|   if (vertex.type === 'square') { | ||||
|     return 'squareRect'; | ||||
|   } | ||||
|   if (vertex.type === 'round') { | ||||
|     return 'roundedRect'; | ||||
|   } | ||||
|  | ||||
|   return vertex.type ?? 'squareRect'; | ||||
| }; | ||||
|  | ||||
| const findNode = (nodes: Node[], id: string) => nodes.find((node) => node.id === id); | ||||
| const destructEdgeType = (type: string | undefined) => { | ||||
|   let arrowTypeStart = 'none'; | ||||
|   let arrowTypeEnd = 'arrow_point'; | ||||
|   switch (type) { | ||||
|     case 'arrow_point': | ||||
|     case 'arrow_circle': | ||||
|     case 'arrow_cross': | ||||
|       arrowTypeEnd = type; | ||||
|       break; | ||||
|  | ||||
|     case 'double_arrow_point': | ||||
|     case 'double_arrow_circle': | ||||
|     case 'double_arrow_cross': | ||||
|       arrowTypeStart = type.replace('double_', ''); | ||||
|       arrowTypeEnd = arrowTypeStart; | ||||
|       break; | ||||
|   } | ||||
|   return { arrowTypeStart, arrowTypeEnd }; | ||||
| }; | ||||
|  | ||||
| const addNodeFromVertex = ( | ||||
|   vertex: FlowVertex, | ||||
|   nodes: Node[], | ||||
|   parentDB: Map<string, string>, | ||||
|   subGraphDB: Map<string, boolean>, | ||||
|   config: any, | ||||
|   look: string | ||||
| ) => { | ||||
|   const parentId = parentDB.get(vertex.id); | ||||
|   const isGroup = subGraphDB.get(vertex.id) ?? false; | ||||
|  | ||||
|   const node = findNode(nodes, vertex.id); | ||||
|   if (node) { | ||||
|     node.cssStyles = vertex.styles; | ||||
|     node.cssCompiledStyles = getCompiledStyles(vertex.classes); | ||||
|     node.cssClasses = vertex.classes.join(' '); | ||||
|   } else { | ||||
|     nodes.push({ | ||||
|       id: vertex.id, | ||||
|       label: vertex.text, | ||||
|       labelStyle: '', | ||||
|       parentId, | ||||
|       padding: config.flowchart?.padding || 8, | ||||
|       cssStyles: vertex.styles, | ||||
|       cssCompiledStyles: getCompiledStyles(['default', 'node', ...vertex.classes]), | ||||
|       cssClasses: 'default ' + vertex.classes.join(' '), | ||||
|       shape: getTypeFromVertex(vertex), | ||||
|       dir: vertex.dir, | ||||
|       domId: vertex.domId, | ||||
|       isGroup, | ||||
|       look, | ||||
|       link: vertex.link, | ||||
|       linkTarget: vertex.linkTarget, | ||||
|       tooltip: getTooltip(vertex.id), | ||||
|     }); | ||||
|   } | ||||
| }; | ||||
|  | ||||
| function getCompiledStyles(classDefs: string[]) { | ||||
|   let compiledStyles: string[] = []; | ||||
|   for (const customClass of classDefs) { | ||||
|     const cssClass = classes.get(customClass); | ||||
|     if (cssClass?.styles) { | ||||
|       compiledStyles = [...compiledStyles, ...(cssClass.styles ?? [])].map((s) => s.trim()); | ||||
|     } | ||||
|     if (cssClass?.textStyles) { | ||||
|       compiledStyles = [...compiledStyles, ...(cssClass.textStyles ?? [])].map((s) => s.trim()); | ||||
|     } | ||||
|   } | ||||
|   return compiledStyles; | ||||
| } | ||||
|  | ||||
| export const getData = () => { | ||||
|   const config = getConfig(); | ||||
|   const nodes: Node[] = []; | ||||
|   const edges: Edge[] = []; | ||||
|  | ||||
|   const subGraphs = getSubGraphs(); | ||||
|   const parentDB = new Map<string, string>(); | ||||
|   const subGraphDB = new Map<string, boolean>(); | ||||
|  | ||||
|   // Setup the subgraph data for adding nodes | ||||
|   for (let i = subGraphs.length - 1; i >= 0; i--) { | ||||
|     const subGraph = subGraphs[i]; | ||||
|     if (subGraph.nodes.length > 0) { | ||||
|       subGraphDB.set(subGraph.id, true); | ||||
|     } | ||||
|     for (const id of subGraph.nodes) { | ||||
|       parentDB.set(id, subGraph.id); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   // Data is setup, add the nodes | ||||
|   for (let i = subGraphs.length - 1; i >= 0; i--) { | ||||
|     const subGraph = subGraphs[i]; | ||||
|     nodes.push({ | ||||
|       id: subGraph.id, | ||||
|       label: subGraph.title, | ||||
|       labelStyle: '', | ||||
|       parentId: parentDB.get(subGraph.id), | ||||
|       padding: 8, | ||||
|       cssCompiledStyles: getCompiledStyles(subGraph.classes), | ||||
|       cssClasses: subGraph.classes.join(' '), | ||||
|       shape: 'rect', | ||||
|       dir: subGraph.dir, | ||||
|       isGroup: true, | ||||
|       look: config.look, | ||||
|     }); | ||||
|   } | ||||
|  | ||||
|   const n = getVertices(); | ||||
|   n.forEach((vertex) => { | ||||
|     addNodeFromVertex(vertex, nodes, parentDB, subGraphDB, config, config.look || 'classic'); | ||||
|   }); | ||||
|  | ||||
|   const e = getEdges(); | ||||
|   e.forEach((rawEdge, index) => { | ||||
|     const { arrowTypeStart, arrowTypeEnd } = destructEdgeType(rawEdge.type); | ||||
|     const styles = [...(e.defaultStyle ?? [])]; | ||||
|  | ||||
|     if (rawEdge.style) { | ||||
|       styles.push(...rawEdge.style); | ||||
|     } | ||||
|     const edge: Edge = { | ||||
|       id: getEdgeId(rawEdge.start, rawEdge.end, { counter: index, prefix: 'L' }), | ||||
|       start: rawEdge.start, | ||||
|       end: rawEdge.end, | ||||
|       type: rawEdge.type ?? 'normal', | ||||
|       label: rawEdge.text, | ||||
|       labelpos: 'c', | ||||
|       thickness: rawEdge.stroke, | ||||
|       minlen: rawEdge.length, | ||||
|       classes: | ||||
|         rawEdge?.stroke === 'invisible' | ||||
|           ? '' | ||||
|           : 'edge-thickness-normal edge-pattern-solid flowchart-link', | ||||
|       arrowTypeStart: rawEdge?.stroke === 'invisible' ? 'none' : arrowTypeStart, | ||||
|       arrowTypeEnd: rawEdge?.stroke === 'invisible' ? 'none' : arrowTypeEnd, | ||||
|       arrowheadStyle: 'fill: #333', | ||||
|       labelStyle: styles, | ||||
|       style: styles, | ||||
|       pattern: rawEdge.stroke, | ||||
|       look: config.look, | ||||
|     }; | ||||
|     edges.push(edge); | ||||
|   }); | ||||
|  | ||||
|   return { nodes, edges, other: {}, config }; | ||||
| }; | ||||
|  | ||||
| export default { | ||||
|   defaultConfig: () => defaultConfig.flowchart, | ||||
|   setAccTitle, | ||||
|   getAccTitle, | ||||
|   getAccDescription, | ||||
|   getData, | ||||
|   setAccDescription, | ||||
|   addVertex, | ||||
|   lookUpDomId, | ||||
|   | ||||
| @@ -1,8 +1,5 @@ | ||||
| import type { | ||||
|   DiagramDetector, | ||||
|   DiagramLoader, | ||||
|   ExternalDiagramDefinition, | ||||
| } from '../../diagram-api/types.js'; | ||||
| import type { DiagramDetector, DiagramLoader } from '../../diagram-api/types.js'; | ||||
| import type { ExternalDiagramDefinition } from '../../diagram-api/types.js'; | ||||
|  | ||||
| const id = 'flowchart-v2'; | ||||
|  | ||||
| @@ -22,7 +19,7 @@ const detector: DiagramDetector = (txt, config) => { | ||||
| }; | ||||
|  | ||||
| const loader: DiagramLoader = async () => { | ||||
|   const { diagram } = await import('./flowDiagram.js'); | ||||
|   const { diagram } = await import('./flowDiagram-v2.js'); | ||||
|   return { id, diagram }; | ||||
| }; | ||||
|  | ||||
|   | ||||
							
								
								
									
										25
									
								
								packages/mermaid/src/diagrams/flowchart/flowDiagram-v2.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										25
									
								
								packages/mermaid/src/diagrams/flowchart/flowDiagram-v2.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,25 @@ | ||||
| // @ts-ignore: JISON doesn't support types | ||||
| import flowParser from './parser/flow.jison'; | ||||
| import flowDb from './flowDb.js'; | ||||
| import flowRendererV2 from './flowRenderer-v2.js'; | ||||
| import flowStyles from './styles.js'; | ||||
| import type { MermaidConfig } from '../../config.type.js'; | ||||
| import { setConfig } from '../../diagram-api/diagramAPI.js'; | ||||
|  | ||||
| export const diagram = { | ||||
|   parser: flowParser, | ||||
|   db: flowDb, | ||||
|   renderer: flowRendererV2, | ||||
|   styles: flowStyles, | ||||
|   init: (cnf: MermaidConfig) => { | ||||
|     if (!cnf.flowchart) { | ||||
|       cnf.flowchart = {}; | ||||
|     } | ||||
|     cnf.flowchart.arrowMarkerAbsolute = cnf.arrowMarkerAbsolute; | ||||
|     // flowchart-v2 uses dagre-wrapper, which doesn't have access to flowchart cnf | ||||
|     setConfig({ flowchart: { arrowMarkerAbsolute: cnf.arrowMarkerAbsolute } }); | ||||
|     flowRendererV2.setConf(cnf.flowchart); | ||||
|     flowDb.clear(); | ||||
|     flowDb.setGen('gen-2'); | ||||
|   }, | ||||
| }; | ||||
| @@ -1,26 +1,24 @@ | ||||
| import type { MermaidConfig } from '../../config.type.js'; | ||||
| import { setConfig } from '../../diagram-api/diagramAPI.js'; | ||||
| import flowDb from './flowDb.js'; | ||||
| import renderer from './flowRenderer-v3-unified.js'; | ||||
| // @ts-ignore: JISON doesn't support types | ||||
| import flowParser from './parser/flow.jison'; | ||||
| import flowDb from './flowDb.js'; | ||||
| import flowRenderer from './flowRenderer.js'; | ||||
| import flowRendererV2 from './flowRenderer-v2.js'; | ||||
| import flowStyles from './styles.js'; | ||||
| import type { MermaidConfig } from '../../config.type.js'; | ||||
|  | ||||
| export const diagram = { | ||||
|   parser: flowParser, | ||||
|   db: flowDb, | ||||
|   renderer, | ||||
|   renderer: flowRendererV2, | ||||
|   styles: flowStyles, | ||||
|   init: (cnf: MermaidConfig) => { | ||||
|     if (!cnf.flowchart) { | ||||
|       cnf.flowchart = {}; | ||||
|     } | ||||
|     if (cnf.layout) { | ||||
|       setConfig({ layout: cnf.layout }); | ||||
|     } | ||||
|     // TODO, broken as of 2022-09-14 (13809b50251845475e6dca65cc395761be38fbd2) | ||||
|     cnf.flowchart.arrowMarkerAbsolute = cnf.arrowMarkerAbsolute; | ||||
|     setConfig({ flowchart: { arrowMarkerAbsolute: cnf.arrowMarkerAbsolute } }); | ||||
|     flowRenderer.setConf(cnf.flowchart); | ||||
|     flowDb.clear(); | ||||
|     flowDb.setGen('gen-2'); | ||||
|     flowDb.setGen('gen-1'); | ||||
|   }, | ||||
| }; | ||||
|   | ||||
							
								
								
									
										515
									
								
								packages/mermaid/src/diagrams/flowchart/flowRenderer-v2.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										515
									
								
								packages/mermaid/src/diagrams/flowchart/flowRenderer-v2.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,515 @@ | ||||
| import * as graphlib from 'dagre-d3-es/src/graphlib/index.js'; | ||||
| import { select, curveLinear, selectAll } from 'd3'; | ||||
| import { getConfig } from '../../diagram-api/diagramAPI.js'; | ||||
| import utils, { getEdgeId } from '../../utils.js'; | ||||
| import { render } from '../../dagre-wrapper/index.js'; | ||||
| import { addHtmlLabel } from 'dagre-d3-es/src/dagre-js/label/add-html-label.js'; | ||||
| import { log } from '../../logger.js'; | ||||
| import common, { evaluate, renderKatex } from '../common/common.js'; | ||||
| import { interpolateToCurve, getStylesFromArray } from '../../utils.js'; | ||||
| import { setupGraphViewbox } from '../../setupGraphViewbox.js'; | ||||
|  | ||||
| const conf = {}; | ||||
| export const setConf = function (cnf) { | ||||
|   const keys = Object.keys(cnf); | ||||
|   for (const key of keys) { | ||||
|     conf[key] = cnf[key]; | ||||
|   } | ||||
| }; | ||||
|  | ||||
| /** | ||||
|  * Function that adds the vertices found during parsing to the graph to be rendered. | ||||
|  * | ||||
|  * @param vert Object containing the vertices. | ||||
|  * @param g The graph that is to be drawn. | ||||
|  * @param svgId | ||||
|  * @param root | ||||
|  * @param doc | ||||
|  * @param diagObj | ||||
|  */ | ||||
| export const addVertices = async function (vert, g, svgId, root, doc, diagObj) { | ||||
|   const svg = root.select(`[id="${svgId}"]`); | ||||
|   const keys = vert.keys(); | ||||
|  | ||||
|   // Iterate through each item in the vertex object (containing all the vertices found) in the graph definition | ||||
|   for (const id of keys) { | ||||
|     const vertex = vert.get(id); | ||||
|  | ||||
|     /** | ||||
|      * Variable for storing the classes for the vertex | ||||
|      * | ||||
|      * @type {string} | ||||
|      */ | ||||
|     let classStr = 'default'; | ||||
|     if (vertex.classes.length > 0) { | ||||
|       classStr = vertex.classes.join(' '); | ||||
|     } | ||||
|     classStr = classStr + ' flowchart-label'; | ||||
|     const styles = getStylesFromArray(vertex.styles); | ||||
|  | ||||
|     // Use vertex id as text in the box if no text is provided by the graph definition | ||||
|     let vertexText = vertex.text !== undefined ? vertex.text : vertex.id; | ||||
|  | ||||
|     // We create a SVG label, either by delegating to addHtmlLabel or manually | ||||
|     let vertexNode; | ||||
|     log.info('vertex', vertex, vertex.labelType); | ||||
|     if (vertex.labelType === 'markdown') { | ||||
|       log.info('vertex', vertex, vertex.labelType); | ||||
|     } else { | ||||
|       if (evaluate(getConfig().flowchart.htmlLabels)) { | ||||
|         // TODO: addHtmlLabel accepts a labelStyle. Do we possibly have that? | ||||
|         const node = { | ||||
|           label: vertexText, | ||||
|         }; | ||||
|         vertexNode = addHtmlLabel(svg, node).node(); | ||||
|         vertexNode.parentNode.removeChild(vertexNode); | ||||
|       } else { | ||||
|         const svgLabel = doc.createElementNS('http://www.w3.org/2000/svg', 'text'); | ||||
|         svgLabel.setAttribute('style', styles.labelStyle.replace('color:', 'fill:')); | ||||
|  | ||||
|         const rows = vertexText.split(common.lineBreakRegex); | ||||
|  | ||||
|         for (const row of rows) { | ||||
|           const tspan = doc.createElementNS('http://www.w3.org/2000/svg', 'tspan'); | ||||
|           tspan.setAttributeNS('http://www.w3.org/XML/1998/namespace', 'xml:space', 'preserve'); | ||||
|           tspan.setAttribute('dy', '1em'); | ||||
|           tspan.setAttribute('x', '1'); | ||||
|           tspan.textContent = row; | ||||
|           svgLabel.appendChild(tspan); | ||||
|         } | ||||
|         vertexNode = svgLabel; | ||||
|       } | ||||
|     } | ||||
|  | ||||
|     let radius = 0; | ||||
|     let _shape = ''; | ||||
|     // Set the shape based parameters | ||||
|     switch (vertex.type) { | ||||
|       case 'round': | ||||
|         radius = 5; | ||||
|         _shape = 'rect'; | ||||
|         break; | ||||
|       case 'square': | ||||
|         _shape = 'rect'; | ||||
|         break; | ||||
|       case 'diamond': | ||||
|         _shape = 'question'; | ||||
|         break; | ||||
|       case 'hexagon': | ||||
|         _shape = 'hexagon'; | ||||
|         break; | ||||
|       case 'odd': | ||||
|         _shape = 'rect_left_inv_arrow'; | ||||
|         break; | ||||
|       case 'lean_right': | ||||
|         _shape = 'lean_right'; | ||||
|         break; | ||||
|       case 'lean_left': | ||||
|         _shape = 'lean_left'; | ||||
|         break; | ||||
|       case 'trapezoid': | ||||
|         _shape = 'trapezoid'; | ||||
|         break; | ||||
|       case 'inv_trapezoid': | ||||
|         _shape = 'inv_trapezoid'; | ||||
|         break; | ||||
|       case 'odd_right': | ||||
|         _shape = 'rect_left_inv_arrow'; | ||||
|         break; | ||||
|       case 'circle': | ||||
|         _shape = 'circle'; | ||||
|         break; | ||||
|       case 'ellipse': | ||||
|         _shape = 'ellipse'; | ||||
|         break; | ||||
|       case 'stadium': | ||||
|         _shape = 'stadium'; | ||||
|         break; | ||||
|       case 'subroutine': | ||||
|         _shape = 'subroutine'; | ||||
|         break; | ||||
|       case 'cylinder': | ||||
|         _shape = 'cylinder'; | ||||
|         break; | ||||
|       case 'group': | ||||
|         _shape = 'rect'; | ||||
|         break; | ||||
|       case 'doublecircle': | ||||
|         _shape = 'doublecircle'; | ||||
|         break; | ||||
|       default: | ||||
|         _shape = 'rect'; | ||||
|     } | ||||
|     const labelText = await renderKatex(vertexText, getConfig()); | ||||
|  | ||||
|     // Add the node | ||||
|     g.setNode(vertex.id, { | ||||
|       labelStyle: styles.labelStyle, | ||||
|       shape: _shape, | ||||
|       labelText, | ||||
|       labelType: vertex.labelType, | ||||
|       rx: radius, | ||||
|       ry: radius, | ||||
|       class: classStr, | ||||
|       style: styles.style, | ||||
|       id: vertex.id, | ||||
|       link: vertex.link, | ||||
|       linkTarget: vertex.linkTarget, | ||||
|       tooltip: diagObj.db.getTooltip(vertex.id) || '', | ||||
|       domId: diagObj.db.lookUpDomId(vertex.id), | ||||
|       haveCallback: vertex.haveCallback, | ||||
|       width: vertex.type === 'group' ? 500 : undefined, | ||||
|       dir: vertex.dir, | ||||
|       type: vertex.type, | ||||
|       props: vertex.props, | ||||
|       padding: getConfig().flowchart.padding, | ||||
|     }); | ||||
|  | ||||
|     log.info('setNode', { | ||||
|       labelStyle: styles.labelStyle, | ||||
|       labelType: vertex.labelType, | ||||
|       shape: _shape, | ||||
|       labelText, | ||||
|       rx: radius, | ||||
|       ry: radius, | ||||
|       class: classStr, | ||||
|       style: styles.style, | ||||
|       id: vertex.id, | ||||
|       domId: diagObj.db.lookUpDomId(vertex.id), | ||||
|       width: vertex.type === 'group' ? 500 : undefined, | ||||
|       type: vertex.type, | ||||
|       dir: vertex.dir, | ||||
|       props: vertex.props, | ||||
|       padding: getConfig().flowchart.padding, | ||||
|     }); | ||||
|   } | ||||
| }; | ||||
|  | ||||
| /** | ||||
|  * Add edges to graph based on parsed graph definition | ||||
|  * | ||||
|  * @param {object} edges The edges to add to the graph | ||||
|  * @param {object} g The graph object | ||||
|  * @param _diagObj | ||||
|  */ | ||||
| export const addEdges = async function (edges, g, _diagObj) { | ||||
|   log.info('abc78 edges = ', edges); | ||||
|   let cnt = 0; | ||||
|   let linkIdCnt = {}; | ||||
|  | ||||
|   let defaultStyle; | ||||
|   let defaultLabelStyle; | ||||
|  | ||||
|   if (edges.defaultStyle !== undefined) { | ||||
|     const defaultStyles = getStylesFromArray(edges.defaultStyle); | ||||
|     defaultStyle = defaultStyles.style; | ||||
|     defaultLabelStyle = defaultStyles.labelStyle; | ||||
|   } | ||||
|  | ||||
|   for (const edge of edges) { | ||||
|     cnt++; | ||||
|  | ||||
|     // Identify Link | ||||
|     const linkIdBase = getEdgeId(edge.start, edge.end, { | ||||
|       counter: cnt, | ||||
|       prefix: 'L', | ||||
|     }); | ||||
|  | ||||
|     // count the links from+to the same node to give unique id | ||||
|     if (linkIdCnt[linkIdBase] === undefined) { | ||||
|       linkIdCnt[linkIdBase] = 0; | ||||
|       log.info('abc78 new entry', linkIdBase, linkIdCnt[linkIdBase]); | ||||
|     } else { | ||||
|       linkIdCnt[linkIdBase]++; | ||||
|       log.info('abc78 new entry', linkIdBase, linkIdCnt[linkIdBase]); | ||||
|     } | ||||
|     let linkId = `${linkIdBase}_${linkIdCnt[linkIdBase]}`; | ||||
|  | ||||
|     log.info('abc78 new link id to be used is', linkIdBase, linkId, linkIdCnt[linkIdBase]); | ||||
|     const linkNameStart = 'LS-' + edge.start; | ||||
|     const linkNameEnd = 'LE-' + edge.end; | ||||
|  | ||||
|     const edgeData = { style: '', labelStyle: '' }; | ||||
|     edgeData.minlen = edge.length || 1; | ||||
|     //edgeData.id = 'id' + cnt; | ||||
|  | ||||
|     // Set link type for rendering | ||||
|     if (edge.type === 'arrow_open') { | ||||
|       edgeData.arrowhead = 'none'; | ||||
|     } else { | ||||
|       edgeData.arrowhead = 'normal'; | ||||
|     } | ||||
|  | ||||
|     // Check of arrow types, placed here in order not to break old rendering | ||||
|     edgeData.arrowTypeStart = 'arrow_open'; | ||||
|     edgeData.arrowTypeEnd = 'arrow_open'; | ||||
|  | ||||
|     /* eslint-disable no-fallthrough */ | ||||
|     switch (edge.type) { | ||||
|       case 'double_arrow_cross': | ||||
|         edgeData.arrowTypeStart = 'arrow_cross'; | ||||
|       case 'arrow_cross': | ||||
|         edgeData.arrowTypeEnd = 'arrow_cross'; | ||||
|         break; | ||||
|       case 'double_arrow_point': | ||||
|         edgeData.arrowTypeStart = 'arrow_point'; | ||||
|       case 'arrow_point': | ||||
|         edgeData.arrowTypeEnd = 'arrow_point'; | ||||
|         break; | ||||
|       case 'double_arrow_circle': | ||||
|         edgeData.arrowTypeStart = 'arrow_circle'; | ||||
|       case 'arrow_circle': | ||||
|         edgeData.arrowTypeEnd = 'arrow_circle'; | ||||
|         break; | ||||
|     } | ||||
|  | ||||
|     let style = ''; | ||||
|     let labelStyle = ''; | ||||
|  | ||||
|     switch (edge.stroke) { | ||||
|       case 'normal': | ||||
|         style = 'fill:none;'; | ||||
|         if (defaultStyle !== undefined) { | ||||
|           style = defaultStyle; | ||||
|         } | ||||
|         if (defaultLabelStyle !== undefined) { | ||||
|           labelStyle = defaultLabelStyle; | ||||
|         } | ||||
|         edgeData.thickness = 'normal'; | ||||
|         edgeData.pattern = 'solid'; | ||||
|         break; | ||||
|       case 'dotted': | ||||
|         edgeData.thickness = 'normal'; | ||||
|         edgeData.pattern = 'dotted'; | ||||
|         edgeData.style = 'fill:none;stroke-width:2px;stroke-dasharray:3;'; | ||||
|         break; | ||||
|       case 'thick': | ||||
|         edgeData.thickness = 'thick'; | ||||
|         edgeData.pattern = 'solid'; | ||||
|         edgeData.style = 'stroke-width: 3.5px;fill:none;'; | ||||
|         break; | ||||
|       case 'invisible': | ||||
|         edgeData.thickness = 'invisible'; | ||||
|         edgeData.pattern = 'solid'; | ||||
|         edgeData.style = 'stroke-width: 0;fill:none;'; | ||||
|         break; | ||||
|     } | ||||
|     if (edge.style !== undefined) { | ||||
|       const styles = getStylesFromArray(edge.style); | ||||
|       style = styles.style; | ||||
|       labelStyle = styles.labelStyle; | ||||
|     } | ||||
|  | ||||
|     edgeData.style = edgeData.style += style; | ||||
|     edgeData.labelStyle = edgeData.labelStyle += labelStyle; | ||||
|  | ||||
|     if (edge.interpolate !== undefined) { | ||||
|       edgeData.curve = interpolateToCurve(edge.interpolate, curveLinear); | ||||
|     } else if (edges.defaultInterpolate !== undefined) { | ||||
|       edgeData.curve = interpolateToCurve(edges.defaultInterpolate, curveLinear); | ||||
|     } else { | ||||
|       edgeData.curve = interpolateToCurve(conf.curve, curveLinear); | ||||
|     } | ||||
|  | ||||
|     if (edge.text === undefined) { | ||||
|       if (edge.style !== undefined) { | ||||
|         edgeData.arrowheadStyle = 'fill: #333'; | ||||
|       } | ||||
|     } else { | ||||
|       edgeData.arrowheadStyle = 'fill: #333'; | ||||
|       edgeData.labelpos = 'c'; | ||||
|     } | ||||
|     edgeData.labelType = edge.labelType; | ||||
|     edgeData.label = await renderKatex(edge.text.replace(common.lineBreakRegex, '\n'), getConfig()); | ||||
|  | ||||
|     if (edge.style === undefined) { | ||||
|       edgeData.style = edgeData.style || 'stroke: #333; stroke-width: 1.5px;fill:none;'; | ||||
|     } | ||||
|  | ||||
|     edgeData.labelStyle = edgeData.labelStyle.replace('color:', 'fill:'); | ||||
|  | ||||
|     edgeData.id = linkId; | ||||
|     edgeData.classes = 'flowchart-link ' + linkNameStart + ' ' + linkNameEnd; | ||||
|  | ||||
|     // Add the edge to the graph | ||||
|     g.setEdge(edge.start, edge.end, edgeData, cnt); | ||||
|   } | ||||
| }; | ||||
|  | ||||
| /** | ||||
|  * Returns the all the styles from classDef statements in the graph definition. | ||||
|  * | ||||
|  * @param text | ||||
|  * @param diagObj | ||||
|  * @returns {Map<string, import('../../diagram-api/types.js').DiagramStyleClassDef>} ClassDef styles | ||||
|  */ | ||||
| export const getClasses = function (text, diagObj) { | ||||
|   return diagObj.db.getClasses(); | ||||
| }; | ||||
|  | ||||
| /** | ||||
|  * Draws a flowchart in the tag with id: id based on the graph definition in text. | ||||
|  * | ||||
|  * @param text | ||||
|  * @param id | ||||
|  * @param _version | ||||
|  * @param diagObj | ||||
|  */ | ||||
|  | ||||
| export const draw = async function (text, id, _version, diagObj) { | ||||
|   log.info('Drawing flowchart'); | ||||
|  | ||||
|   // Fetch the default direction, use TD if none was found | ||||
|   let dir = diagObj.db.getDirection(); | ||||
|   if (dir === undefined) { | ||||
|     dir = 'TD'; | ||||
|   } | ||||
|  | ||||
|   const { securityLevel, flowchart: conf } = getConfig(); | ||||
|   const nodeSpacing = conf.nodeSpacing ?? 50; | ||||
|   const rankSpacing = conf.rankSpacing ?? 50; | ||||
|  | ||||
|   // Handle root and document for when rendering in sandbox mode | ||||
|   let sandboxElement; | ||||
|   if (securityLevel === 'sandbox') { | ||||
|     sandboxElement = select('#i' + id); | ||||
|   } | ||||
|   const root = | ||||
|     securityLevel === 'sandbox' | ||||
|       ? select(sandboxElement.nodes()[0].contentDocument.body) | ||||
|       : select('body'); | ||||
|   const doc = securityLevel === 'sandbox' ? sandboxElement.nodes()[0].contentDocument : document; | ||||
|  | ||||
|   // Create the input mermaid.graph | ||||
|   const g = new graphlib.Graph({ | ||||
|     multigraph: true, | ||||
|     compound: true, | ||||
|   }) | ||||
|     .setGraph({ | ||||
|       rankdir: dir, | ||||
|       nodesep: nodeSpacing, | ||||
|       ranksep: rankSpacing, | ||||
|       marginx: 0, | ||||
|       marginy: 0, | ||||
|     }) | ||||
|     .setDefaultEdgeLabel(function () { | ||||
|       return {}; | ||||
|     }); | ||||
|  | ||||
|   let subG; | ||||
|   const subGraphs = diagObj.db.getSubGraphs(); | ||||
|   log.info('Subgraphs - ', subGraphs); | ||||
|   for (let i = subGraphs.length - 1; i >= 0; i--) { | ||||
|     subG = subGraphs[i]; | ||||
|     log.info('Subgraph - ', subG); | ||||
|     diagObj.db.addVertex( | ||||
|       subG.id, | ||||
|       { text: subG.title, type: subG.labelType }, | ||||
|       'group', | ||||
|       undefined, | ||||
|       subG.classes, | ||||
|       subG.dir | ||||
|     ); | ||||
|   } | ||||
|  | ||||
|   // Fetch the vertices/nodes and edges/links from the parsed graph definition | ||||
|   const vert = diagObj.db.getVertices(); | ||||
|  | ||||
|   const edges = diagObj.db.getEdges(); | ||||
|  | ||||
|   log.info('Edges', edges); | ||||
|   let i = 0; | ||||
|   for (i = subGraphs.length - 1; i >= 0; i--) { | ||||
|     subG = subGraphs[i]; | ||||
|  | ||||
|     selectAll('cluster').append('text'); | ||||
|  | ||||
|     for (const node of subG.nodes) { | ||||
|       log.info('Setting up subgraphs', node, subG.id); | ||||
|       g.setParent(node, subG.id); | ||||
|     } | ||||
|   } | ||||
|   await addVertices(vert, g, id, root, doc, diagObj); | ||||
|   await addEdges(edges, g, diagObj); | ||||
|  | ||||
|   // Add custom shapes | ||||
|   // flowChartShapes.addToRenderV2(addShape); | ||||
|  | ||||
|   // Set up an SVG group so that we can translate the final graph. | ||||
|   const svg = root.select(`[id="${id}"]`); | ||||
|  | ||||
|   // Run the renderer. This is what draws the final graph. | ||||
|   const element = root.select('#' + id + ' g'); | ||||
|   await render(element, g, ['point', 'circle', 'cross'], 'flowchart', id); | ||||
|  | ||||
|   utils.insertTitle(svg, 'flowchartTitleText', conf.titleTopMargin, diagObj.db.getDiagramTitle()); | ||||
|  | ||||
|   setupGraphViewbox(g, svg, conf.diagramPadding, conf.useMaxWidth); | ||||
|  | ||||
|   // Index nodes | ||||
|   diagObj.db.indexNodes('subGraph' + i); | ||||
|  | ||||
|   // Add label rects for non html labels | ||||
|   if (!conf.htmlLabels) { | ||||
|     const labels = doc.querySelectorAll('[id="' + id + '"] .edgeLabel .label'); | ||||
|     for (const label of labels) { | ||||
|       // Get dimensions of label | ||||
|       const dim = label.getBBox(); | ||||
|  | ||||
|       const rect = doc.createElementNS('http://www.w3.org/2000/svg', 'rect'); | ||||
|       rect.setAttribute('rx', 0); | ||||
|       rect.setAttribute('ry', 0); | ||||
|       rect.setAttribute('width', dim.width); | ||||
|       rect.setAttribute('height', dim.height); | ||||
|  | ||||
|       label.insertBefore(rect, label.firstChild); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   // If node has a link, wrap it in an anchor SVG object. | ||||
|   const keys = [...vert.keys()]; | ||||
|   keys.forEach((key) => { | ||||
|     const vertex = vert.get(key); | ||||
|  | ||||
|     if (vertex.link) { | ||||
|       const node = select('#' + id + ' [id="' + key + '"]'); | ||||
|       if (node) { | ||||
|         const link = doc.createElementNS('http://www.w3.org/2000/svg', 'a'); | ||||
|         link.setAttributeNS('http://www.w3.org/2000/svg', 'class', vertex.classes.join(' ')); | ||||
|         link.setAttributeNS('http://www.w3.org/2000/svg', 'href', vertex.link); | ||||
|         link.setAttributeNS('http://www.w3.org/2000/svg', 'rel', 'noopener'); | ||||
|         if (securityLevel === 'sandbox') { | ||||
|           link.setAttributeNS('http://www.w3.org/2000/svg', 'target', '_top'); | ||||
|         } else if (vertex.linkTarget) { | ||||
|           link.setAttributeNS('http://www.w3.org/2000/svg', 'target', vertex.linkTarget); | ||||
|         } | ||||
|  | ||||
|         const linkNode = node.insert(function () { | ||||
|           return link; | ||||
|         }, ':first-child'); | ||||
|  | ||||
|         const shape = node.select('.label-container'); | ||||
|         if (shape) { | ||||
|           linkNode.append(function () { | ||||
|             return shape.node(); | ||||
|           }); | ||||
|         } | ||||
|  | ||||
|         const label = node.select('.label'); | ||||
|         if (label) { | ||||
|           linkNode.append(function () { | ||||
|             return label.node(); | ||||
|           }); | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|   }); | ||||
| }; | ||||
|  | ||||
| export default { | ||||
|   setConf, | ||||
|   addVertices, | ||||
|   addEdges, | ||||
|   getClasses, | ||||
|   draw, | ||||
| }; | ||||
| @@ -1,104 +0,0 @@ | ||||
| import { select } from 'd3'; | ||||
| import { getConfig } from '../../diagram-api/diagramAPI.js'; | ||||
| import type { DiagramStyleClassDef } from '../../diagram-api/types.js'; | ||||
| import { log } from '../../logger.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 utils from '../../utils.js'; | ||||
| import { getDirection } from './flowDb.js'; | ||||
|  | ||||
| export const getClasses = function ( | ||||
|   text: string, | ||||
|   diagramObj: any | ||||
| ): Map<string, DiagramStyleClassDef> { | ||||
|   return diagramObj.db.getClasses(); | ||||
| }; | ||||
|  | ||||
| export const draw = async function (text: string, id: string, _version: string, diag: any) { | ||||
|   log.info('REF0:'); | ||||
|   log.info('Drawing state diagram (v2)', id); | ||||
|   const { securityLevel, flowchart: conf, layout } = getConfig(); | ||||
|  | ||||
|   // Handle root and document for when rendering in sandbox mode | ||||
|   let sandboxElement; | ||||
|   if (securityLevel === 'sandbox') { | ||||
|     sandboxElement = select('#i' + id); | ||||
|   } | ||||
|  | ||||
|   // @ts-ignore - document is always available | ||||
|   const doc = securityLevel === 'sandbox' ? sandboxElement.nodes()[0].contentDocument : document; | ||||
|  | ||||
|   // The getData method provided in all supported diagrams is used to extract the data from the parsed structure | ||||
|   // into the Layout data format | ||||
|   log.debug('Before getData: '); | ||||
|   const data4Layout = diag.db.getData() as LayoutData; | ||||
|   log.debug('Data: ', data4Layout); | ||||
|   // Create the root SVG | ||||
|   const svg = getDiagramElement(id, securityLevel); | ||||
|   const direction = getDirection(); | ||||
|  | ||||
|   data4Layout.type = diag.type; | ||||
|   data4Layout.layoutAlgorithm = getRegisteredLayoutAlgorithm(layout); | ||||
|   if (data4Layout.layoutAlgorithm === 'dagre' && layout === 'elk') { | ||||
|     log.warn( | ||||
|       'flowchart-elk was moved to an external package in Mermaid v11. Please refer [release notes](https://github.com/mermaid-js/mermaid/releases/tag/v11.0.0) for more details. This diagram will be rendered using `dagre` layout as a fallback.' | ||||
|     ); | ||||
|   } | ||||
|   data4Layout.direction = direction; | ||||
|   data4Layout.nodeSpacing = conf?.nodeSpacing || 50; | ||||
|   data4Layout.rankSpacing = conf?.rankSpacing || 50; | ||||
|   data4Layout.markers = ['point', 'circle', 'cross']; | ||||
|  | ||||
|   data4Layout.diagramId = id; | ||||
|   log.debug('REF1:', data4Layout); | ||||
|   await render(data4Layout, svg); | ||||
|   const padding = data4Layout.config.flowchart?.diagramPadding ?? 8; | ||||
|   utils.insertTitle( | ||||
|     svg, | ||||
|     'flowchartTitleText', | ||||
|     conf?.titleTopMargin || 0, | ||||
|     diag.db.getDiagramTitle() | ||||
|   ); | ||||
|   setupViewPortForSVG(svg, padding, 'flowchart', conf?.useMaxWidth || false); | ||||
|  | ||||
|   // If node has a link, wrap it in an anchor SVG object. | ||||
|   for (const vertex of data4Layout.nodes) { | ||||
|     const node = select(`#${id} [id="${vertex.id}"]`); | ||||
|     if (!node || !vertex.link) { | ||||
|       continue; | ||||
|     } | ||||
|     const link = doc.createElementNS('http://www.w3.org/2000/svg', 'a'); | ||||
|     link.setAttributeNS('http://www.w3.org/2000/svg', 'class', vertex.cssClasses); | ||||
|     link.setAttributeNS('http://www.w3.org/2000/svg', 'rel', 'noopener'); | ||||
|     if (securityLevel === 'sandbox') { | ||||
|       link.setAttributeNS('http://www.w3.org/2000/svg', 'target', '_top'); | ||||
|     } else if (vertex.linkTarget) { | ||||
|       link.setAttributeNS('http://www.w3.org/2000/svg', 'target', vertex.linkTarget); | ||||
|     } | ||||
|  | ||||
|     const linkNode = node.insert(function () { | ||||
|       return link; | ||||
|     }, ':first-child'); | ||||
|  | ||||
|     const shape = node.select('.label-container'); | ||||
|     if (shape) { | ||||
|       linkNode.append(function () { | ||||
|         return shape.node(); | ||||
|       }); | ||||
|     } | ||||
|  | ||||
|     const label = node.select('.label'); | ||||
|     if (label) { | ||||
|       linkNode.append(function () { | ||||
|         return label.node(); | ||||
|       }); | ||||
|     } | ||||
|   } | ||||
| }; | ||||
|  | ||||
| export default { | ||||
|   getClasses, | ||||
|   draw, | ||||
| }; | ||||
| @@ -0,0 +1,150 @@ | ||||
| import flowDb from './flowDb.js'; | ||||
| import { parser } from './parser/flow.jison'; | ||||
| import flowRenderer from './flowRenderer.js'; | ||||
| import { addDiagrams } from '../../diagram-api/diagram-orchestration.js'; | ||||
|  | ||||
| const diag = { | ||||
|   db: flowDb, | ||||
| }; | ||||
| addDiagrams(); | ||||
|  | ||||
| describe('when using mermaid and ', function () { | ||||
|   describe('when calling addEdges ', function () { | ||||
|     beforeEach(function () { | ||||
|       parser.yy = flowDb; | ||||
|       flowDb.clear(); | ||||
|       flowDb.setGen('gen-2'); | ||||
|     }); | ||||
|     it('should handle edges with text', async () => { | ||||
|       parser.parse('graph TD;A-->|text ex|B;'); | ||||
|       flowDb.getVertices(); | ||||
|       const edges = flowDb.getEdges(); | ||||
|  | ||||
|       const mockG = { | ||||
|         setEdge: function (start, end, options) { | ||||
|           expect(start).toContain('flowchart-A-'); | ||||
|           expect(end).toContain('flowchart-B-'); | ||||
|           expect(options.arrowhead).toBe('normal'); | ||||
|           expect(options.label.match('text ex')).toBeTruthy(); | ||||
|         }, | ||||
|       }; | ||||
|  | ||||
|       await flowRenderer.addEdges(edges, mockG, diag); | ||||
|     }); | ||||
|  | ||||
|     it('should handle edges without text', async function () { | ||||
|       parser.parse('graph TD;A-->B;'); | ||||
|       flowDb.getVertices(); | ||||
|       const edges = flowDb.getEdges(); | ||||
|  | ||||
|       const mockG = { | ||||
|         setEdge: function (start, end, options) { | ||||
|           expect(start).toContain('flowchart-A-'); | ||||
|           expect(end).toContain('flowchart-B-'); | ||||
|           expect(options.arrowhead).toBe('normal'); | ||||
|         }, | ||||
|       }; | ||||
|  | ||||
|       await flowRenderer.addEdges(edges, mockG, diag); | ||||
|     }); | ||||
|  | ||||
|     it('should handle open-ended edges', async () => { | ||||
|       parser.parse('graph TD;A---B;'); | ||||
|       flowDb.getVertices(); | ||||
|       const edges = flowDb.getEdges(); | ||||
|  | ||||
|       const mockG = { | ||||
|         setEdge: function (start, end, options) { | ||||
|           expect(start).toContain('flowchart-A-'); | ||||
|           expect(end).toContain('flowchart-B-'); | ||||
|           expect(options.arrowhead).toBe('none'); | ||||
|         }, | ||||
|       }; | ||||
|  | ||||
|       await flowRenderer.addEdges(edges, mockG, diag); | ||||
|     }); | ||||
|  | ||||
|     it('should handle edges with styles defined', async () => { | ||||
|       parser.parse('graph TD;A---B; linkStyle 0 stroke:val1,stroke-width:val2;'); | ||||
|       flowDb.getVertices(); | ||||
|       const edges = flowDb.getEdges(); | ||||
|  | ||||
|       const mockG = { | ||||
|         setEdge: function (start, end, options) { | ||||
|           expect(start).toContain('flowchart-A-'); | ||||
|           expect(end).toContain('flowchart-B-'); | ||||
|           expect(options.arrowhead).toBe('none'); | ||||
|           expect(options.style).toBe('stroke:val1;stroke-width:val2;fill:none;'); | ||||
|         }, | ||||
|       }; | ||||
|  | ||||
|       await flowRenderer.addEdges(edges, mockG, diag); | ||||
|     }); | ||||
|     it('should handle edges with interpolation defined', async () => { | ||||
|       parser.parse('graph TD;A---B; linkStyle 0 interpolate basis'); | ||||
|       flowDb.getVertices(); | ||||
|       const edges = flowDb.getEdges(); | ||||
|  | ||||
|       const mockG = { | ||||
|         setEdge: function (start, end, options) { | ||||
|           expect(start).toContain('flowchart-A-'); | ||||
|           expect(end).toContain('flowchart-B-'); | ||||
|           expect(options.arrowhead).toBe('none'); | ||||
|           expect(options.curve).toBe('basis'); // mocked as string | ||||
|         }, | ||||
|       }; | ||||
|  | ||||
|       await flowRenderer.addEdges(edges, mockG, diag); | ||||
|     }); | ||||
|     it('should handle edges with text and styles defined', async () => { | ||||
|       parser.parse('graph TD;A---|the text|B; linkStyle 0 stroke:val1,stroke-width:val2;'); | ||||
|       flowDb.getVertices(); | ||||
|       const edges = flowDb.getEdges(); | ||||
|  | ||||
|       const mockG = { | ||||
|         setEdge: function (start, end, options) { | ||||
|           expect(start).toContain('flowchart-A-'); | ||||
|           expect(end).toContain('flowchart-B-'); | ||||
|           expect(options.arrowhead).toBe('none'); | ||||
|           expect(options.label.match('the text')).toBeTruthy(); | ||||
|           expect(options.style).toBe('stroke:val1;stroke-width:val2;fill:none;'); | ||||
|         }, | ||||
|       }; | ||||
|  | ||||
|       await flowRenderer.addEdges(edges, mockG, diag); | ||||
|     }); | ||||
|  | ||||
|     it('should set fill to "none" by default when handling edges', async () => { | ||||
|       parser.parse('graph TD;A---B; linkStyle 0 stroke:val1,stroke-width:val2;'); | ||||
|       flowDb.getVertices(); | ||||
|       const edges = flowDb.getEdges(); | ||||
|  | ||||
|       const mockG = { | ||||
|         setEdge: function (start, end, options) { | ||||
|           expect(start).toContain('flowchart-A-'); | ||||
|           expect(end).toContain('flowchart-B'); | ||||
|           expect(options.arrowhead).toBe('none'); | ||||
|           expect(options.style).toBe('stroke:val1;stroke-width:val2;fill:none;'); | ||||
|         }, | ||||
|       }; | ||||
|  | ||||
|       await flowRenderer.addEdges(edges, mockG, diag); | ||||
|     }); | ||||
|  | ||||
|     it('should not set fill to none if fill is set in linkStyle', async () => { | ||||
|       parser.parse('graph TD;A---B; linkStyle 0 stroke:val1,stroke-width:val2,fill:blue;'); | ||||
|       flowDb.getVertices(); | ||||
|       const edges = flowDb.getEdges(); | ||||
|       const mockG = { | ||||
|         setEdge: function (start, end, options) { | ||||
|           expect(start).toContain('flowchart-A-'); | ||||
|           expect(end).toContain('flowchart-B-'); | ||||
|           expect(options.arrowhead).toBe('none'); | ||||
|           expect(options.style).toBe('stroke:val1;stroke-width:val2;fill:blue;'); | ||||
|         }, | ||||
|       }; | ||||
|  | ||||
|       await flowRenderer.addEdges(edges, mockG, diag); | ||||
|     }); | ||||
|   }); | ||||
| }); | ||||
							
								
								
									
										503
									
								
								packages/mermaid/src/diagrams/flowchart/flowRenderer.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										503
									
								
								packages/mermaid/src/diagrams/flowchart/flowRenderer.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,503 @@ | ||||
| import * as graphlib from 'dagre-d3-es/src/graphlib/index.js'; | ||||
| import { select, curveLinear, selectAll } from 'd3'; | ||||
| import { getConfig } from '../../diagram-api/diagramAPI.js'; | ||||
| import { render as Render } from 'dagre-d3-es'; | ||||
| import { applyStyle } from 'dagre-d3-es/src/dagre-js/util.js'; | ||||
| import { addHtmlLabel } from 'dagre-d3-es/src/dagre-js/label/add-html-label.js'; | ||||
| import { log } from '../../logger.js'; | ||||
| import common, { evaluate, renderKatex } from '../common/common.js'; | ||||
| import { interpolateToCurve, getStylesFromArray, getEdgeId } from '../../utils.js'; | ||||
| import { setupGraphViewbox } from '../../setupGraphViewbox.js'; | ||||
| import flowChartShapes from './flowChartShapes.js'; | ||||
| import { replaceIconSubstring } from '../../rendering-util/createText.js'; | ||||
|  | ||||
| const conf = {}; | ||||
| export const setConf = function (cnf) { | ||||
|   const keys = Object.keys(cnf); | ||||
|   for (const key of keys) { | ||||
|     conf[key] = cnf[key]; | ||||
|   } | ||||
| }; | ||||
|  | ||||
| /** | ||||
|  * Function that adds the vertices found in the graph definition to the graph to be rendered. | ||||
|  * | ||||
|  * @param vert Object containing the vertices. | ||||
|  * @param g The graph that is to be drawn. | ||||
|  * @param svgId | ||||
|  * @param root | ||||
|  * @param _doc | ||||
|  * @param diagObj | ||||
|  */ | ||||
| export const addVertices = async function (vert, g, svgId, root, _doc, diagObj) { | ||||
|   const svg = !root ? select(`[id="${svgId}"]`) : root.select(`[id="${svgId}"]`); | ||||
|   const doc = !_doc ? document : _doc; | ||||
|   const keys = Object.keys(vert); | ||||
|  | ||||
|   // Iterate through each item in the vertex object (containing all the vertices found) in the graph definition | ||||
|   for (const id of keys) { | ||||
|     const vertex = vert[id]; | ||||
|  | ||||
|     /** | ||||
|      * Variable for storing the classes for the vertex | ||||
|      * | ||||
|      * @type {string} | ||||
|      */ | ||||
|     let classStr = 'default'; | ||||
|     if (vertex.classes.length > 0) { | ||||
|       classStr = vertex.classes.join(' '); | ||||
|     } | ||||
|  | ||||
|     const styles = getStylesFromArray(vertex.styles); | ||||
|  | ||||
|     // Use vertex id as text in the box if no text is provided by the graph definition | ||||
|     let vertexText = vertex.text !== undefined ? vertex.text : vertex.id; | ||||
|  | ||||
|     // We create a SVG label, either by delegating to addHtmlLabel or manually | ||||
|     let vertexNode; | ||||
|     if (evaluate(getConfig().flowchart.htmlLabels)) { | ||||
|       // TODO: addHtmlLabel accepts a labelStyle. Do we possibly have that? | ||||
|       const replacedVertexText = replaceIconSubstring(vertexText); | ||||
|       const node = { | ||||
|         label: await renderKatex(replacedVertexText, getConfig()), | ||||
|       }; | ||||
|       vertexNode = addHtmlLabel(svg, node).node(); | ||||
|       vertexNode.parentNode.removeChild(vertexNode); | ||||
|     } else { | ||||
|       const svgLabel = doc.createElementNS('http://www.w3.org/2000/svg', 'text'); | ||||
|       svgLabel.setAttribute('style', styles.labelStyle.replace('color:', 'fill:')); | ||||
|  | ||||
|       const rows = vertexText.split(common.lineBreakRegex); | ||||
|  | ||||
|       for (const row of rows) { | ||||
|         const tspan = doc.createElementNS('http://www.w3.org/2000/svg', 'tspan'); | ||||
|         tspan.setAttributeNS('http://www.w3.org/XML/1998/namespace', 'xml:space', 'preserve'); | ||||
|         tspan.setAttribute('dy', '1em'); | ||||
|         tspan.setAttribute('x', '1'); | ||||
|         tspan.textContent = row; | ||||
|         svgLabel.appendChild(tspan); | ||||
|       } | ||||
|       vertexNode = svgLabel; | ||||
|     } | ||||
|  | ||||
|     let radius = 0; | ||||
|     let _shape = ''; | ||||
|     // Set the shape based parameters | ||||
|     switch (vertex.type) { | ||||
|       case 'round': | ||||
|         radius = 5; | ||||
|         _shape = 'rect'; | ||||
|         break; | ||||
|       case 'square': | ||||
|         _shape = 'rect'; | ||||
|         break; | ||||
|       case 'diamond': | ||||
|         _shape = 'question'; | ||||
|         break; | ||||
|       case 'hexagon': | ||||
|         _shape = 'hexagon'; | ||||
|         break; | ||||
|       case 'odd': | ||||
|         _shape = 'rect_left_inv_arrow'; | ||||
|         break; | ||||
|       case 'lean_right': | ||||
|         _shape = 'lean_right'; | ||||
|         break; | ||||
|       case 'lean_left': | ||||
|         _shape = 'lean_left'; | ||||
|         break; | ||||
|       case 'trapezoid': | ||||
|         _shape = 'trapezoid'; | ||||
|         break; | ||||
|       case 'inv_trapezoid': | ||||
|         _shape = 'inv_trapezoid'; | ||||
|         break; | ||||
|       case 'odd_right': | ||||
|         _shape = 'rect_left_inv_arrow'; | ||||
|         break; | ||||
|       case 'circle': | ||||
|         _shape = 'circle'; | ||||
|         break; | ||||
|       case 'ellipse': | ||||
|         _shape = 'ellipse'; | ||||
|         break; | ||||
|       case 'stadium': | ||||
|         _shape = 'stadium'; | ||||
|         break; | ||||
|       case 'subroutine': | ||||
|         _shape = 'subroutine'; | ||||
|         break; | ||||
|       case 'cylinder': | ||||
|         _shape = 'cylinder'; | ||||
|         break; | ||||
|       case 'group': | ||||
|         _shape = 'rect'; | ||||
|         break; | ||||
|       default: | ||||
|         _shape = 'rect'; | ||||
|     } | ||||
|     // Add the node | ||||
|     log.warn('Adding node', vertex.id, vertex.domId); | ||||
|     g.setNode(diagObj.db.lookUpDomId(vertex.id), { | ||||
|       labelType: 'svg', | ||||
|       labelStyle: styles.labelStyle, | ||||
|       shape: _shape, | ||||
|       label: vertexNode, | ||||
|       rx: radius, | ||||
|       ry: radius, | ||||
|       class: classStr, | ||||
|       style: styles.style, | ||||
|       id: diagObj.db.lookUpDomId(vertex.id), | ||||
|     }); | ||||
|   } | ||||
| }; | ||||
|  | ||||
| /** | ||||
|  * Add edges to graph based on parsed graph definition | ||||
|  * | ||||
|  * @param {object} edges The edges to add to the graph | ||||
|  * @param {object} g The graph object | ||||
|  * @param diagObj | ||||
|  */ | ||||
| export const addEdges = async function (edges, g, diagObj) { | ||||
|   let cnt = 0; | ||||
|  | ||||
|   let defaultStyle; | ||||
|   let defaultLabelStyle; | ||||
|  | ||||
|   if (edges.defaultStyle !== undefined) { | ||||
|     const defaultStyles = getStylesFromArray(edges.defaultStyle); | ||||
|     defaultStyle = defaultStyles.style; | ||||
|     defaultLabelStyle = defaultStyles.labelStyle; | ||||
|   } | ||||
|  | ||||
|   for (const edge of edges) { | ||||
|     cnt++; | ||||
|  | ||||
|     // Identify Link | ||||
|     const linkId = getEdgeId(edge.start, edge.end, { | ||||
|       counter: cnt, | ||||
|       prefix: 'L', | ||||
|     }); | ||||
|     const linkNameStart = 'LS-' + edge.start; | ||||
|     const linkNameEnd = 'LE-' + edge.end; | ||||
|  | ||||
|     const edgeData = {}; | ||||
|  | ||||
|     // Set link type for rendering | ||||
|     if (edge.type === 'arrow_open') { | ||||
|       edgeData.arrowhead = 'none'; | ||||
|     } else { | ||||
|       edgeData.arrowhead = 'normal'; | ||||
|     } | ||||
|  | ||||
|     let style = ''; | ||||
|     let labelStyle = ''; | ||||
|  | ||||
|     if (edge.style !== undefined) { | ||||
|       const styles = getStylesFromArray(edge.style); | ||||
|       style = styles.style; | ||||
|       labelStyle = styles.labelStyle; | ||||
|     } else { | ||||
|       switch (edge.stroke) { | ||||
|         case 'normal': | ||||
|           style = 'fill:none'; | ||||
|           if (defaultStyle !== undefined) { | ||||
|             style = defaultStyle; | ||||
|           } | ||||
|           if (defaultLabelStyle !== undefined) { | ||||
|             labelStyle = defaultLabelStyle; | ||||
|           } | ||||
|           break; | ||||
|         case 'dotted': | ||||
|           style = 'fill:none;stroke-width:2px;stroke-dasharray:3;'; | ||||
|           break; | ||||
|         case 'thick': | ||||
|           style = ' stroke-width: 3.5px;fill:none'; | ||||
|           break; | ||||
|       } | ||||
|     } | ||||
|  | ||||
|     edgeData.style = style; | ||||
|     edgeData.labelStyle = labelStyle; | ||||
|  | ||||
|     if (edge.interpolate !== undefined) { | ||||
|       edgeData.curve = interpolateToCurve(edge.interpolate, curveLinear); | ||||
|     } else if (edges.defaultInterpolate !== undefined) { | ||||
|       edgeData.curve = interpolateToCurve(edges.defaultInterpolate, curveLinear); | ||||
|     } else { | ||||
|       edgeData.curve = interpolateToCurve(conf.curve, curveLinear); | ||||
|     } | ||||
|  | ||||
|     if (edge.text === undefined) { | ||||
|       if (edge.style !== undefined) { | ||||
|         edgeData.arrowheadStyle = 'fill: #333'; | ||||
|       } | ||||
|     } else { | ||||
|       edgeData.arrowheadStyle = 'fill: #333'; | ||||
|       edgeData.labelpos = 'c'; | ||||
|  | ||||
|       if (evaluate(getConfig().flowchart.htmlLabels)) { | ||||
|         edgeData.labelType = 'html'; | ||||
|         edgeData.label = `<span id="L-${linkId}" class="edgeLabel L-${linkNameStart}' L-${linkNameEnd}" style="${ | ||||
|           edgeData.labelStyle | ||||
|         }">${await renderKatex(replaceIconSubstring(edge.text), getConfig())}</span>`; | ||||
|       } else { | ||||
|         edgeData.labelType = 'text'; | ||||
|         edgeData.label = edge.text.replace(common.lineBreakRegex, '\n'); | ||||
|  | ||||
|         if (edge.style === undefined) { | ||||
|           edgeData.style = edgeData.style || 'stroke: #333; stroke-width: 1.5px;fill:none'; | ||||
|         } | ||||
|  | ||||
|         edgeData.labelStyle = edgeData.labelStyle.replace('color:', 'fill:'); | ||||
|       } | ||||
|     } | ||||
|  | ||||
|     edgeData.id = linkId; | ||||
|     edgeData.class = linkNameStart + ' ' + linkNameEnd; | ||||
|     edgeData.minlen = edge.length || 1; | ||||
|  | ||||
|     // Add the edge to the graph | ||||
|     g.setEdge(diagObj.db.lookUpDomId(edge.start), diagObj.db.lookUpDomId(edge.end), edgeData, cnt); | ||||
|   } | ||||
| }; | ||||
|  | ||||
| /** | ||||
|  * Returns the all the styles from classDef statements in the graph definition. | ||||
|  * | ||||
|  * @param text | ||||
|  * @param diagObj | ||||
|  * @returns {Map<string, import('../../diagram-api/types.js').DiagramStyleClassDef>} ClassDef styles | ||||
|  */ | ||||
| export const getClasses = function (text, diagObj) { | ||||
|   log.info('Extracting classes'); | ||||
|   return diagObj.db.getClasses(); | ||||
| }; | ||||
|  | ||||
| /** | ||||
|  * Draws a flowchart in the tag with id: id based on the graph definition in text. | ||||
|  * | ||||
|  * @param text | ||||
|  * @param id | ||||
|  * @param _version | ||||
|  * @param diagObj | ||||
|  */ | ||||
| export const draw = async function (text, id, _version, diagObj) { | ||||
|   log.info('Drawing flowchart'); | ||||
|   const { securityLevel, flowchart: conf } = getConfig(); | ||||
|   let sandboxElement; | ||||
|   if (securityLevel === 'sandbox') { | ||||
|     sandboxElement = select('#i' + id); | ||||
|   } | ||||
|   const root = | ||||
|     securityLevel === 'sandbox' | ||||
|       ? select(sandboxElement.nodes()[0].contentDocument.body) | ||||
|       : select('body'); | ||||
|   const doc = securityLevel === 'sandbox' ? sandboxElement.nodes()[0].contentDocument : document; | ||||
|  | ||||
|   // Fetch the default direction, use TD if none was found | ||||
|   let dir = diagObj.db.getDirection(); | ||||
|   if (dir === undefined) { | ||||
|     dir = 'TD'; | ||||
|   } | ||||
|   const nodeSpacing = conf.nodeSpacing ?? 50; | ||||
|   const rankSpacing = conf.rankSpacing ?? 50; | ||||
|  | ||||
|   // Create the input mermaid.graph | ||||
|   const g = new graphlib.Graph({ | ||||
|     multigraph: true, | ||||
|     compound: true, | ||||
|   }) | ||||
|     .setGraph({ | ||||
|       rankdir: dir, | ||||
|       nodesep: nodeSpacing, | ||||
|       ranksep: rankSpacing, | ||||
|       marginx: 8, | ||||
|       marginy: 8, | ||||
|     }) | ||||
|     .setDefaultEdgeLabel(function () { | ||||
|       return {}; | ||||
|     }); | ||||
|  | ||||
|   let subG; | ||||
|   const subGraphs = diagObj.db.getSubGraphs(); | ||||
|   for (let i = subGraphs.length - 1; i >= 0; i--) { | ||||
|     subG = subGraphs[i]; | ||||
|     diagObj.db.addVertex(subG.id, subG.title, 'group', undefined, subG.classes); | ||||
|   } | ||||
|  | ||||
|   // Fetch the vertices/nodes and edges/links from the parsed graph definition | ||||
|   const vert = diagObj.db.getVertices(); | ||||
|   log.warn('Get vertices', vert); | ||||
|  | ||||
|   const edges = diagObj.db.getEdges(); | ||||
|  | ||||
|   let i = 0; | ||||
|   for (i = subGraphs.length - 1; i >= 0; i--) { | ||||
|     subG = subGraphs[i]; | ||||
|  | ||||
|     selectAll('cluster').append('text'); | ||||
|  | ||||
|     for (const node of subG.nodes) { | ||||
|       log.warn( | ||||
|         'Setting subgraph', | ||||
|         node, | ||||
|         diagObj.db.lookUpDomId(node), | ||||
|         diagObj.db.lookUpDomId(subG.id) | ||||
|       ); | ||||
|       g.setParent(diagObj.db.lookUpDomId(node), diagObj.db.lookUpDomId(subG.id)); | ||||
|     } | ||||
|   } | ||||
|   await addVertices(vert, g, id, root, doc, diagObj); | ||||
|   await addEdges(edges, g, diagObj); | ||||
|  | ||||
|   // Create the renderer | ||||
|   const render = new Render(); | ||||
|  | ||||
|   // Add custom shapes | ||||
|   flowChartShapes.addToRender(render); | ||||
|  | ||||
|   // Add our custom arrow - an empty arrowhead | ||||
|   render.arrows().none = function normal(parent, id, edge, type) { | ||||
|     const marker = parent | ||||
|       .append('marker') | ||||
|       .attr('id', id) | ||||
|       .attr('viewBox', '0 0 10 10') | ||||
|       .attr('refX', 9) | ||||
|       .attr('refY', 5) | ||||
|       .attr('markerUnits', 'strokeWidth') | ||||
|       .attr('markerWidth', 8) | ||||
|       .attr('markerHeight', 6) | ||||
|       .attr('orient', 'auto'); | ||||
|  | ||||
|     const path = marker.append('path').attr('d', 'M 0 0 L 0 0 L 0 0 z'); | ||||
|     applyStyle(path, edge[type + 'Style']); | ||||
|   }; | ||||
|  | ||||
|   // Override normal arrowhead defined in d3. Remove style & add class to allow css styling. | ||||
|   render.arrows().normal = function normal(parent, id) { | ||||
|     const marker = parent | ||||
|       .append('marker') | ||||
|       .attr('id', id) | ||||
|       .attr('viewBox', '0 0 10 10') | ||||
|       .attr('refX', 9) | ||||
|       .attr('refY', 5) | ||||
|       .attr('markerUnits', 'strokeWidth') | ||||
|       .attr('markerWidth', 8) | ||||
|       .attr('markerHeight', 6) | ||||
|       .attr('orient', 'auto'); | ||||
|  | ||||
|     marker | ||||
|       .append('path') | ||||
|       .attr('d', 'M 0 0 L 10 5 L 0 10 z') | ||||
|       .attr('class', 'arrowheadPath') | ||||
|       .style('stroke-width', 1) | ||||
|       .style('stroke-dasharray', '1,0'); | ||||
|   }; | ||||
|  | ||||
|   // Set up an SVG group so that we can translate the final graph. | ||||
|   const svg = root.select(`[id="${id}"]`); | ||||
|  | ||||
|   // Run the renderer. This is what draws the final graph. | ||||
|   const element = root.select('#' + id + ' g'); | ||||
|   render(element, g); | ||||
|  | ||||
|   element.selectAll('g.node').attr('title', function () { | ||||
|     return diagObj.db.getTooltip(this.id); | ||||
|   }); | ||||
|  | ||||
|   // Index nodes | ||||
|   diagObj.db.indexNodes('subGraph' + i); | ||||
|  | ||||
|   // reposition labels | ||||
|   for (i = 0; i < subGraphs.length; i++) { | ||||
|     subG = subGraphs[i]; | ||||
|     if (subG.title !== 'undefined') { | ||||
|       const clusterRects = doc.querySelectorAll( | ||||
|         '#' + id + ' [id="' + diagObj.db.lookUpDomId(subG.id) + '"] rect' | ||||
|       ); | ||||
|       const clusterEl = doc.querySelectorAll( | ||||
|         '#' + id + ' [id="' + diagObj.db.lookUpDomId(subG.id) + '"]' | ||||
|       ); | ||||
|  | ||||
|       const xPos = clusterRects[0].x.baseVal.value; | ||||
|       const yPos = clusterRects[0].y.baseVal.value; | ||||
|       const _width = clusterRects[0].width.baseVal.value; | ||||
|       const cluster = select(clusterEl[0]); | ||||
|       const te = cluster.select('.label'); | ||||
|       te.attr('transform', `translate(${xPos + _width / 2}, ${yPos + 14})`); | ||||
|       te.attr('id', id + 'Text'); | ||||
|  | ||||
|       for (const className of subG.classes) { | ||||
|         clusterEl[0].classList.add(className); | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   // Add label rects for non html labels | ||||
|   if (!conf.htmlLabels) { | ||||
|     const labels = doc.querySelectorAll('[id="' + id + '"] .edgeLabel .label'); | ||||
|     for (const label of labels) { | ||||
|       // Get dimensions of label | ||||
|       const dim = label.getBBox(); | ||||
|  | ||||
|       const rect = doc.createElementNS('http://www.w3.org/2000/svg', 'rect'); | ||||
|       rect.setAttribute('rx', 0); | ||||
|       rect.setAttribute('ry', 0); | ||||
|       rect.setAttribute('width', dim.width); | ||||
|       rect.setAttribute('height', dim.height); | ||||
|       // rect.setAttribute('style', 'fill:#e8e8e8;'); | ||||
|  | ||||
|       label.insertBefore(rect, label.firstChild); | ||||
|     } | ||||
|   } | ||||
|   setupGraphViewbox(g, svg, conf.diagramPadding, conf.useMaxWidth); | ||||
|  | ||||
|   // If node has a link, wrap it in an anchor SVG object. | ||||
|   const keys = [...vert.keys()]; | ||||
|   keys.forEach(function (key) { | ||||
|     const vertex = vert.get(key); | ||||
|  | ||||
|     if (vertex.link) { | ||||
|       const node = root.select('#' + id + ' [id="' + diagObj.db.lookUpDomId(key) + '"]'); | ||||
|       if (node) { | ||||
|         const link = doc.createElementNS('http://www.w3.org/2000/svg', 'a'); | ||||
|         link.setAttributeNS('http://www.w3.org/2000/svg', 'class', vertex.classes.join(' ')); | ||||
|         link.setAttributeNS('http://www.w3.org/2000/svg', 'href', vertex.link); | ||||
|         link.setAttributeNS('http://www.w3.org/2000/svg', 'rel', 'noopener'); | ||||
|         if (securityLevel === 'sandbox') { | ||||
|           link.setAttributeNS('http://www.w3.org/2000/svg', 'target', '_top'); | ||||
|         } else if (vertex.linkTarget) { | ||||
|           link.setAttributeNS('http://www.w3.org/2000/svg', 'target', vertex.linkTarget); | ||||
|         } | ||||
|  | ||||
|         const linkNode = node.insert(function () { | ||||
|           return link; | ||||
|         }, ':first-child'); | ||||
|  | ||||
|         const shape = node.select('.label-container'); | ||||
|         if (shape) { | ||||
|           linkNode.append(function () { | ||||
|             return shape.node(); | ||||
|           }); | ||||
|         } | ||||
|  | ||||
|         const label = node.select('.label'); | ||||
|         if (label) { | ||||
|           linkNode.append(function () { | ||||
|             return label.node(); | ||||
|           }); | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|   }); | ||||
| }; | ||||
|  | ||||
| export default { | ||||
|   setConf, | ||||
|   addVertices, | ||||
|   addEdges, | ||||
|   getClasses, | ||||
|   draw, | ||||
| }; | ||||
							
								
								
									
										277
									
								
								packages/mermaid/src/diagrams/flowchart/flowRenderer.spec.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										277
									
								
								packages/mermaid/src/diagrams/flowchart/flowRenderer.spec.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,277 @@ | ||||
| /* eslint-disable @typescript-eslint/restrict-template-expressions */ | ||||
| import { addVertices, addEdges } from './flowRenderer.js'; | ||||
| import { setConfig } from '../../diagram-api/diagramAPI.js'; | ||||
|  | ||||
| setConfig({ | ||||
|   flowchart: { | ||||
|     htmlLabels: false, | ||||
|   }, | ||||
| }); | ||||
|  | ||||
| describe('the flowchart renderer', function () { | ||||
|   describe('when adding vertices to a graph', function () { | ||||
|     [ | ||||
|       ['round', 'rect', 5], | ||||
|       ['square', 'rect'], | ||||
|       ['diamond', 'question'], | ||||
|       ['hexagon', 'hexagon'], | ||||
|       ['odd', 'rect_left_inv_arrow'], | ||||
|       ['lean_right', 'lean_right'], | ||||
|       ['lean_left', 'lean_left'], | ||||
|       ['trapezoid', 'trapezoid'], | ||||
|       ['inv_trapezoid', 'inv_trapezoid'], | ||||
|       ['odd_right', 'rect_left_inv_arrow'], | ||||
|       ['circle', 'circle'], | ||||
|       ['ellipse', 'ellipse'], | ||||
|       ['stadium', 'stadium'], | ||||
|       ['subroutine', 'subroutine'], | ||||
|       ['cylinder', 'cylinder'], | ||||
|       ['group', 'rect'], | ||||
|     ].forEach(function ([type, expectedShape, expectedRadios = 0]) { | ||||
|       it(`should add the correct shaped node to the graph for vertex type ${type}`, async function () { | ||||
|         const fakeDiag = { | ||||
|           db: { | ||||
|             lookUpDomId: () => { | ||||
|               return 'my-node-id'; | ||||
|             }, | ||||
|           }, | ||||
|         }; | ||||
|         const addedNodes = []; | ||||
|         const mockG = { | ||||
|           setNode: function (id, object) { | ||||
|             addedNodes.push([id, object]); | ||||
|           }, | ||||
|         }; | ||||
|         await addVertices( | ||||
|           { | ||||
|             v1: { | ||||
|               type, | ||||
|               id: 'my-node-id', | ||||
|               classes: [], | ||||
|               styles: [], | ||||
|               text: 'my vertex text', | ||||
|             }, | ||||
|           }, | ||||
|           mockG, | ||||
|           'svg-id', | ||||
|           undefined, | ||||
|           undefined, | ||||
|           fakeDiag | ||||
|         ); | ||||
|         expect(addedNodes).toHaveLength(1); | ||||
|         expect(addedNodes[0][0]).toEqual('my-node-id'); | ||||
|         expect(addedNodes[0][1]).toHaveProperty('id', 'my-node-id'); | ||||
|         expect(addedNodes[0][1]).toHaveProperty('labelType', 'svg'); | ||||
|         expect(addedNodes[0][1]).toHaveProperty('shape', expectedShape); | ||||
|         expect(addedNodes[0][1]).toHaveProperty('rx', expectedRadios); | ||||
|         expect(addedNodes[0][1]).toHaveProperty('ry', expectedRadios); | ||||
|       }); | ||||
|     }); | ||||
|  | ||||
|     ['Multi<br>Line', 'Multi<br/>Line', 'Multi<br />Line', 'Multi<br\t/>Line'].forEach( | ||||
|       function (labelText) { | ||||
|         it('should handle multiline texts with different line breaks', async function () { | ||||
|           const addedNodes = []; | ||||
|           const fakeDiag = { | ||||
|             db: { | ||||
|               lookUpDomId: () => { | ||||
|                 return 'my-node-id'; | ||||
|               }, | ||||
|             }, | ||||
|           }; | ||||
|           const mockG = { | ||||
|             setNode: function (id, object) { | ||||
|               addedNodes.push([id, object]); | ||||
|             }, | ||||
|           }; | ||||
|           await addVertices( | ||||
|             { | ||||
|               v1: { | ||||
|                 type: 'rect', | ||||
|                 id: 'my-node-id', | ||||
|                 classes: [], | ||||
|                 styles: [], | ||||
|                 text: 'Multi<br>Line', | ||||
|               }, | ||||
|             }, | ||||
|             mockG, | ||||
|             'svg-id', | ||||
|             false, | ||||
|             document, | ||||
|             fakeDiag | ||||
|           ); | ||||
|           expect(addedNodes).toHaveLength(1); | ||||
|           expect(addedNodes[0][0]).toEqual('my-node-id'); | ||||
|           expect(addedNodes[0][1]).toHaveProperty('id', 'my-node-id'); | ||||
|           expect(addedNodes[0][1]).toHaveProperty('labelType', 'svg'); | ||||
|           expect(addedNodes[0][1].label).toBeDefined(); | ||||
|           expect(addedNodes[0][1].label).toBeDefined(); // <text> node | ||||
|           expect(addedNodes[0][1].label.firstChild.innerHTML).toEqual('Multi'); // <tspan> node, line 1 | ||||
|           expect(addedNodes[0][1].label.lastChild.innerHTML).toEqual('Line'); // <tspan> node, line 2 | ||||
|         }); | ||||
|       } | ||||
|     ); | ||||
|  | ||||
|     [ | ||||
|       [['fill:#fff'], 'fill:#fff;', ''], | ||||
|       [['color:#ccc'], '', 'color:#ccc;'], | ||||
|       [['fill:#fff', 'color:#ccc'], 'fill:#fff;', 'color:#ccc;'], | ||||
|       [ | ||||
|         ['fill:#fff', 'color:#ccc', 'text-align:center'], | ||||
|         'fill:#fff;', | ||||
|         'color:#ccc;text-align:center;', | ||||
|       ], | ||||
|     ].forEach(function ([style, expectedStyle, expectedLabelStyle]) { | ||||
|       it(`should add the styles to style and/or labelStyle for style ${style}`, async function () { | ||||
|         const addedNodes = []; | ||||
|         const fakeDiag = { | ||||
|           db: { | ||||
|             lookUpDomId: () => { | ||||
|               return 'my-node-id'; | ||||
|             }, | ||||
|           }, | ||||
|         }; | ||||
|         const mockG = { | ||||
|           setNode: function (id, object) { | ||||
|             addedNodes.push([id, object]); | ||||
|           }, | ||||
|         }; | ||||
|         await addVertices( | ||||
|           { | ||||
|             v1: { | ||||
|               type: 'rect', | ||||
|               id: 'my-node-id', | ||||
|               classes: [], | ||||
|               styles: style, | ||||
|               text: 'my vertex text', | ||||
|             }, | ||||
|           }, | ||||
|           mockG, | ||||
|           'svg-id', | ||||
|           undefined, | ||||
|           undefined, | ||||
|           fakeDiag | ||||
|         ); | ||||
|         expect(addedNodes).toHaveLength(1); | ||||
|         expect(addedNodes[0][0]).toEqual('my-node-id'); | ||||
|         expect(addedNodes[0][1]).toHaveProperty('id', 'my-node-id'); | ||||
|         expect(addedNodes[0][1]).toHaveProperty('labelType', 'svg'); | ||||
|         expect(addedNodes[0][1]).toHaveProperty('style', expectedStyle); | ||||
|         expect(addedNodes[0][1]).toHaveProperty('labelStyle', expectedLabelStyle); | ||||
|       }); | ||||
|     }); | ||||
|  | ||||
|     it(`should add default class to all nodes which do not have another class assigned`, async function () { | ||||
|       const addedNodes = []; | ||||
|       const mockG = { | ||||
|         setNode: function (id, object) { | ||||
|           addedNodes.push([id, object]); | ||||
|         }, | ||||
|       }; | ||||
|       const fakeDiag = { | ||||
|         db: { | ||||
|           lookUpDomId: () => { | ||||
|             return 'my-node-id'; | ||||
|           }, | ||||
|         }, | ||||
|       }; | ||||
|       await addVertices( | ||||
|         { | ||||
|           v1: { | ||||
|             type: 'rect', | ||||
|             id: 'my-node-id', | ||||
|             classes: [], | ||||
|             styles: [], | ||||
|             text: 'my vertex text', | ||||
|           }, | ||||
|           v2: { | ||||
|             type: 'rect', | ||||
|             id: 'myNode', | ||||
|             classes: ['myClass'], | ||||
|             styles: [], | ||||
|             text: 'my vertex text', | ||||
|           }, | ||||
|         }, | ||||
|         mockG, | ||||
|         'svg-id', | ||||
|         undefined, | ||||
|         undefined, | ||||
|         fakeDiag | ||||
|       ); | ||||
|       expect(addedNodes).toHaveLength(2); | ||||
|       expect(addedNodes[0][0]).toEqual('my-node-id'); | ||||
|       expect(addedNodes[0][1]).toHaveProperty('class', 'default'); | ||||
|       expect(addedNodes[1][0]).toEqual('my-node-id'); | ||||
|       expect(addedNodes[1][1]).toHaveProperty('class', 'myClass'); | ||||
|     }); | ||||
|   }); | ||||
|  | ||||
|   describe('when adding edges to a graph', function () { | ||||
|     it('should handle multiline texts and set centered label position', async function () { | ||||
|       const addedEdges = []; | ||||
|       const fakeDiag = { | ||||
|         db: { | ||||
|           lookUpDomId: () => { | ||||
|             return 'my-node-id'; | ||||
|           }, | ||||
|         }, | ||||
|       }; | ||||
|       const mockG = { | ||||
|         setEdge: function (s, e, data, c) { | ||||
|           addedEdges.push(data); | ||||
|         }, | ||||
|       }; | ||||
|       await addEdges( | ||||
|         [ | ||||
|           { text: 'Multi<br>Line' }, | ||||
|           { text: 'Multi<br/>Line' }, | ||||
|           { text: 'Multi<br />Line' }, | ||||
|           { text: 'Multi<br\t/>Line' }, | ||||
|           { style: ['stroke:DarkGray', 'stroke-width:2px'], text: 'Multi<br>Line' }, | ||||
|           { style: ['stroke:DarkGray', 'stroke-width:2px'], text: 'Multi<br/>Line' }, | ||||
|           { style: ['stroke:DarkGray', 'stroke-width:2px'], text: 'Multi<br />Line' }, | ||||
|           { style: ['stroke:DarkGray', 'stroke-width:2px'], text: 'Multi<br\t/>Line' }, | ||||
|         ], | ||||
|         mockG, | ||||
|         fakeDiag | ||||
|       ); | ||||
|  | ||||
|       addedEdges.forEach(function (edge) { | ||||
|         expect(edge).toHaveProperty('label', 'Multi\nLine'); | ||||
|         expect(edge).toHaveProperty('labelpos', 'c'); | ||||
|       }); | ||||
|     }); | ||||
|  | ||||
|     [ | ||||
|       [['stroke:DarkGray'], 'stroke:DarkGray;', ''], | ||||
|       [['color:red'], '', 'fill:red;'], | ||||
|       [['stroke:DarkGray', 'color:red'], 'stroke:DarkGray;', 'fill:red;'], | ||||
|       [ | ||||
|         ['stroke:DarkGray', 'color:red', 'stroke-width:2px'], | ||||
|         'stroke:DarkGray;stroke-width:2px;', | ||||
|         'fill:red;', | ||||
|       ], | ||||
|     ].forEach(function ([style, expectedStyle, expectedLabelStyle]) { | ||||
|       it(`should add the styles to style and/or labelStyle for style ${style}`, async function () { | ||||
|         const addedEdges = []; | ||||
|         const fakeDiag = { | ||||
|           db: { | ||||
|             lookUpDomId: () => { | ||||
|               return 'my-node-id'; | ||||
|             }, | ||||
|           }, | ||||
|         }; | ||||
|         const mockG = { | ||||
|           setEdge: function (s, e, data, c) { | ||||
|             addedEdges.push(data); | ||||
|           }, | ||||
|         }; | ||||
|         await addEdges([{ style: style, text: 'styling' }], mockG, fakeDiag); | ||||
|  | ||||
|         expect(addedEdges).toHaveLength(1); | ||||
|         expect(addedEdges[0]).toHaveProperty('style', expectedStyle); | ||||
|         expect(addedEdges[0]).toHaveProperty('labelStyle', expectedLabelStyle); | ||||
|       }); | ||||
|     }); | ||||
|   }); | ||||
| }); | ||||
| @@ -38,14 +38,11 @@ const getStyles = (options: FlowChartStyleOptions) => | ||||
|   .cluster-label text { | ||||
|     fill: ${options.titleColor}; | ||||
|   } | ||||
|   .cluster-label span { | ||||
|   .cluster-label span,p { | ||||
|     color: ${options.titleColor}; | ||||
|   } | ||||
|   .cluster-label span p { | ||||
|     background-color: transparent; | ||||
|   } | ||||
|  | ||||
|   .label text,span { | ||||
|   .label text,span,p { | ||||
|     fill: ${options.nodeTextColor || options.textColor}; | ||||
|     color: ${options.nodeTextColor || options.textColor}; | ||||
|   } | ||||
| @@ -59,7 +56,7 @@ const getStyles = (options: FlowChartStyleOptions) => | ||||
|     stroke: ${options.nodeBorder}; | ||||
|     stroke-width: 1px; | ||||
|   } | ||||
|   .rough-node .label text , .node .label text { | ||||
|   .flowchart-label text { | ||||
|     text-anchor: middle; | ||||
|   } | ||||
|   // .flowchart-label .text-outer-tspan { | ||||
| @@ -98,9 +95,6 @@ const getStyles = (options: FlowChartStyleOptions) => | ||||
|  | ||||
|   .edgeLabel { | ||||
|     background-color: ${options.edgeLabelBackground}; | ||||
|     p { | ||||
|       background-color: ${options.edgeLabelBackground}; | ||||
|     } | ||||
|     rect { | ||||
|       opacity: 0.5; | ||||
|       background-color: ${options.edgeLabelBackground}; | ||||
| @@ -112,7 +106,7 @@ const getStyles = (options: FlowChartStyleOptions) => | ||||
|   /* For html labels only */ | ||||
|   .labelBkg { | ||||
|     background-color: ${fade(options.edgeLabelBackground, 0.5)}; | ||||
|     // background-color: | ||||
|     // background-color:  | ||||
|   } | ||||
|  | ||||
|   .cluster rect { | ||||
| @@ -125,7 +119,7 @@ const getStyles = (options: FlowChartStyleOptions) => | ||||
|     fill: ${options.titleColor}; | ||||
|   } | ||||
|  | ||||
|   .cluster span { | ||||
|   .cluster span,p { | ||||
|     color: ${options.titleColor}; | ||||
|   } | ||||
|   /* .cluster div { | ||||
|   | ||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user