diff --git a/.build/jsonSchema.ts b/.build/jsonSchema.ts index 51b1184fc3..d29ef7920a 100644 --- a/.build/jsonSchema.ts +++ b/.build/jsonSchema.ts @@ -17,6 +17,7 @@ const MERMAID_CONFIG_DIAGRAM_KEYS = [ 'er', 'pie', 'quadrantChart', + 'xyChart', 'requirement', 'mindmap', 'timeline', @@ -34,7 +35,7 @@ const MERMAID_CONFIG_DIAGRAM_KEYS = [ * @param mermaidConfigSchema - The Mermaid JSON Schema to use. * @returns The default mermaid config object. */ -export function generateDefaults(mermaidConfigSchema: JSONSchemaType) { +function generateDefaults(mermaidConfigSchema: JSONSchemaType) { const ajv = new Ajv2019({ useDefaults: true, allowUnionTypes: true, diff --git a/.github/workflows/build-docs.yml b/.github/workflows/build-docs.yml index 152b177ae9..acfb1887e9 100644 --- a/.github/workflows/build-docs.yml +++ b/.github/workflows/build-docs.yml @@ -16,12 +16,12 @@ jobs: runs-on: ubuntu-latest steps: - name: Checkout - uses: actions/checkout@v3 + uses: actions/checkout@v4 - uses: pnpm/action-setup@v2 - name: Setup Node.js - uses: actions/setup-node@v3 + uses: actions/setup-node@v4 with: cache: pnpm node-version: 18 @@ -29,7 +29,7 @@ jobs: - name: Install Packages run: pnpm install --frozen-lockfile - - name: Verify release verion + - name: Verify release version if: ${{ github.event_name == 'push' && (github.ref == 'refs/heads/master' || startsWith(github.ref, 'refs/heads/release')) }} run: pnpm --filter mermaid run docs:verify-version diff --git a/.github/workflows/build.yml b/.github/workflows/build.yml index eeb557ebb9..605dea9ab3 100644 --- a/.github/workflows/build.yml +++ b/.github/workflows/build.yml @@ -19,13 +19,13 @@ jobs: matrix: node-version: [18.x] steps: - - uses: actions/checkout@v3 + - uses: actions/checkout@v4 - uses: pnpm/action-setup@v2 # uses version from "packageManager" field in package.json - name: Setup Node.js ${{ matrix.node-version }} - uses: actions/setup-node@v3 + uses: actions/setup-node@v4 with: cache: pnpm node-version: ${{ matrix.node-version }} diff --git a/.github/workflows/check-readme-in-sync.yml b/.github/workflows/check-readme-in-sync.yml index 5a8ca319b2..ad6df66b50 100644 --- a/.github/workflows/check-readme-in-sync.yml +++ b/.github/workflows/check-readme-in-sync.yml @@ -18,7 +18,7 @@ jobs: runs-on: ubuntu-latest steps: - name: Checkout repository - uses: actions/checkout@v3 + uses: actions/checkout@v4 - name: Check for difference in README.md and docs/README.md run: | diff --git a/.github/workflows/checks.yml b/.github/workflows/checks.yml index 9f9f316c40..012fbf19d5 100644 --- a/.github/workflows/checks.yml +++ b/.github/workflows/checks.yml @@ -15,7 +15,7 @@ jobs: name: check tests if: github.repository_owner == 'mermaid-js' steps: - - uses: actions/checkout@v3 + - uses: actions/checkout@v4 with: fetch-depth: 0 - uses: testomatio/check-tests@stable diff --git a/.github/workflows/codeql.yml b/.github/workflows/codeql.yml index 26cb2db268..f8c50f47fa 100644 --- a/.github/workflows/codeql.yml +++ b/.github/workflows/codeql.yml @@ -29,7 +29,7 @@ jobs: steps: - name: Checkout repository - uses: actions/checkout@v3 + uses: actions/checkout@v4 # Initializes the CodeQL tools for scanning. - name: Initialize CodeQL diff --git a/.github/workflows/dependency-review.yml b/.github/workflows/dependency-review.yml index 34b14c395b..4e75197790 100644 --- a/.github/workflows/dependency-review.yml +++ b/.github/workflows/dependency-review.yml @@ -1,6 +1,6 @@ # Dependency Review Action # -# This Action will scan dependency manifest files that change as part of a Pull Reqest, surfacing known-vulnerable versions of the packages declared or updated in the PR. Once installed, if the workflow run is marked as required, PRs introducing known-vulnerable packages will be blocked from merging. +# This Action will scan dependency manifest files that change as part of a Pull Request, surfacing known-vulnerable versions of the packages declared or updated in the PR. Once installed, if the workflow run is marked as required, PRs introducing known-vulnerable packages will be blocked from merging. # # Source repository: https://github.com/actions/dependency-review-action # Public documentation: https://docs.github.com/en/code-security/supply-chain-security/understanding-your-software-supply-chain/about-dependency-review#dependency-review-enforcement @@ -15,6 +15,6 @@ jobs: runs-on: ubuntu-latest steps: - name: 'Checkout Repository' - uses: actions/checkout@v3 + uses: actions/checkout@v4 - name: 'Dependency Review' uses: actions/dependency-review-action@v3 diff --git a/.github/workflows/e2e-applitools.yml b/.github/workflows/e2e-applitools.yml index 5b19431421..543fb5dbb4 100644 --- a/.github/workflows/e2e-applitools.yml +++ b/.github/workflows/e2e-applitools.yml @@ -30,13 +30,13 @@ jobs: run: | echo "::error,title=Not using Applitols::APPLITOOLS_API_KEY is empty, disabling Applitools for this run." - - uses: actions/checkout@v3 + - uses: actions/checkout@v4 - uses: pnpm/action-setup@v2 # uses version from "packageManager" field in package.json - name: Setup Node.js ${{ matrix.node-version }} - uses: actions/setup-node@v3 + uses: actions/setup-node@v4 with: node-version: ${{ matrix.node-version }} diff --git a/.github/workflows/e2e.yml b/.github/workflows/e2e.yml index 3e6966677b..71806a9c46 100644 --- a/.github/workflows/e2e.yml +++ b/.github/workflows/e2e.yml @@ -17,13 +17,13 @@ jobs: node-version: [18.x] containers: [1, 2, 3, 4] steps: - - uses: actions/checkout@v3 + - uses: actions/checkout@v4 - uses: pnpm/action-setup@v2 # uses version from "packageManager" field in package.json - name: Setup Node.js ${{ matrix.node-version }} - uses: actions/setup-node@v3 + uses: actions/setup-node@v4 with: node-version: ${{ matrix.node-version }} diff --git a/.github/workflows/link-checker.yml b/.github/workflows/link-checker.yml index 70580bfff1..c3e2ee44fe 100644 --- a/.github/workflows/link-checker.yml +++ b/.github/workflows/link-checker.yml @@ -26,7 +26,7 @@ jobs: # lychee only uses the GITHUB_TOKEN to avoid rate-limiting contents: read steps: - - uses: actions/checkout@v3 + - uses: actions/checkout@v4 - name: Restore lychee cache uses: actions/cache@v3 diff --git a/.github/workflows/lint.yml b/.github/workflows/lint.yml index f59c8af31d..f0c5560a1e 100644 --- a/.github/workflows/lint.yml +++ b/.github/workflows/lint.yml @@ -20,13 +20,13 @@ jobs: matrix: node-version: [18.x] steps: - - uses: actions/checkout@v3 + - uses: actions/checkout@v4 - uses: pnpm/action-setup@v2 # uses version from "packageManager" field in package.json - name: Setup Node.js ${{ matrix.node-version }} - uses: actions/setup-node@v3 + uses: actions/setup-node@v4 with: cache: pnpm node-version: ${{ matrix.node-version }} diff --git a/.github/workflows/pr-labeler-config-validator.yml b/.github/workflows/pr-labeler-config-validator.yml index ff5d8d0a1f..8bdfed21bc 100644 --- a/.github/workflows/pr-labeler-config-validator.yml +++ b/.github/workflows/pr-labeler-config-validator.yml @@ -16,7 +16,7 @@ jobs: runs-on: ubuntu-latest steps: - name: Checkout Repository - uses: actions/checkout@v3 + uses: actions/checkout@v4 - name: Validate Configuration uses: Yash-Singh1/pr-labeler-config-validator@releases/v0.0.3 with: diff --git a/.github/workflows/publish-docs.yml b/.github/workflows/publish-docs.yml index f63e587502..05cd68aff1 100644 --- a/.github/workflows/publish-docs.yml +++ b/.github/workflows/publish-docs.yml @@ -23,12 +23,12 @@ jobs: runs-on: ubuntu-latest steps: - name: Checkout - uses: actions/checkout@v3 + uses: actions/checkout@v4 - uses: pnpm/action-setup@v2 - name: Setup Node.js - uses: actions/setup-node@v3 + uses: actions/setup-node@v4 with: cache: pnpm node-version: 18 diff --git a/.github/workflows/release-preview-publish.yml b/.github/workflows/release-preview-publish.yml index 221e3836ee..c6503847d9 100644 --- a/.github/workflows/release-preview-publish.yml +++ b/.github/workflows/release-preview-publish.yml @@ -9,14 +9,14 @@ jobs: publish-preview: runs-on: ubuntu-latest steps: - - uses: actions/checkout@v3 + - uses: actions/checkout@v4 with: fetch-depth: 0 - uses: pnpm/action-setup@v2 - name: Setup Node.js - uses: actions/setup-node@v3 + uses: actions/setup-node@v4 with: cache: pnpm node-version: 18.x diff --git a/.github/workflows/release-publish.yml b/.github/workflows/release-publish.yml index eb28fe9c8d..69ef749402 100644 --- a/.github/workflows/release-publish.yml +++ b/.github/workflows/release-publish.yml @@ -8,14 +8,14 @@ jobs: publish: runs-on: ubuntu-latest steps: - - uses: actions/checkout@v3 + - uses: actions/checkout@v4 - uses: fregante/setup-git-user@v2 - uses: pnpm/action-setup@v2 # uses version from "packageManager" field in package.json - name: Setup Node.js v18 - uses: actions/setup-node@v3 + uses: actions/setup-node@v4 with: cache: pnpm node-version: 18.x diff --git a/.github/workflows/test.yml b/.github/workflows/test.yml index 7c32795e8d..a18b31c9cd 100644 --- a/.github/workflows/test.yml +++ b/.github/workflows/test.yml @@ -12,13 +12,13 @@ jobs: matrix: node-version: [18.x] steps: - - uses: actions/checkout@v3 + - uses: actions/checkout@v4 - uses: pnpm/action-setup@v2 # uses version from "packageManager" field in package.json - name: Setup Node.js ${{ matrix.node-version }} - uses: actions/setup-node@v3 + uses: actions/setup-node@v4 with: cache: pnpm node-version: ${{ matrix.node-version }} diff --git a/.github/workflows/update-browserlist.yml b/.github/workflows/update-browserlist.yml index 813a400b36..0a83df795d 100644 --- a/.github/workflows/update-browserlist.yml +++ b/.github/workflows/update-browserlist.yml @@ -8,7 +8,7 @@ jobs: update-browser-list: runs-on: ubuntu-latest steps: - - uses: actions/checkout@v3 + - uses: actions/checkout@v4 - run: npx browserslist@latest --update-db - name: Commit changes uses: EndBug/add-and-commit@v9 diff --git a/CHANGELOG.md b/CHANGELOG.md index 7552efa3b7..ede5e19ad6 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -68,7 +68,7 @@ try { ### Init deprecated and InitThrowsErrors removed -The config passed to `init` was not being used eariler. +The config passed to `init` was not being used earlier. It will now be used. The `init` function is deprecated and will be removed in the next major release. init currently works as a wrapper to `initialize` and `run`. @@ -195,7 +195,7 @@ mermaid.run({ - "Cannot activate" in sequenceDiagram [\#647](https://github.com/knsv/mermaid/issues/647) - Link \("click" statement\) in flowchart does not work in exported SVG [\#646](https://github.com/knsv/mermaid/issues/646) - How to pass styling [\#639](https://github.com/knsv/mermaid/issues/639) -- The live editor cant show seq diagram with notes for 8.0.0-alpha.3 [\#638](https://github.com/knsv/mermaid/issues/638) +- The live editor can't show seq diagram with notes for 8.0.0-alpha.3 [\#638](https://github.com/knsv/mermaid/issues/638) - import mermaid.css with ES6 + NPM [\#634](https://github.com/knsv/mermaid/issues/634) - Actor line cuts through other elements [\#633](https://github.com/knsv/mermaid/issues/633) - Graph TD line out of the picture \(left side\) [\#630](https://github.com/knsv/mermaid/issues/630) @@ -504,7 +504,7 @@ mermaid.run({ - Docs css: code hard to read [\#324](https://github.com/knsv/mermaid/issues/324) - About Markpad integration [\#323](https://github.com/knsv/mermaid/issues/323) -- How to link backwords in flowchat? [\#321](https://github.com/knsv/mermaid/issues/321) +- How to link backwards in flowchat? [\#321](https://github.com/knsv/mermaid/issues/321) - Help with editor [\#310](https://github.com/knsv/mermaid/issues/310) - +1 [\#293](https://github.com/knsv/mermaid/issues/293) - Basic chart does not render on Chome, but does in Firefox [\#290](https://github.com/knsv/mermaid/issues/290) @@ -619,7 +619,7 @@ mermaid.run({ - render to png from the cli does not display the marker-end arrow heads [\#181](https://github.com/knsv/mermaid/issues/181) - Links in sequence diagrams [\#159](https://github.com/knsv/mermaid/issues/159) - comment characters `%%` cause parse error [\#141](https://github.com/knsv/mermaid/issues/141) -- Add a reversed assymetric shape [\#124](https://github.com/knsv/mermaid/issues/124) +- Add a reversed asymmetric shape [\#124](https://github.com/knsv/mermaid/issues/124) - Add syntax for double headed arrows [\#123](https://github.com/knsv/mermaid/issues/123) - Support for font-awesome [\#49](https://github.com/knsv/mermaid/issues/49) @@ -659,7 +659,7 @@ mermaid.run({ - Auto linewrap for notes in sequence diagrams [\#178](https://github.com/knsv/mermaid/issues/178) - Execute code after initialize [\#176](https://github.com/knsv/mermaid/issues/176) - Autoscaling for all diagram types [\#175](https://github.com/knsv/mermaid/issues/175) -- Problem wit click event callback [\#174](https://github.com/knsv/mermaid/issues/174) +- Problem with click event callback [\#174](https://github.com/knsv/mermaid/issues/174) - How to escape characters? [\#170](https://github.com/knsv/mermaid/issues/170) - it can not work [\#167](https://github.com/knsv/mermaid/issues/167) - UML Class diagram [\#154](https://github.com/knsv/mermaid/issues/154) @@ -762,7 +762,7 @@ mermaid.run({ - subgraph background is black in rendered flowchart PNG via CLI [\#121](https://github.com/knsv/mermaid/issues/121) - Integrate editor at https://github.com/naseer/mermaid-webapp [\#110](https://github.com/knsv/mermaid/issues/110) - Internet Explorer Support [\#99](https://github.com/knsv/mermaid/issues/99) -- Assymetric shapes not documented [\#82](https://github.com/knsv/mermaid/issues/82) +- Asymmetric shapes not documented [\#82](https://github.com/knsv/mermaid/issues/82) - NoModificationAllowedError [\#23](https://github.com/knsv/mermaid/issues/23) - Improve arrows [\#3](https://github.com/knsv/mermaid/issues/3) @@ -908,7 +908,7 @@ mermaid.run({ - Question marks don't render properly with /dist/mermaid.full.min.js [\#30](https://github.com/knsv/mermaid/issues/30) - Error with some characters [\#25](https://github.com/knsv/mermaid/issues/25) -- Provide parse function in browser widthout `require`? [\#21](https://github.com/knsv/mermaid/issues/21) +- Provide parse function in browser without `require`? [\#21](https://github.com/knsv/mermaid/issues/21) - Better label text support [\#18](https://github.com/knsv/mermaid/issues/18) - Cap-cased words break parser [\#8](https://github.com/knsv/mermaid/issues/8) diff --git a/README.md b/README.md index e5b8738afe..cf21fdb8e6 100644 --- a/README.md +++ b/README.md @@ -34,7 +34,7 @@ Try Live Editor previews of future releases: @@ -44,6 +44,22 @@ Try Live Editor previews of future releases: Explore Mermaid.js in depth, with real-world examples, tips & tricks from the creator... The first official book on Mermaid is available for purchase. Check it out! +## Table of content + +
+Expand contents + +- [About](#about) +- [Examples](#examples) +- [Release](#release) +- [Related projects](#related-projects) +- [Contributors](#contributors) +- [Security and safe diagrams](#security-and-safe-diagrams) +- [Reporting vulnerabilities](#reporting-vulnerabilities) +- [Appreciation](#appreciation) + +
+ ## About @@ -58,7 +74,7 @@ Mermaid addresses this problem by enabling users to create easily modifiable dia
Mermaid allows even non-programmers to easily create detailed diagrams through the [Mermaid Live Editor](https://mermaid.live/).
-[Tutorials](./docs/config/Tutorials.md) has video tutorials. +For video tutorials, visit our [Tutorials](./docs/config/Tutorials.md) page. Use Mermaid with your favorite applications, check out the list of [Integrations and Usages of Mermaid](./docs/ecosystem/integrations-community.md). You can also use Mermaid within [GitHub](https://github.blog/2022-02-14-include-diagrams-markdown-files-mermaid/) as well many of your other favorite applications—check out the list of [Integrations and Usages of Mermaid](./docs/ecosystem/integrations-community.md). diff --git a/README.zh-CN.md b/README.zh-CN.md index da83494052..98975ea331 100644 --- a/README.zh-CN.md +++ b/README.zh-CN.md @@ -35,7 +35,7 @@ Mermaid [![CDN Status](https://img.shields.io/jsdelivr/npm/hm/mermaid)](https://www.jsdelivr.com/package/npm/mermaid) [![NPM Downloads](https://img.shields.io/npm/dm/mermaid)](https://www.npmjs.com/package/mermaid) [![Join our Slack!](https://img.shields.io/static/v1?message=join%20chat&color=9cf&logo=slack&label=slack)](https://join.slack.com/t/mermaid-talk/shared_invite/enQtNzc4NDIyNzk4OTAyLWVhYjQxOTI2OTg4YmE1ZmJkY2Y4MTU3ODliYmIwOTY3NDJlYjA0YjIyZTdkMDMyZTUwOGI0NjEzYmEwODcwOTE) -[![Twitter Follow](https://img.shields.io/badge/Social-mermaidjs__-blue?style=social&logo=twitter)](https://twitter.com/mermaidjs_) +[![Twitter Follow](https://img.shields.io/badge/Social-mermaidjs__-blue?style=social&logo=X)](https://twitter.com/mermaidjs_) diff --git a/cSpell.json b/cSpell.json index d32862469c..f60e7b3785 100644 --- a/cSpell.json +++ b/cSpell.json @@ -160,6 +160,7 @@ "vitepress", "vueuse", "xlink", + "xychart", "yash", "yokozuna", "zenuml", diff --git a/cypress/integration/rendering/classDiagram.spec.js b/cypress/integration/rendering/classDiagram.spec.js index a23430b083..cab3649df4 100644 --- a/cypress/integration/rendering/classDiagram.spec.js +++ b/cypress/integration/rendering/classDiagram.spec.js @@ -501,4 +501,16 @@ describe('Class diagram', () => { B : -methods() `); }); + + it('should handle notes with anchor tag having target attribute', () => { + renderGraph( + `classDiagram + class test { } + note for test "note about mermaid"` + ); + + cy.get('svg').then((svg) => { + cy.get('a').should('have.attr', 'target', '_blank').should('have.attr', 'rel', 'noopener'); + }); + }); }); diff --git a/cypress/integration/rendering/gitGraph.spec.js b/cypress/integration/rendering/gitGraph.spec.js index c01a557961..9f040a36f0 100644 --- a/cypress/integration/rendering/gitGraph.spec.js +++ b/cypress/integration/rendering/gitGraph.spec.js @@ -26,7 +26,7 @@ describe('Git Graph diagram', () => { `gitGraph commit id: "Normal Commit" commit id: "Reverse Commit" type: REVERSE - commit id: "Hightlight Commit" type: HIGHLIGHT + commit id: "Highlight Commit" type: HIGHLIGHT `, {} ); @@ -36,7 +36,7 @@ describe('Git Graph diagram', () => { `gitGraph commit id: "Normal Commit with tag" tag: "v1.0.0" commit id: "Reverse Commit with tag" type: REVERSE tag: "RC_1" - commit id: "Hightlight Commit" type: HIGHLIGHT tag: "8.8.4" + commit id: "Highlight Commit" type: HIGHLIGHT tag: "8.8.4" `, {} ); @@ -102,7 +102,7 @@ describe('Git Graph diagram', () => { {} ); }); - it('8: should render a simple gitgraph with more than 8 branchs & overriding variables', () => { + it('8: should render a simple gitgraph with more than 8 branches & overriding variables', () => { imgSnapshotTest( `%%{init: { 'logLevel': 'debug', 'theme': 'default' , 'themeVariables': { 'gitBranchLabel0': '#ffffff', @@ -358,7 +358,7 @@ gitGraph `gitGraph TB: commit id: "Normal Commit" commit id: "Reverse Commit" type: REVERSE - commit id: "Hightlight Commit" type: HIGHLIGHT + commit id: "Highlight Commit" type: HIGHLIGHT `, {} ); @@ -368,7 +368,7 @@ gitGraph `gitGraph TB: commit id: "Normal Commit with tag" tag: "v1.0.0" commit id: "Reverse Commit with tag" type: REVERSE tag: "RC_1" - commit id: "Hightlight Commit" type: HIGHLIGHT tag: "8.8.4" + commit id: "Highlight Commit" type: HIGHLIGHT tag: "8.8.4" `, {} ); @@ -434,7 +434,7 @@ gitGraph {} ); }); - it('22: should render a simple gitgraph with more than 8 branchs & overriding variables | Vertical Branch', () => { + it('22: should render a simple gitgraph with more than 8 branches & overriding variables | Vertical Branch', () => { imgSnapshotTest( `%%{init: { 'logLevel': 'debug', 'theme': 'default' , 'themeVariables': { 'gitBranchLabel0': '#ffffff', diff --git a/cypress/integration/rendering/timeline.spec.ts b/cypress/integration/rendering/timeline.spec.ts index 68da01d503..c748b54d3c 100644 --- a/cypress/integration/rendering/timeline.spec.ts +++ b/cypress/integration/rendering/timeline.spec.ts @@ -57,7 +57,7 @@ describe('Timeline diagram', () => { {} ); }); - it('5: should render a simple timeline with directive overriden colors', () => { + it('5: should render a simple timeline with directive overridden colors', () => { imgSnapshotTest( ` %%{init: { 'logLevel': 'debug', 'theme': 'default' , 'themeVariables': { 'cScale0': '#ff0000', diff --git a/cypress/integration/rendering/xyChart.spec.js b/cypress/integration/rendering/xyChart.spec.js new file mode 100644 index 0000000000..85d998c50b --- /dev/null +++ b/cypress/integration/rendering/xyChart.spec.js @@ -0,0 +1,322 @@ +import { imgSnapshotTest, renderGraph } from '../../helpers/util.ts'; + +describe('XY Chart', () => { + it('should render the simplest possible chart', () => { + imgSnapshotTest( + ` + xychart-beta + line [10, 30, 20] + `, + {} + ); + cy.get('svg'); + }); + it('Should render a complete chart', () => { + imgSnapshotTest( + ` + 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] + line [5000, 6000, 7500, 8200, 9500, 10500, 11000, 10200, 9200, 8500, 7000, 6000] + `, + {} + ); + }); + it('Should render a chart without title', () => { + imgSnapshotTest( + ` + xychart-beta + 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] + line [5000, 6000, 7500, 8200, 9500, 10500, 11000, 10200, 9200, 8500, 7000, 6000] + `, + {} + ); + cy.get('svg'); + }); + it('y-axis title not required', () => { + imgSnapshotTest( + ` + xychart-beta + x-axis Months [jan, feb, mar, apr, may, jun, jul, aug, sep, oct, nov, dec] + y-axis 4000 --> 11000 + bar [5000, 6000, 7500, 8200, 9500, 10500, 11000, 10200, 9200, 8500, 7000, 6000] + line [5000, 6000, 7500, 8200, 9500, 10500, 11000, 10200, 9200, 8500, 7000, 6000] + `, + {} + ); + cy.get('svg'); + }); + it('Should render a chart without y-axis with different range', () => { + imgSnapshotTest( + ` + xychart-beta + x-axis Months [jan, feb, mar, apr, may, jun, jul, aug, sep, oct, nov, dec] + bar [5000, 6000, 7500, 8200, 9500, 10500, 14000, 3200, 9200, 9900, 3400, 6000] + line [2000, 7000, 6500, 9200, 9500, 7500, 11000, 10200, 3200, 8500, 7000, 8800] + `, + {} + ); + cy.get('svg'); + }); + it('x axis title not required', () => { + imgSnapshotTest( + ` + xychart-beta + x-axis [jan, feb, mar, apr, may, jun, jul, aug, sep, oct, nov, dec] + bar [5000, 6000, 7500, 8200, 9500, 10500, 14000, 3200, 9200, 9900, 3400, 6000] + line [2000, 7000, 6500, 9200, 9500, 7500, 11000, 10200, 3200, 8500, 7000, 8800] + `, + {} + ); + cy.get('svg'); + }); + it('Multiple plots can be rendered', () => { + imgSnapshotTest( + ` + xychart-beta + line [23, 46, 77, 34] + line [45, 32, 33, 12] + bar [87, 54, 99, 85] + line [78, 88, 22, 4] + line [22, 29, 75, 33] + bar [52, 96, 35, 10] + `, + {} + ); + cy.get('svg'); + }); + it('Decimals and negative numbers are supported', () => { + imgSnapshotTest( + ` + xychart-beta + y-axis -2.4 --> 3.5 + line [+1.3, .6, 2.4, -.34] + `, + {} + ); + cy.get('svg'); + }); + it('Render spark line with "plotReservedSpacePercent"', () => { + imgSnapshotTest( + ` + --- + config: + theme: dark + xyChart: + width: 200 + height: 20 + plotReservedSpacePercent: 100 + --- + xychart-beta + line [5000, 9000, 7500, 6200, 9500, 5500, 11000, 8200, 9200, 9500, 7000, 8800] + `, + {} + ); + cy.get('svg'); + }); + it('Render spark bar without displaying other property', () => { + imgSnapshotTest( + ` + --- + config: + theme: dark + xyChart: + width: 200 + height: 20 + xAxis: + showLabel: false + showTitle: false + showTick: false + showAxisLine: false + yAxis: + showLabel: false + showTitle: false + showTick: false + showAxisLine: false + --- + xychart-beta + bar [5000, 9000, 7500, 6200, 9500, 5500, 11000, 8200, 9200, 9500, 7000, 8800] + `, + {} + ); + cy.get('svg'); + }); + it('Should use all the config from directive', () => { + imgSnapshotTest( + ` + %%{init: {"xyChart": {"width": 1000, "height": 600, "titlePadding": 5, "titleFontSize": 10, "xAxis": {"labelFontSize": "20", "labelPadding": 10, "titleFontSize": 30, "titlePadding": 20, "tickLength": 10, "tickWidth": 5}, "yAxis": {"labelFontSize": "20", "labelPadding": 10, "titleFontSize": 30, "titlePadding": 20, "tickLength": 10, "tickWidth": 5}, "plotBorderWidth": 5, "chartOrientation": "horizontal", "plotReservedSpacePercent": 60 }}}%% + 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] + line [5000, 6000, 7500, 8200, 9500, 10500, 11000, 10200, 9200, 8500, 7000, 6000] + `, + {} + ); + cy.get('svg'); + }); + it('Should use all the config from yaml', () => { + imgSnapshotTest( + ` + --- + config: + theme: forest + xyChart: + width: 1000 + height: 600 + titlePadding: 5 + titleFontSize: 10 + xAxis: + labelFontSize: 20 + labelPadding: 10 + titleFontSize: 30 + titlePadding: 20 + tickLength: 10 + tickWidth: 5 + axisLineWidth: 5 + yAxis: + labelFontSize: 20 + labelPadding: 10 + titleFontSize: 30 + titlePadding: 20 + tickLength: 10 + tickWidth: 5 + axisLineWidth: 5 + chartOrientation: horizontal + plotReservedSpacePercent: 60 + --- + 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] + line [5000, 6000, 7500, 8200, 9500, 10500, 11000, 10200, 9200, 8500, 7000, 6000] + `, + {} + ); + cy.get('svg'); + }); + it('Render with show axis title false', () => { + imgSnapshotTest( + ` + --- + config: + xyChart: + xAxis: + showTitle: false + yAxis: + showTitle: false + --- + 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] + line [5000, 6000, 7500, 8200, 9500, 10500, 11000, 10200, 9200, 8500, 7000, 6000] + `, + {} + ); + cy.get('svg'); + }); + it('Render with show axis label false', () => { + imgSnapshotTest( + ` + --- + config: + xyChart: + xAxis: + showLabel: false + yAxis: + showLabel: false + --- + 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] + line [5000, 6000, 7500, 8200, 9500, 10500, 11000, 10200, 9200, 8500, 7000, 6000] + `, + {} + ); + cy.get('svg'); + }); + it('Render with show axis tick false', () => { + imgSnapshotTest( + ` + --- + config: + xyChart: + xAxis: + showTick: false + yAxis: + showTick: false + --- + 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] + line [5000, 6000, 7500, 8200, 9500, 10500, 11000, 10200, 9200, 8500, 7000, 6000] + `, + {} + ); + cy.get('svg'); + }); + it('Render with show axis line false', () => { + imgSnapshotTest( + ` + --- + config: + xyChart: + xAxis: + showAxisLine: false + yAxis: + showAxisLine: false + --- + 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] + line [5000, 6000, 7500, 8200, 9500, 10500, 11000, 10200, 9200, 8500, 7000, 6000] + `, + {} + ); + cy.get('svg'); + }); + it('Render all the theme color', () => { + imgSnapshotTest( + ` + --- + config: + themeVariables: + xyChart: + titleColor: "#ff0000" + backgroundColor: "#f0f8ff" + yAxisLabelColor: "#ee82ee" + yAxisTitleColor: "#7fffd4" + yAxisTickColor: "#87ceeb" + yAxisLineColor: "#ff6347" + xAxisLabelColor: "#7fffd4" + xAxisTitleColor: "#ee82ee" + xAxisTickColor: "#ff6347" + xAxisLineColor: "#87ceeb" + plotColorPalette: "#008000, #faba63" + --- + 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] + line [5000, 6000, 7500, 8200, 9500, 10500, 11000, 10200, 9200, 8500, 7000, 6000] + `, + {} + ); + cy.get('svg'); + }); +}); diff --git a/cypress/platform/knsv2.html b/cypress/platform/knsv2.html index 6ade6a2e5c..020ea8b482 100644 --- a/cypress/platform/knsv2.html +++ b/cypress/platform/knsv2.html @@ -58,8 +58,19 @@
-      classDiagram
-  `Class` <|-- `Class2`
+flowchart TB
+    C & D & E & F & G & H & I & J & K & L & M & N & O & P & Q & R & S & T & U & V & W & X & Y & Z & A1 & A2 & A3 & A4 & A5 & A6 & A7 & A8
+      ---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------->
+    C & D & E & F & G & H & I & J & K & L & M & N & O & P & Q & R & S & T & U & V & W & X & Y & Z & A1 & A2 & A3 & A4 & A5 & A6 & A7 & A8
+
+    
+
+    flowchart TB
+      A & A & A & A & A & A & A & A --->  C & D & E & F & G & H & I & J & K & L & M & N & O & P & Q & R & S & T & U & V & W & X & Y & Z
+    
+
+    flowchart TB
+      A1 & A2 & A3 & A4 & A5 & A6 & A7 & A8 -->  C & D & E & F & G & H & I & J & K & L & M & N & O & P & Q & R & S & T & U & V & W & X & Y & Z
     
 flowchart
@@ -439,6 +450,7 @@
           messageFontFamily: 'courier',
         },
         fontSize: 16,
+        logLevel: 0,
       });
       function callback() {
         alert('It worked');
diff --git a/demos/c4context.html b/demos/c4context.html
index cf358b5501..f674054a2f 100644
--- a/demos/c4context.html
+++ b/demos/c4context.html
@@ -173,7 +173,7 @@ 

C4 context diagram demos

Container(mobile, "Mobile App", "Xamarin", "Provides a limited subset of the Internet Banking functionality to customers via their mobile device.") } - Deployment_Node(comp, "Customer's computer", "Mircosoft Windows or Apple macOS"){ + Deployment_Node(comp, "Customer's computer", "Microsoft Windows or Apple macOS"){ Deployment_Node(browser, "Web Browser", "Google Chrome, Mozilla Firefox,
Apple Safari or Microsoft Edge"){ Container(spa, "Single Page Application", "JavaScript and Angular", "Provides all of the Internet Banking functionality to customers via their web browser.") } diff --git a/demos/flowchart.html b/demos/flowchart.html index 8389510b28..d7032a663c 100644 --- a/demos/flowchart.html +++ b/demos/flowchart.html @@ -22,7 +22,7 @@

graph

--- graph LR accTitle: This is a complicated flow - accDescr: This is the descriptoin for the complicated flow. + accDescr: This is the description for the complicated flow. sid-B3655226-6C29-4D00-B685-3D5C734DC7E1[" diff --git a/demos/index.html b/demos/index.html index 24c4fbf3b0..c634aad2d4 100644 --- a/demos/index.html +++ b/demos/index.html @@ -60,6 +60,9 @@

Pie

  • Quadrant charts

  • +
  • +

    XY charts

    +
  • Requirements

  • diff --git a/demos/requirements.html b/demos/requirements.html index 3ede080588..2510db8dd1 100644 --- a/demos/requirements.html +++ b/demos/requirements.html @@ -17,7 +17,7 @@

    Requirement diagram demos

     	requirementDiagram
    -	  accTitle: Requirments demo in black and white
    +	  accTitle: Requirements demo in black and white
     		accDescr: A series of requirement boxes showing relationships among them. Has meaningless task names
     
     	requirement test_req {
    diff --git a/demos/sequence.html b/demos/sequence.html
    index b2733a384e..3345fed17a 100644
    --- a/demos/sequence.html
    +++ b/demos/sequence.html
    @@ -164,6 +164,13 @@ 

    Sequence diagram demos

    end
    +
    +    sequenceDiagram
    +      actor Alice
    +      actor John
    +      Alice-xJohn: Hello John, how are you?
    +      John--xAlice: Great!
    +    
    + + diff --git a/docker-compose.yml b/docker-compose.yml index 125ce5da8f..e2484bdc52 100644 --- a/docker-compose.yml +++ b/docker-compose.yml @@ -1,7 +1,7 @@ version: '3.9' services: mermaid: - image: node:18.18.0-alpine3.18 + image: node:18.18.2-alpine3.18 stdin_open: true tty: true working_dir: /mermaid diff --git a/docs/community/questions-and-suggestions.md b/docs/community/questions-and-suggestions.md index 23d6de50f3..badb53a35a 100644 --- a/docs/community/questions-and-suggestions.md +++ b/docs/community/questions-and-suggestions.md @@ -10,9 +10,8 @@ ## First search to see if someone has already asked (and hopefully been answered) or suggested the same thing. -- Search in Discussions -- Search in open Issues -- Search in closed Issues +- [Search in Discussions](https://github.com/orgs/mermaid-js/discussions) +- [Search in Issues (Open & Closed)](https://github.com/mermaid-js/mermaid/issues?q=is%3Aissue) If you find an open issue or discussion thread that is similar to your question but isn't answered, you can let us know that you are also interested in it. Use the GitHub reactions to add a thumbs-up to the issue or discussion thread. diff --git a/docs/config/accessibility.md b/docs/config/accessibility.md index bf8b3e5915..836d6bcb23 100644 --- a/docs/config/accessibility.md +++ b/docs/config/accessibility.md @@ -97,7 +97,7 @@ See [the accTitle and accDescr usage examples](#acctitle-and-accdescr-usage-exam graph LR accTitle: Big Decisions accDescr: Bob's Burgers process for making big decisions - A[Identify Big Descision] --> B{Make Big Decision} + A[Identify Big Decision] --> B{Make Big Decision} B --> D[Be done] ``` @@ -105,7 +105,7 @@ See [the accTitle and accDescr usage examples](#acctitle-and-accdescr-usage-exam graph LR accTitle: Big Decisions accDescr: Bob's Burgers process for making big decisions - A[Identify Big Descision] --> B{Make Big Decision} + A[Identify Big Decision] --> B{Make Big Decision} B --> D[Be done] ``` @@ -137,7 +137,7 @@ Here is the HTML generated for the SVG element: _(Note that some of the SVG attr for making very, very big decisions. This is actually a very simple flow: identify the big decision and then make the big decision. } - A[Identify Big Descision] --> B{Make Big Decision} + A[Identify Big Decision] --> B{Make Big Decision} B --> D[Be done] ``` @@ -149,7 +149,7 @@ Here is the HTML generated for the SVG element: _(Note that some of the SVG attr for making very, very big decisions. This is actually a very simple flow: identify the big decision and then make the big decision. } - A[Identify Big Descision] --> B{Make Big Decision} + A[Identify Big Decision] --> B{Make Big Decision} B --> D[Be done] ``` diff --git a/docs/config/configuration.md b/docs/config/configuration.md index eb703a9d29..05fcd9d468 100644 --- a/docs/config/configuration.md +++ b/docs/config/configuration.md @@ -10,8 +10,8 @@ When mermaid starts, configuration is extracted to determine a configuration to - The default configuration - Overrides at the site level are set by the initialize call, and will be applied to all diagrams in the site/app. The term for this is the **siteConfig**. -- Frontmatter (v10.5.0+) - diagram authors can update select configuration parameters in the frontmatter of the diagram. These are applied to the render config. -- Directives (Deprecated by Frontmatter) - diagram authors can update select configuration parameters directly in the diagram code via directives. These are applied to the render config. +- Frontmatter (v10.5.0+) - diagram authors can update selected configuration parameters in the frontmatter of the diagram. These are applied to the render config. +- Directives (Deprecated by Frontmatter) - diagram authors can update selected configuration parameters directly in the diagram code via directives. These are applied to the render config. **The render config** is configuration that is used when rendering by applying these configurations. diff --git a/docs/config/setup/interfaces/mermaidAPI.ParseOptions.md b/docs/config/setup/interfaces/mermaidAPI.ParseOptions.md index ea390899e4..4386be9380 100644 --- a/docs/config/setup/interfaces/mermaidAPI.ParseOptions.md +++ b/docs/config/setup/interfaces/mermaidAPI.ParseOptions.md @@ -16,4 +16,4 @@ #### Defined in -[mermaidAPI.ts:59](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/mermaidAPI.ts#L59) +[mermaidAPI.ts:60](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/mermaidAPI.ts#L60) diff --git a/docs/config/setup/interfaces/mermaidAPI.RenderResult.md b/docs/config/setup/interfaces/mermaidAPI.RenderResult.md index 18ee5e4316..6209782f75 100644 --- a/docs/config/setup/interfaces/mermaidAPI.RenderResult.md +++ b/docs/config/setup/interfaces/mermaidAPI.RenderResult.md @@ -39,7 +39,7 @@ bindFunctions?.(div); // To call bindFunctions only if it's present. #### Defined in -[mermaidAPI.ts:79](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/mermaidAPI.ts#L79) +[mermaidAPI.ts:80](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/mermaidAPI.ts#L80) --- @@ -51,4 +51,4 @@ The svg code for the rendered graph. #### Defined in -[mermaidAPI.ts:69](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/mermaidAPI.ts#L69) +[mermaidAPI.ts:70](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/mermaidAPI.ts#L70) diff --git a/docs/config/setup/modules/defaultConfig.md b/docs/config/setup/modules/defaultConfig.md index effaec7b13..7a9b891c43 100644 --- a/docs/config/setup/modules/defaultConfig.md +++ b/docs/config/setup/modules/defaultConfig.md @@ -14,7 +14,7 @@ #### Defined in -[defaultConfig.ts:268](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/defaultConfig.ts#L268) +[defaultConfig.ts:272](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/defaultConfig.ts#L272) --- diff --git a/docs/config/setup/modules/mermaidAPI.md b/docs/config/setup/modules/mermaidAPI.md index 0a948b6f39..6337807883 100644 --- a/docs/config/setup/modules/mermaidAPI.md +++ b/docs/config/setup/modules/mermaidAPI.md @@ -25,7 +25,7 @@ Renames and re-exports [mermaidAPI](mermaidAPI.md#mermaidapi) #### Defined in -[mermaidAPI.ts:63](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/mermaidAPI.ts#L63) +[mermaidAPI.ts:64](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/mermaidAPI.ts#L64) ## Variables @@ -96,7 +96,7 @@ mermaid.initialize(config); #### Defined in -[mermaidAPI.ts:641](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/mermaidAPI.ts#L641) +[mermaidAPI.ts:603](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/mermaidAPI.ts#L603) ## Functions @@ -127,7 +127,7 @@ Return the last node appended #### Defined in -[mermaidAPI.ts:299](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/mermaidAPI.ts#L299) +[mermaidAPI.ts:263](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/mermaidAPI.ts#L263) --- @@ -153,7 +153,7 @@ the cleaned up svgCode #### Defined in -[mermaidAPI.ts:245](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/mermaidAPI.ts#L245) +[mermaidAPI.ts:209](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/mermaidAPI.ts#L209) --- @@ -178,7 +178,7 @@ the string with all the user styles #### Defined in -[mermaidAPI.ts:175](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/mermaidAPI.ts#L175) +[mermaidAPI.ts:139](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/mermaidAPI.ts#L139) --- @@ -201,7 +201,7 @@ the string with all the user styles #### Defined in -[mermaidAPI.ts:222](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/mermaidAPI.ts#L222) +[mermaidAPI.ts:186](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/mermaidAPI.ts#L186) --- @@ -228,47 +228,7 @@ with an enclosing block that has each of the cssClasses followed by !important; #### Defined in -[mermaidAPI.ts:160](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/mermaidAPI.ts#L160) - ---- - -### decodeEntities - -▸ **decodeEntities**(`text`): `string` - -#### Parameters - -| Name | Type | Description | -| :----- | :------- | :----------------- | -| `text` | `string` | text to be decoded | - -#### Returns - -`string` - -#### Defined in - -[mermaidAPI.ts:146](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/mermaidAPI.ts#L146) - ---- - -### encodeEntities - -▸ **encodeEntities**(`text`): `string` - -#### Parameters - -| Name | Type | Description | -| :----- | :------- | :----------------- | -| `text` | `string` | text to be encoded | - -#### Returns - -`string` - -#### Defined in - -[mermaidAPI.ts:117](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/mermaidAPI.ts#L117) +[mermaidAPI.ts:124](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/mermaidAPI.ts#L124) --- @@ -294,7 +254,7 @@ Put the svgCode into an iFrame. Return the iFrame code #### Defined in -[mermaidAPI.ts:276](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/mermaidAPI.ts#L276) +[mermaidAPI.ts:240](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/mermaidAPI.ts#L240) --- @@ -319,4 +279,4 @@ Remove any existing elements from the given document #### Defined in -[mermaidAPI.ts:349](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/mermaidAPI.ts#L349) +[mermaidAPI.ts:313](https://github.com/mermaid-js/mermaid/blob/master/packages/mermaid/src/mermaidAPI.ts#L313) diff --git a/docs/ecosystem/integrations-community.md b/docs/ecosystem/integrations-community.md index 2c67cc3618..e979544cff 100644 --- a/docs/ecosystem/integrations-community.md +++ b/docs/ecosystem/integrations-community.md @@ -35,12 +35,19 @@ Below are a list of community plugins and integrations created with Mermaid. - [Notion](https://notion.so) ✅ - [Observable](https://observablehq.com/@observablehq/mermaid) ✅ - [Obsidian](https://help.obsidian.md/Editing+and+formatting/Advanced+formatting+syntax#Diagram) ✅ +- [NotesHub](https://noteshub.app) ✅ - [GitBook](https://gitbook.com) - [Mermaid Plugin](https://github.com/JozoVilcek/gitbook-plugin-mermaid) - [Markdown with Mermaid CLI](https://github.com/miao1007/gitbook-plugin-mermaid-cli) - [Mermaid plugin for GitBook](https://github.com/wwformat/gitbook-plugin-mermaid-pdf) - [LiveBook](https://livebook.dev) ✅ - [Atlassian Products](https://www.atlassian.com) + - [Mermaid for Confluence](https://marketplace.atlassian.com/apps/1224722/mermaid-for-confluence?hosting=cloud&tab=overview) + - [Mermaid Integration for Confluence](https://marketplace.atlassian.com/apps/1222792/mermaid-integration-for-confluence?hosting=cloud&tab=overview) + - [Mermaid Diagrams for Confluence](https://marketplace.atlassian.com/apps/1226945/mermaid-diagrams-for-confluence?hosting=cloud&tab=overview) + - [Mermaid Macro for Confluence](https://marketplace.atlassian.com/apps/1231150/mermaid-macro-for-confluence?hosting=cloud&tab=overview) + - [EliteSoft Mermaid Charts and Diagrams](https://marketplace.atlassian.com/apps/1227286/elitesoft-mermaid-charts-and-diagrams?hosting=cloud&tab=overview) + - [Mermaid for Jira Cloud - Draw UML diagrams easily](https://marketplace.atlassian.com/apps/1223053/mermaid-for-jira-cloud-draw-uml-diagrams-easily?hosting=cloud&tab=overview) - [Mermaid Charts & Diagrams for Confluence](https://marketplace.atlassian.com/apps/1222572/) - [Mermaid Charts & Diagrams for Jira](https://marketplace.atlassian.com/apps/1224537/) - [Mermaid Live Editor for Confluence Cloud](https://marketplace.atlassian.com/apps/1231571/mermaid-live-editor-for-confluence?hosting=cloud&tab=overview) @@ -99,6 +106,8 @@ Communication tools and platforms - [phpbb-ext-mermaid](https://github.com/AlfredoRamos/phpbb-ext-mermaid) - [NodeBB](https://nodebb.org) - [Mermaid Plugin](https://www.npmjs.com/package/nodebb-plugin-mermaid) +- [Slack](https://slack.com) + - [Mermaid for Slack](https://github.com/JackuB/mermaid-for-slack) ### Wikis @@ -169,13 +178,14 @@ Communication tools and platforms ### Document Generation +- [Docusaurus](https://docusaurus.io/docs/markdown-features/diagrams) ✅ - [Swimm - Up-to-date diagrams with Swimm, the knowledge management tool for code](https://docs.swimm.io/Features/diagrams-and-charts) - [Sphinx](https://www.sphinx-doc.org/en/master/) - [sphinxcontrib-mermaid](https://github.com/mgaitan/sphinxcontrib-mermaid) - [remark](https://remark.js.org/) - [remark-mermaidjs](https://github.com/remcohaszing/remark-mermaidjs) - [rehype](https://github.com/rehypejs/rehype) - - [rehype-mermaidjs](https://github.com/remcohaszing/rehype-mermaidjs) + - [rehype-mermaid](https://github.com/remcohaszing/rehype-mermaid) - [Gatsby](https://www.gatsbyjs.com/) - [gatsby-remark-mermaid](https://github.com/remcohaszing/gatsby-remark-mermaid) - [JSDoc](https://jsdoc.app/) diff --git a/docs/intro/index.md b/docs/intro/index.md index 50b7e416e9..ab5b1a5002 100644 --- a/docs/intro/index.md +++ b/docs/intro/index.md @@ -283,6 +283,26 @@ quadrantChart Campaign F: [0.35, 0.78] ``` +### [XY Chart](../syntax/xyChart.md) + +```mermaid-example +xychart-beta + title "Sales Revenue" + x-axis [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] + line [5000, 6000, 7500, 8200, 9500, 10500, 11000, 10200, 9200, 8500, 7000, 6000] +``` + +```mermaid +xychart-beta + title "Sales Revenue" + x-axis [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] + line [5000, 6000, 7500, 8200, 9500, 10500, 11000, 10200, 9200, 8500, 7000, 6000] +``` + ## Installation **In depth guides and examples can be found at [Getting Started](./getting-started.md) and [Usage](../config/usage.md).** @@ -320,7 +340,7 @@ To Deploy Mermaid: ``` -**Doing so commands the mermaid parser to look for the `
    ` or `
    ` tags with `class="mermaid"`. From these tags, mermaid tries read the diagram/chart definitions and render them into SVG charts.**
    +**Doing so commands the mermaid parser to look for the `
    ` or `
    ` tags with `class="mermaid"`. From these tags, mermaid tries to read the diagram/chart definitions and render them into SVG charts.**
     
     **Examples can be found in** [Other examples](../syntax/examples.md)
     
    diff --git a/docs/syntax/c4.md b/docs/syntax/c4.md
    index 1676708f50..e6b7736c33 100644
    --- a/docs/syntax/c4.md
    +++ b/docs/syntax/c4.md
    @@ -399,7 +399,7 @@ UpdateRelStyle(customerA, bankA, $offsetY="60")
         title Component diagram for Internet Banking System - API Application
     
         Container(spa, "Single Page Application", "javascript and angular", "Provides all the internet banking functionality to customers via their web browser.")
    -    Container(ma, "Mobile App", "Xamarin", "Provides a limited subset ot the internet banking functionality to customers via their mobile mobile device.")
    +    Container(ma, "Mobile App", "Xamarin", "Provides a limited subset to the internet banking functionality to customers via their mobile mobile device.")
         ContainerDb(db, "Database", "Relational Database Schema", "Stores user registration information, hashed authentication credentials, access logs, etc.")
         System_Ext(mbs, "Mainframe Banking System", "Stores all of the core banking information about customers, accounts, transactions, etc.")
     
    @@ -439,7 +439,7 @@ UpdateRelStyle(customerA, bankA, $offsetY="60")
         title Component diagram for Internet Banking System - API Application
     
         Container(spa, "Single Page Application", "javascript and angular", "Provides all the internet banking functionality to customers via their web browser.")
    -    Container(ma, "Mobile App", "Xamarin", "Provides a limited subset ot the internet banking functionality to customers via their mobile mobile device.")
    +    Container(ma, "Mobile App", "Xamarin", "Provides a limited subset to the internet banking functionality to customers via their mobile mobile device.")
         ContainerDb(db, "Database", "Relational Database Schema", "Stores user registration information, hashed authentication credentials, access logs, etc.")
         System_Ext(mbs, "Mainframe Banking System", "Stores all of the core banking information about customers, accounts, transactions, etc.")
     
    diff --git a/docs/syntax/classDiagram.md b/docs/syntax/classDiagram.md
    index a6109149a1..2f2c3da888 100644
    --- a/docs/syntax/classDiagram.md
    +++ b/docs/syntax/classDiagram.md
    @@ -425,8 +425,6 @@ And `Link` can be one of:
     
     A namespace groups classes.
     
    -Code:
    -
     ```mermaid-example
     classDiagram
     namespace BaseShapes {
    diff --git a/docs/syntax/flowchart.md b/docs/syntax/flowchart.md
    index acd7c2db5b..d9ddf0cbe4 100644
    --- a/docs/syntax/flowchart.md
    +++ b/docs/syntax/flowchart.md
    @@ -467,7 +467,7 @@ flowchart TB
         A & B--> C & D
     ```
     
    -If you describe the same diagram using the the basic syntax, it will take four lines. A
    +If you describe the same diagram using the basic syntax, it will take four lines. A
     word of warning, one could go overboard with this making the flowchart harder to read in
     markdown form. The Swedish word `lagom` comes to mind. It means, not too much and not too little.
     This goes for expressive syntaxes as well.
    @@ -764,7 +764,7 @@ flowchart LR
         end
         %% ^ These subgraphs are identical, except for the links to them:
     
    -    %% Link *to* subgraph1: subgraph1 direction is mantained
    +    %% Link *to* subgraph1: subgraph1 direction is maintained
         outside --> subgraph1
         %% Link *within* subgraph2:
         %% subgraph2 inherits the direction of the top-level graph (LR)
    @@ -783,7 +783,7 @@ flowchart LR
         end
         %% ^ These subgraphs are identical, except for the links to them:
     
    -    %% Link *to* subgraph1: subgraph1 direction is mantained
    +    %% Link *to* subgraph1: subgraph1 direction is maintained
         outside --> subgraph1
         %% Link *within* subgraph2:
         %% subgraph2 inherits the direction of the top-level graph (LR)
    @@ -1112,7 +1112,7 @@ flowchart TD
         B-->E(A fa:fa-camera-retro perhaps?)
     ```
     
    -Mermaid is compatible with Font Awesome up to verion 5, Free icons only. Check that the icons you use are from the [supported set of icons](https://fontawesome.com/v5/search?o=r&m=free).
    +Mermaid is compatible with Font Awesome up to version 5, Free icons only. Check that the icons you use are from the [supported set of icons](https://fontawesome.com/v5/search?o=r&m=free).
     
     ## Graph declarations with spaces between vertices and link and without semicolon
     
    diff --git a/docs/syntax/gitgraph.md b/docs/syntax/gitgraph.md
    index f2609e31c5..8d39ddbcbd 100644
    --- a/docs/syntax/gitgraph.md
    +++ b/docs/syntax/gitgraph.md
    @@ -827,18 +827,25 @@ Here, we have changed the default main branch name to `MetroLine1`.
     
     ## Orientation (v10.3.0+)
     
    -In Mermaid, the default orientation is Left to Right. The branches are lined vertically.
    +Mermaid supports two graph orientations: **Left-to-Right** (default) and **Top-to-Bottom**.
    +
    +You can set this with either `LR:` (for [**Left-to-Right**](#left-to-right-default-lr)) or `TB:` (for [**Top-to-Bottom**](#top-to-bottom-tb)) after `gitGraph`.
    +
    +### Left to Right (default, `LR:`)
    +
    +In Mermaid, the default orientation is for commits to run from left to right and for branches to be stacked on top of one another.
    +
    +However, you can set this explicitly with `LR:` after `gitGraph`.
     
     Usage example:
     
     ```mermaid-example
    -    gitGraph
    +    gitGraph LR:
            commit
            commit
            branch develop
            commit
            commit
    -       commit
            checkout main
            commit
            commit
    @@ -848,13 +855,12 @@ Usage example:
     ```
     
     ```mermaid
    -    gitGraph
    +    gitGraph LR:
            commit
            commit
            branch develop
            commit
            commit
    -       commit
            checkout main
            commit
            commit
    @@ -863,9 +869,11 @@ Usage example:
            commit
     ```
     
    -Sometimes we may want to change the orientation. Currently, Mermaid supports two orientations: **Left to Right**(default) and **Top to Bottom**.
    +### Top to Bottom (`TB:`)
    +
    +In `TB` (**Top-to-Bottom**) orientation, the commits run from top to bottom of the graph and branches are arranged side-by-side.
     
    -In order to change the orientation from top to bottom i.e. branches lined horizontally, you need to add `TB` along with `gitGraph`.
    +To orient the graph this way, you need to add `TB:` after gitGraph.
     
     Usage example:
     
    @@ -876,7 +884,6 @@ Usage example:
            branch develop
            commit
            commit
    -       commit
            checkout main
            commit
            commit
    @@ -892,7 +899,6 @@ Usage example:
            branch develop
            commit
            commit
    -       commit
            checkout main
            commit
            commit
    diff --git a/docs/syntax/quadrantChart.md b/docs/syntax/quadrantChart.md
    index 39b57fd131..9f22fd5753 100644
    --- a/docs/syntax/quadrantChart.md
    +++ b/docs/syntax/quadrantChart.md
    @@ -102,26 +102,26 @@ Points are used to plot a circle inside the quadrantChart. The syntax is `
     
     ## Chart Configurations
     
    -| Parameter                         | Description                                                                                       | Default value |
    -| --------------------------------- | ------------------------------------------------------------------------------------------------- | :-----------: |
    -| chartWidth                        | Width of the chart                                                                                |      500      |
    -| chartHeight                       | Height of the chart                                                                               |      500      |
    -| titlePadding                      | Top and Bottom padding of the title                                                               |      10       |
    -| titleFontSize                     | Title font size                                                                                   |      20       |
    -| quadrantPadding                   | Padding outside all the quadrants                                                                 |       5       |
    -| quadrantTextTopPadding            | Quadrant text top padding when text is drawn on top ( not data points are there)                  |       5       |
    -| quadrantLabelFontSize             | Quadrant text font size                                                                           |      16       |
    -| quadrantInternalBorderStrokeWidth | Border stroke width inside the quadrants                                                          |       1       |
    -| quadrantExternalBorderStrokeWidth | Quadrant external border stroke width                                                             |       2       |
    -| xAxisLabelPadding                 | Top and bottom padding of x-axis text                                                             |       5       |
    -| xAxisLabelFontSize                | X-axis texts font size                                                                            |      16       |
    -| xAxisPosition                     | Position of x-axis (top , bottom) if there are points the x-axis will alway be rendered in bottom |     'top'     |
    -| yAxisLabelPadding                 | Left and Right padding of y-axis text                                                             |       5       |
    -| yAxisLabelFontSize                | Y-axis texts font size                                                                            |      16       |
    -| yAxisPosition                     | Position of y-axis (left , right)                                                                 |    'left'     |
    -| pointTextPadding                  | Padding between point and the below text                                                          |       5       |
    -| pointLabelFontSize                | Point text font size                                                                              |      12       |
    -| pointRadius                       | Radius of the point to be drawn                                                                   |       5       |
    +| Parameter                         | Description                                                                                        | Default value |
    +| --------------------------------- | -------------------------------------------------------------------------------------------------- | :-----------: |
    +| chartWidth                        | Width of the chart                                                                                 |      500      |
    +| chartHeight                       | Height of the chart                                                                                |      500      |
    +| titlePadding                      | Top and Bottom padding of the title                                                                |      10       |
    +| titleFontSize                     | Title font size                                                                                    |      20       |
    +| quadrantPadding                   | Padding outside all the quadrants                                                                  |       5       |
    +| quadrantTextTopPadding            | Quadrant text top padding when text is drawn on top ( not data points are there)                   |       5       |
    +| quadrantLabelFontSize             | Quadrant text font size                                                                            |      16       |
    +| quadrantInternalBorderStrokeWidth | Border stroke width inside the quadrants                                                           |       1       |
    +| quadrantExternalBorderStrokeWidth | Quadrant external border stroke width                                                              |       2       |
    +| xAxisLabelPadding                 | Top and bottom padding of x-axis text                                                              |       5       |
    +| xAxisLabelFontSize                | X-axis texts font size                                                                             |      16       |
    +| xAxisPosition                     | Position of x-axis (top , bottom) if there are points the x-axis will always be rendered in bottom |     'top'     |
    +| yAxisLabelPadding                 | Left and Right padding of y-axis text                                                              |       5       |
    +| yAxisLabelFontSize                | Y-axis texts font size                                                                             |      16       |
    +| yAxisPosition                     | Position of y-axis (left , right)                                                                  |    'left'     |
    +| pointTextPadding                  | Padding between point and the below text                                                           |       5       |
    +| pointLabelFontSize                | Point text font size                                                                               |      12       |
    +| pointRadius                       | Radius of the point to be drawn                                                                    |       5       |
     
     ## Chart Theme Variables
     
    diff --git a/docs/syntax/timeline.md b/docs/syntax/timeline.md
    index 5eafc37c31..5c2894c80d 100644
    --- a/docs/syntax/timeline.md
    +++ b/docs/syntax/timeline.md
    @@ -217,7 +217,7 @@ Note that there are no sections defined, and each time period and its correspond
     
     2.  Disable the multiColor option using the `disableMultiColor` option. This will make all time periods and events follow the same color scheme.
     
    -You will need to add this option either via mermaid.intialize function or directives.
    +You will need to add this option either via mermaid.initialize function or directives.
     
     ```javascript
     mermaid.initialize({
    diff --git a/docs/syntax/xyChart.md b/docs/syntax/xyChart.md
    new file mode 100644
    index 0000000000..7e91863f9a
    --- /dev/null
    +++ b/docs/syntax/xyChart.md
    @@ -0,0 +1,188 @@
    +> **Warning**
    +>
    +> ## THIS IS AN AUTOGENERATED FILE. DO NOT EDIT.
    +>
    +> ## Please edit the corresponding file in [/packages/mermaid/src/docs/syntax/xyChart.md](../../packages/mermaid/src/docs/syntax/xyChart.md).
    +
    +# XY Chart
    +
    +> In the context of mermaid-js, the XY chart is a comprehensive charting module that encompasses various types of charts that utilize both x-axis and y-axis for data representation. Presently, it includes two fundamental chart types: the bar chart and the line chart. These charts are designed to visually display and analyze data that involve two numerical variables.
    +
    +> It's important to note that while the current implementation of mermaid-js includes these two chart types, the framework is designed to be dynamic and adaptable. Therefore, it has the capacity for expansion and the inclusion of additional chart types in the future. This means that users can expect an evolving suite of charting options within the XY chart module, catering to various data visualization needs as new chart types are introduced over time.
    +
    +## Example
    +
    +```mermaid-example
    +xychart-beta
    +    title "Sales Revenue"
    +    x-axis [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]
    +    line [5000, 6000, 7500, 8200, 9500, 10500, 11000, 10200, 9200, 8500, 7000, 6000]
    +```
    +
    +```mermaid
    +xychart-beta
    +    title "Sales Revenue"
    +    x-axis [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]
    +    line [5000, 6000, 7500, 8200, 9500, 10500, 11000, 10200, 9200, 8500, 7000, 6000]
    +```
    +
    +## Syntax
    +
    +> **Note**
    +> All text values that contain only one word can be written without `"`. If a text value has many words in it, specifically if it contains spaces, enclose the value in `"`
    +
    +### Orientations
    +
    +The chart can be drawn horizontal or vertical, default value is vertical.
    +
    +    xychart-beta horizontal
    +    ...
    +
    +### Title
    +
    +The title is a short description of the chart and it will always render on top of the chart.
    +
    +#### Example
    +
    +    xychart-beta
    +        title "This is a simple example"
    +        ...
    +
    +> **Note**
    +> If the title is a single word one no need to use `"`, but if it has space `"` is needed
    +
    +### x-axis
    +
    +The x-axis primarily serves as a categorical value, although it can also function as a numeric range value when needed.
    +
    +#### Example
    +
    +1.  `x-axis title min --> max` x-axis will function as numeric with the given range
    +2.  `x-axis "title with space" [cat1, "cat2 with space", cat3]` x-axis if categorical, categories are text type
    +
    +### y-axis
    +
    +The y-axis is employed to represent numerical range values, it cannot have categorical values.
    +
    +#### Example
    +
    +1.  `y-axis title min --> max`
    +2.  `y-axis title` it will only add the title, the range will be auto generated from data.
    +
    +> **Note**
    +> Both x and y axis are optional if not provided we will try to create the range
    +
    +### Line chart
    +
    +A line chart offers the capability to graphically depict lines.
    +
    +#### Example
    +
    +1.  `line [2.3, 45, .98, -3.4]` it can have all valid numeric values.
    +
    +### Bar chart
    +
    +A bar chart offers the capability to graphically depict bars.
    +
    +#### Example
    +
    +1.  `bar [2.3, 45, .98, -3.4]` it can have all valid numeric values.
    +
    +#### Simplest example
    +
    +The only two things required are the chart name (`xychart-beta`) and one data set. So you will be able to draw a chart with a simple config like
    +
    +    xychart-beta
    +        line [+1.3, .6, 2.4, -.34]
    +
    +## Chart Configurations
    +
    +| Parameter                | Description                                    | Default value |
    +| ------------------------ | ---------------------------------------------- | :-----------: |
    +| width                    | Width of the chart                             |      700      |
    +| height                   | Height of the chart                            |      500      |
    +| titlePadding             | Top and Bottom padding of the title            |      10       |
    +| titleFontSize            | Title font size                                |      20       |
    +| showTitle                | Title to be shown or not                       |     true      |
    +| xAxis                    | xAxis configuration                            |  AxisConfig   |
    +| yAxis                    | yAxis configuration                            |  AxisConfig   |
    +| chartOrientation         | 'vertical' or 'horizontal'                     |  'vertical'   |
    +| plotReservedSpacePercent | Minimum space plots will take inside the chart |      50       |
    +
    +### AxisConfig
    +
    +| Parameter     | Description                          | Default value |
    +| ------------- | ------------------------------------ | :-----------: |
    +| showLabel     | Show axis labels or tick values      |     true      |
    +| labelFontSize | Font size of the label to be drawn   |      14       |
    +| labelPadding  | Top and Bottom padding of the label  |       5       |
    +| showTitle     | Axis title to be shown or not        |     true      |
    +| titleFontSize | Axis title font size                 |      16       |
    +| titlePadding  | Top and Bottom padding of Axis title |       5       |
    +| showTick      | Tick to be shown or not              |     true      |
    +| tickLength    | How long the tick will be            |       5       |
    +| tickWidth     | How width the tick will be           |       2       |
    +| showAxisLine  | Axis line to be shown or not         |     true      |
    +| axisLineWidth | Thickness of the axis line           |       2       |
    +
    +## Chart Theme Variables
    +
    +> **Note**
    +> Themes for xychart resides inside xychart attribute so to set the variables use this syntax
    +> %%{init: { "themeVariables": {"xyChart": {"titleColor": "#ff0000"} } }}%%
    +
    +| Parameter        | Description                                               |
    +| ---------------- | --------------------------------------------------------- |
    +| backgroundColor  | Background color of the whole chart                       |
    +| titleColor       | Color of the Title text                                   |
    +| xAxisLableColor  | Color of the x-axis labels                                |
    +| xAxisTitleColor  | Color of the x-axis title                                 |
    +| xAxisTickColor   | Color of the x-axis tick                                  |
    +| xAxisLineColor   | Color of the x-axis line                                  |
    +| yAxisLableColor  | Color of the y-axis labels                                |
    +| yAxisTitleColor  | Color of the y-axis title                                 |
    +| yAxisTickColor   | Color of the y-axis tick                                  |
    +| yAxisLineColor   | Color of the y-axis line                                  |
    +| plotColorPalette | String of colors separated by comma e.g. "#f3456, #43445" |
    +
    +## Example on config and theme
    +
    +```mermaid-example
    +---
    +config:
    +    xyChart:
    +        width: 900
    +        height: 600
    +    themeVariables:
    +        xyChart:
    +            titleColor: "#ff0000"
    +---
    +xychart-beta
    +    title "Sales Revenue"
    +    x-axis [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]
    +    line [5000, 6000, 7500, 8200, 9500, 10500, 11000, 10200, 9200, 8500, 7000, 6000]
    +```
    +
    +```mermaid
    +---
    +config:
    +    xyChart:
    +        width: 900
    +        height: 600
    +    themeVariables:
    +        xyChart:
    +            titleColor: "#ff0000"
    +---
    +xychart-beta
    +    title "Sales Revenue"
    +    x-axis [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]
    +    line [5000, 6000, 7500, 8200, 9500, 10500, 11000, 10200, 9200, 8500, 7000, 6000]
    +```
    diff --git a/package.json b/package.json
    index 56bbfb1858..1cd2352d7f 100644
    --- a/package.json
    +++ b/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@8.7.6",
    +  "packageManager": "pnpm@8.10.4",
       "keywords": [
         "diagram",
         "markdown",
    @@ -89,7 +89,7 @@
         "cypress": "^12.10.0",
         "cypress-image-snapshot": "^4.0.1",
         "esbuild": "^0.19.0",
    -    "eslint": "^8.39.0",
    +    "eslint": "^8.47.0",
         "eslint-config-prettier": "^8.8.0",
         "eslint-plugin-cypress": "^2.13.2",
         "eslint-plugin-html": "^7.1.0",
    @@ -125,7 +125,7 @@
         "vitest": "^0.34.0"
       },
       "volta": {
    -    "node": "18.18.0"
    +    "node": "18.18.2"
       },
       "nyc": {
         "report-dir": "coverage/cypress"
    diff --git a/packages/mermaid/scripts/create-types-from-json-schema.mts b/packages/mermaid/scripts/create-types-from-json-schema.mts
    index 836aaa4481..b028fe818d 100644
    --- a/packages/mermaid/scripts/create-types-from-json-schema.mts
    +++ b/packages/mermaid/scripts/create-types-from-json-schema.mts
    @@ -47,6 +47,7 @@ const MERMAID_CONFIG_DIAGRAM_KEYS = [
       'er',
       'pie',
       'quadrantChart',
    +  'xyChart',
       'requirement',
       'mindmap',
       'timeline',
    diff --git a/packages/mermaid/src/Diagram.ts b/packages/mermaid/src/Diagram.ts
    index 5ee04c03c2..b56697e9de 100644
    --- a/packages/mermaid/src/Diagram.ts
    +++ b/packages/mermaid/src/Diagram.ts
    @@ -3,6 +3,8 @@ import { log } from './logger.js';
     import { getDiagram, registerDiagram } from './diagram-api/diagramAPI.js';
     import { detectType, getDiagramLoader } from './diagram-api/detectType.js';
     import { UnknownDiagramError } from './errors.js';
    +import { encodeEntities } from './utils.js';
    +
     import type { DetailedError } from './utils.js';
     import type { DiagramDefinition, DiagramMetadata } from './diagram-api/types.js';
     
    @@ -21,6 +23,7 @@ export class Diagram {
     
       private detectError?: UnknownDiagramError;
       constructor(public text: string, public metadata: Pick = {}) {
    +    this.text = encodeEntities(text);
         this.text += '\n';
         const cnf = configApi.getConfig();
         try {
    diff --git a/packages/mermaid/src/config.type.ts b/packages/mermaid/src/config.type.ts
    index bed6d8b9e8..1f97a644d4 100644
    --- a/packages/mermaid/src/config.type.ts
    +++ b/packages/mermaid/src/config.type.ts
    @@ -135,6 +135,7 @@ export interface MermaidConfig {
       er?: ErDiagramConfig;
       pie?: PieDiagramConfig;
       quadrantChart?: QuadrantChartConfig;
    +  xyChart?: XYChartConfig;
       requirement?: RequirementDiagramConfig;
       mindmap?: MindmapDiagramConfig;
       gitGraph?: GitGraphDiagramConfig;
    @@ -688,6 +689,194 @@ export interface QuadrantChartConfig extends BaseDiagramConfig {
        */
       quadrantExternalBorderStrokeWidth?: number;
     }
    +/**
    + * This object contains configuration for XYChart axis config
    + *
    + * This interface was referenced by `MermaidConfig`'s JSON-Schema
    + * via the `definition` "XYChartAxisConfig".
    + */
    +export interface XYChartAxisConfig {
    +  /**
    +   * Should show the axis labels (tick text)
    +   */
    +  showLabel?: boolean;
    +  /**
    +   * font size of the axis labels (tick text)
    +   */
    +  labelFontSize?: number;
    +  /**
    +   * top and bottom space from axis label (tick text)
    +   */
    +  labelPadding?: number;
    +  /**
    +   * Should show the axis title
    +   */
    +  showTitle?: boolean;
    +  /**
    +   * font size of the axis title
    +   */
    +  titleFontSize?: number;
    +  /**
    +   * top and bottom space from axis title
    +   */
    +  titlePadding?: number;
    +  /**
    +   * Should show the axis tick lines
    +   */
    +  showTick?: boolean;
    +  /**
    +   * length of the axis tick lines
    +   */
    +  tickLength?: number;
    +  /**
    +   * width of the axis tick lines
    +   */
    +  tickWidth?: number;
    +  /**
    +   * Show line across the axis
    +   */
    +  showAxisLine?: boolean;
    +  /**
    +   * Width of the axis line
    +   */
    +  axisLineWidth?: number;
    +}
    +/**
    + * This object contains configuration specific to XYCharts
    + *
    + * This interface was referenced by `MermaidConfig`'s JSON-Schema
    + * via the `definition` "XYChartConfig".
    + */
    +export interface XYChartConfig extends BaseDiagramConfig {
    +  /**
    +   * width of the chart
    +   */
    +  width?: number;
    +  /**
    +   * height of the chart
    +   */
    +  height?: number;
    +  /**
    +   * Font size of the chart title
    +   */
    +  titleFontSize?: number;
    +  /**
    +   * Top and bottom space from the chart title
    +   */
    +  titlePadding?: number;
    +  /**
    +   * Should show the chart title
    +   */
    +  showTitle?: boolean;
    +  xAxis?: XYChartAxisConfig1;
    +  yAxis?: XYChartAxisConfig2;
    +  /**
    +   * How to plot will be drawn horizontal or vertical
    +   */
    +  chartOrientation?: 'vertical' | 'horizontal';
    +  /**
    +   * Minimum percent of space plots of the chart will take
    +   */
    +  plotReservedSpacePercent?: number;
    +}
    +/**
    + * This object contains configuration for XYChart axis config
    + */
    +export interface XYChartAxisConfig1 {
    +  /**
    +   * Should show the axis labels (tick text)
    +   */
    +  showLabel?: boolean;
    +  /**
    +   * font size of the axis labels (tick text)
    +   */
    +  labelFontSize?: number;
    +  /**
    +   * top and bottom space from axis label (tick text)
    +   */
    +  labelPadding?: number;
    +  /**
    +   * Should show the axis title
    +   */
    +  showTitle?: boolean;
    +  /**
    +   * font size of the axis title
    +   */
    +  titleFontSize?: number;
    +  /**
    +   * top and bottom space from axis title
    +   */
    +  titlePadding?: number;
    +  /**
    +   * Should show the axis tick lines
    +   */
    +  showTick?: boolean;
    +  /**
    +   * length of the axis tick lines
    +   */
    +  tickLength?: number;
    +  /**
    +   * width of the axis tick lines
    +   */
    +  tickWidth?: number;
    +  /**
    +   * Show line across the axis
    +   */
    +  showAxisLine?: boolean;
    +  /**
    +   * Width of the axis line
    +   */
    +  axisLineWidth?: number;
    +}
    +/**
    + * This object contains configuration for XYChart axis config
    + */
    +export interface XYChartAxisConfig2 {
    +  /**
    +   * Should show the axis labels (tick text)
    +   */
    +  showLabel?: boolean;
    +  /**
    +   * font size of the axis labels (tick text)
    +   */
    +  labelFontSize?: number;
    +  /**
    +   * top and bottom space from axis label (tick text)
    +   */
    +  labelPadding?: number;
    +  /**
    +   * Should show the axis title
    +   */
    +  showTitle?: boolean;
    +  /**
    +   * font size of the axis title
    +   */
    +  titleFontSize?: number;
    +  /**
    +   * top and bottom space from axis title
    +   */
    +  titlePadding?: number;
    +  /**
    +   * Should show the axis tick lines
    +   */
    +  showTick?: boolean;
    +  /**
    +   * length of the axis tick lines
    +   */
    +  tickLength?: number;
    +  /**
    +   * width of the axis tick lines
    +   */
    +  tickWidth?: number;
    +  /**
    +   * Show line across the axis
    +   */
    +  showAxisLine?: boolean;
    +  /**
    +   * Width of the axis line
    +   */
    +  axisLineWidth?: number;
    +}
     /**
      * The object containing configurations specific for entity relationship diagrams
      *
    diff --git a/packages/mermaid/src/dagre-wrapper/clusters.js b/packages/mermaid/src/dagre-wrapper/clusters.js
    index 1ce7166890..5c6e5a4e05 100644
    --- a/packages/mermaid/src/dagre-wrapper/clusters.js
    +++ b/packages/mermaid/src/dagre-wrapper/clusters.js
    @@ -3,7 +3,7 @@ import { log } from '../logger.js';
     import createLabel from './createLabel.js';
     import { createText } from '../rendering-util/createText.js';
     import { select } from 'd3';
    -import { getConfig } from '../config.js';
    +import { getConfig } from '../diagram-api/diagramAPI.js';
     import { evaluate } from '../diagrams/common/common.js';
     
     const rect = (parent, node) => {
    diff --git a/packages/mermaid/src/dagre-wrapper/createLabel.js b/packages/mermaid/src/dagre-wrapper/createLabel.js
    index a8351c812f..4c947879cb 100644
    --- a/packages/mermaid/src/dagre-wrapper/createLabel.js
    +++ b/packages/mermaid/src/dagre-wrapper/createLabel.js
    @@ -1,8 +1,8 @@
     import { select } from 'd3';
     import { log } from '../logger.js';
    -import { getConfig } from '../config.js';
    +import { getConfig } from '../diagram-api/diagramAPI.js';
     import { evaluate } from '../diagrams/common/common.js';
    -import { decodeEntities } from '../mermaidAPI.js';
    +import { decodeEntities } from '../utils.js';
     
     /**
      * @param dom
    diff --git a/packages/mermaid/src/dagre-wrapper/edges.js b/packages/mermaid/src/dagre-wrapper/edges.js
    index 1b3e172c01..ced9a3bc2c 100644
    --- a/packages/mermaid/src/dagre-wrapper/edges.js
    +++ b/packages/mermaid/src/dagre-wrapper/edges.js
    @@ -2,7 +2,7 @@ import { log } from '../logger.js';
     import createLabel from './createLabel.js';
     import { createText } from '../rendering-util/createText.js';
     import { line, curveBasis, select } from 'd3';
    -import { getConfig } from '../config.js';
    +import { getConfig } from '../diagram-api/diagramAPI.js';
     import utils from '../utils.js';
     import { evaluate } from '../diagrams/common/common.js';
     import { getLineFunctionsWithOffset } from '../utils/lineWithOffset.js';
    diff --git a/packages/mermaid/src/dagre-wrapper/mermaid-graphlib.js b/packages/mermaid/src/dagre-wrapper/mermaid-graphlib.js
    index 1e376054dd..f42cc34633 100644
    --- a/packages/mermaid/src/dagre-wrapper/mermaid-graphlib.js
    +++ b/packages/mermaid/src/dagre-wrapper/mermaid-graphlib.js
    @@ -231,12 +231,12 @@ export const adjustClustersAndEdges = (graph, depth) => {
         if (children.length > 0) {
           log.debug('Cluster identified', id, descendants);
           edges.forEach((edge) => {
    -        // log.debug('Edge, decendants: ', edge, decendants[id]);
    +        // log.debug('Edge, descendants: ', edge, descendants[id]);
     
             // Check if any edge leaves the cluster (not the actual cluster, that's a link from the box)
             if (edge.v !== id && edge.w !== id) {
               // Any edge where either the one of the nodes is descending to the cluster but not the other
    -          // if (decendants[id].indexOf(edge.v) < 0 && decendants[id].indexOf(edge.w) < 0) {
    +          // if (descendants[id].indexOf(edge.v) < 0 && descendants[id].indexOf(edge.w) < 0) {
     
               const d1 = isDescendant(edge.v, id);
               const d2 = isDescendant(edge.w, id);
    diff --git a/packages/mermaid/src/dagre-wrapper/mermaid-graphlib.spec.js b/packages/mermaid/src/dagre-wrapper/mermaid-graphlib.spec.js
    index 1444a82c48..d44e543911 100644
    --- a/packages/mermaid/src/dagre-wrapper/mermaid-graphlib.spec.js
    +++ b/packages/mermaid/src/dagre-wrapper/mermaid-graphlib.spec.js
    @@ -419,7 +419,7 @@ describe('extractDescendants', function () {
           return {};
         });
       });
    -  it('Simple case of one level decendants GLB9', function () {
    +  it('Simple case of one level descendants GLB9', function () {
         /*
         subgraph A
           a
    diff --git a/packages/mermaid/src/dagre-wrapper/nodes.js b/packages/mermaid/src/dagre-wrapper/nodes.js
    index 51ff9ef116..325322798c 100644
    --- a/packages/mermaid/src/dagre-wrapper/nodes.js
    +++ b/packages/mermaid/src/dagre-wrapper/nodes.js
    @@ -1,7 +1,7 @@
     import { select } from 'd3';
     import { log } from '../logger.js';
     import { labelHelper, updateNodeBounds, insertPolygonShape } from './shapes/util.js';
    -import { getConfig } from '../config.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';
    diff --git a/packages/mermaid/src/dagre-wrapper/shapes/note.js b/packages/mermaid/src/dagre-wrapper/shapes/note.js
    index 17661e1696..514457cf02 100644
    --- a/packages/mermaid/src/dagre-wrapper/shapes/note.js
    +++ b/packages/mermaid/src/dagre-wrapper/shapes/note.js
    @@ -1,6 +1,6 @@
     import { updateNodeBounds, labelHelper } from './util.js';
     import { log } from '../../logger.js';
    -import { getConfig } from '../../config.js';
    +import { getConfig } from '../../diagram-api/diagramAPI.js';
     import intersect from '../intersect/index.js';
     
     const note = async (parent, node) => {
    diff --git a/packages/mermaid/src/dagre-wrapper/shapes/util.js b/packages/mermaid/src/dagre-wrapper/shapes/util.js
    index 95b82ddc07..079125e3a6 100644
    --- a/packages/mermaid/src/dagre-wrapper/shapes/util.js
    +++ b/packages/mermaid/src/dagre-wrapper/shapes/util.js
    @@ -1,9 +1,9 @@
     import createLabel from '../createLabel.js';
     import { createText } from '../../rendering-util/createText.js';
    -import { getConfig } from '../../config.js';
    -import { decodeEntities } from '../../mermaidAPI.js';
    +import { getConfig } from '../../diagram-api/diagramAPI.js';
     import { select } from 'd3';
     import { evaluate, sanitizeText } from '../../diagrams/common/common.js';
    +import { decodeEntities } from '../../utils.js';
     
     export const labelHelper = async (parent, node, _classes, isNode) => {
       let classes;
    diff --git a/packages/mermaid/src/defaultConfig.ts b/packages/mermaid/src/defaultConfig.ts
    index f8bd9b0b53..fb9db0c6a9 100644
    --- a/packages/mermaid/src/defaultConfig.ts
    +++ b/packages/mermaid/src/defaultConfig.ts
    @@ -236,6 +236,10 @@ const config: RequiredDeep = {
         ...defaultConfigJson.pie,
         useWidth: 984,
       },
    +  xyChart: {
    +    ...defaultConfigJson.xyChart,
    +    useWidth: undefined,
    +  },
       requirement: {
         ...defaultConfigJson.requirement,
         useWidth: undefined,
    diff --git a/packages/mermaid/src/diagram-api/diagram-orchestration.ts b/packages/mermaid/src/diagram-api/diagram-orchestration.ts
    index 624b0286ee..d6b9c00d8f 100644
    --- a/packages/mermaid/src/diagram-api/diagram-orchestration.ts
    +++ b/packages/mermaid/src/diagram-api/diagram-orchestration.ts
    @@ -7,6 +7,7 @@ import gantt from '../diagrams/gantt/ganttDetector.js';
     import { info } from '../diagrams/info/infoDetector.js';
     import { pie } from '../diagrams/pie/pieDetector.js';
     import quadrantChart from '../diagrams/quadrant-chart/quadrantDetector.js';
    +import xychart from '../diagrams/xychart/xychartDetector.js';
     import requirement from '../diagrams/requirement/requirementDetector.js';
     import sequence from '../diagrams/sequence/sequenceDetector.js';
     import classDiagram from '../diagrams/class/classDetector.js';
    @@ -84,6 +85,7 @@ export const addDiagrams = () => {
         state,
         journey,
         quadrantChart,
    -    sankey
    +    sankey,
    +    xychart
       );
     };
    diff --git a/packages/mermaid/src/diagram-api/diagramAPI.ts b/packages/mermaid/src/diagram-api/diagramAPI.ts
    index ea3c10159f..7ca9d58043 100644
    --- a/packages/mermaid/src/diagram-api/diagramAPI.ts
    +++ b/packages/mermaid/src/diagram-api/diagramAPI.ts
    @@ -1,6 +1,11 @@
     import { addDetector } from './detectType.js';
     import { log as _log, setLogLevel as _setLogLevel } from '../logger.js';
    -import { getConfig as _getConfig } from '../config.js';
    +import {
    +  getConfig as _getConfig,
    +  setConfig as _setConfig,
    +  defaultConfig as _defaultConfig,
    +  setSiteConfig as _setSiteConfig,
    +} from '../config.js';
     import { sanitizeText as _sanitizeText } from '../diagrams/common/common.js';
     import { setupGraphViewbox as _setupGraphViewbox } from '../setupGraphViewbox.js';
     import { addStylesForDiagram } from '../styles.js';
    @@ -15,6 +20,9 @@ import * as _commonDb from '../diagrams/common/commonDb.js';
     export const log = _log;
     export const setLogLevel = _setLogLevel;
     export const getConfig = _getConfig;
    +export const setConfig = _setConfig;
    +export const defaultConfig = _defaultConfig;
    +export const setSiteConfig = _setSiteConfig;
     export const sanitizeText = (text: string) => _sanitizeText(text, getConfig());
     export const setupGraphViewbox = _setupGraphViewbox;
     export const getCommonDb = () => {
    diff --git a/packages/mermaid/src/diagram.spec.ts b/packages/mermaid/src/diagram.spec.ts
    index b9c8ff60b8..d116399ac1 100644
    --- a/packages/mermaid/src/diagram.spec.ts
    +++ b/packages/mermaid/src/diagram.spec.ts
    @@ -66,4 +66,18 @@ Expecting 'TXT', got 'NEWLINE'"
           '"No diagram type detected matching given configuration for text: thor TD; A-->B"'
         );
       });
    +
    +  test('should consider entity codes when present in diagram defination', async () => {
    +    const diagram = await getDiagramFromText(`sequenceDiagram
    +    A->>B: I #9829; you!
    +    B->>A: I #9829; you #infin; times more!`);
    +    // @ts-ignore: we need to add types for sequenceDb which will be done in separate PR
    +    const messages = diagram.db?.getMessages?.();
    +    if (!messages) {
    +      throw new Error('Messages not found!');
    +    }
    +
    +    expect(messages[0].message).toBe('I fl°°9829¶ß you!');
    +    expect(messages[1].message).toBe('I fl°°9829¶ß you fl°infin¶ß times more!');
    +  });
     });
    diff --git a/packages/mermaid/src/diagrams/c4/c4Db.js b/packages/mermaid/src/diagrams/c4/c4Db.js
    index 71c1785853..3fc7e0afca 100644
    --- a/packages/mermaid/src/diagrams/c4/c4Db.js
    +++ b/packages/mermaid/src/diagrams/c4/c4Db.js
    @@ -1,4 +1,4 @@
    -import * as configApi from '../../config.js';
    +import { getConfig } from '../../diagram-api/diagramAPI.js';
     import { sanitizeText } from '../common/common.js';
     import {
       setAccTitle,
    @@ -33,7 +33,7 @@ export const getC4Type = function () {
     };
     
     export const setC4Type = function (c4TypeParam) {
    -  let sanitizedText = sanitizeText(c4TypeParam, configApi.getConfig());
    +  let sanitizedText = sanitizeText(c4TypeParam, getConfig());
       c4Type = sanitizedText;
     };
     
    @@ -783,7 +783,7 @@ export const PLACEMENT = {
     };
     
     export const setTitle = function (txt) {
    -  let sanitizedText = sanitizeText(txt, configApi.getConfig());
    +  let sanitizedText = sanitizeText(txt, getConfig());
       title = sanitizedText;
     };
     
    @@ -816,7 +816,7 @@ export default {
       getAccTitle,
       getAccDescription,
       setAccDescription,
    -  getConfig: () => configApi.getConfig().c4,
    +  getConfig: () => getConfig().c4,
       clear,
       LINETYPE,
       ARROWTYPE,
    diff --git a/packages/mermaid/src/diagrams/c4/c4Renderer.js b/packages/mermaid/src/diagrams/c4/c4Renderer.js
    index e60e58f21c..326d3060e8 100644
    --- a/packages/mermaid/src/diagrams/c4/c4Renderer.js
    +++ b/packages/mermaid/src/diagrams/c4/c4Renderer.js
    @@ -4,7 +4,7 @@ import { log } from '../../logger.js';
     import { parser } from './parser/c4Diagram.jison';
     import common from '../common/common.js';
     import c4Db from './c4Db.js';
    -import * as configApi from '../../config.js';
    +import { getConfig } from '../../diagram-api/diagramAPI.js';
     import assignWithDepth from '../../assignWithDepth.js';
     import { wrapLabel, calculateTextWidth, calculateTextHeight } from '../../utils.js';
     import { configureSvgSize } from '../../setupGraphViewbox.js';
    @@ -580,8 +580,8 @@ function drawInsideBoundary(
      * @param diagObj
      */
     export const draw = function (_text, id, _version, diagObj) {
    -  conf = configApi.getConfig().c4;
    -  const securityLevel = configApi.getConfig().securityLevel;
    +  conf = getConfig().c4;
    +  const securityLevel = getConfig().securityLevel;
       // Handle root and Document for when rendering in sandbox mode
       let sandboxElement;
       if (securityLevel === 'sandbox') {
    diff --git a/packages/mermaid/src/diagrams/class/classDb.ts b/packages/mermaid/src/diagrams/class/classDb.ts
    index 45ca1ed16c..312dda7a69 100644
    --- a/packages/mermaid/src/diagrams/class/classDb.ts
    +++ b/packages/mermaid/src/diagrams/class/classDb.ts
    @@ -1,7 +1,7 @@
     import type { Selection } from 'd3';
     import { select } from 'd3';
     import { log } from '../../logger.js';
    -import * as configApi from '../../config.js';
    +import { getConfig } from '../../diagram-api/diagramAPI.js';
     import common from '../common/common.js';
     import utils from '../../utils.js';
     import {
    @@ -34,10 +34,10 @@ let namespaceCounter = 0;
     
     let functions: any[] = [];
     
    -const sanitizeText = (txt: string) => common.sanitizeText(txt, configApi.getConfig());
    +const sanitizeText = (txt: string) => common.sanitizeText(txt, getConfig());
     
     const splitClassNameAndType = function (_id: string) {
    -  const id = common.sanitizeText(_id, configApi.getConfig());
    +  const id = common.sanitizeText(_id, getConfig());
       let genericType = '';
       let className = id;
     
    @@ -51,7 +51,7 @@ const splitClassNameAndType = function (_id: string) {
     };
     
     export const setClassLabel = function (_id: string, label: string) {
    -  const id = common.sanitizeText(_id, configApi.getConfig());
    +  const id = common.sanitizeText(_id, getConfig());
       if (label) {
         label = sanitizeText(label);
       }
    @@ -67,14 +67,14 @@ export const setClassLabel = function (_id: string, label: string) {
      * @public
      */
     export const addClass = function (_id: string) {
    -  const id = common.sanitizeText(_id, configApi.getConfig());
    +  const id = common.sanitizeText(_id, getConfig());
       const { className, type } = splitClassNameAndType(id);
       // Only add class if not exists
       if (Object.hasOwn(classes, className)) {
         return;
       }
       // alert('Adding class: ' + className);
    -  const name = common.sanitizeText(className, configApi.getConfig());
    +  const name = common.sanitizeText(className, getConfig());
       // alert('Adding class after: ' + name);
       classes[name] = {
         id: name,
    @@ -97,7 +97,7 @@ export const addClass = function (_id: string) {
      * @public
      */
     export const lookUpDomId = function (_id: string): string {
    -  const id = common.sanitizeText(_id, configApi.getConfig());
    +  const id = common.sanitizeText(_id, getConfig());
       if (id in classes) {
         return classes[id].domId;
       }
    @@ -139,15 +139,9 @@ export const addRelation = function (relation: ClassRelation) {
       relation.id1 = splitClassNameAndType(relation.id1).className;
       relation.id2 = splitClassNameAndType(relation.id2).className;
     
    -  relation.relationTitle1 = common.sanitizeText(
    -    relation.relationTitle1.trim(),
    -    configApi.getConfig()
    -  );
    +  relation.relationTitle1 = common.sanitizeText(relation.relationTitle1.trim(), getConfig());
     
    -  relation.relationTitle2 = common.sanitizeText(
    -    relation.relationTitle2.trim(),
    -    configApi.getConfig()
    -  );
    +  relation.relationTitle2 = common.sanitizeText(relation.relationTitle2.trim(), getConfig());
     
       relations.push(relation);
     };
    @@ -267,7 +261,7 @@ export const getTooltip = function (id: string, namespace?: string) {
      * @param target - Target of the link, _blank by default as originally defined in the svgDraw.js file
      */
     export const setLink = function (ids: string, linkStr: string, target: string) {
    -  const config = configApi.getConfig();
    +  const config = getConfig();
       ids.split(',').forEach(function (_id) {
         let id = _id;
         if (_id[0].match(/\d/)) {
    @@ -303,8 +297,8 @@ export const setClickEvent = function (ids: string, functionName: string, functi
     };
     
     const setClickFunc = function (_domId: string, functionName: string, functionArgs: string) {
    -  const domId = common.sanitizeText(_domId, configApi.getConfig());
    -  const config = configApi.getConfig();
    +  const domId = common.sanitizeText(_domId, getConfig());
    +  const config = getConfig();
       if (config.securityLevel !== 'loose') {
         return;
       }
    @@ -465,7 +459,7 @@ export default {
       getAccTitle,
       getAccDescription,
       setAccDescription,
    -  getConfig: () => configApi.getConfig().class,
    +  getConfig: () => getConfig().class,
       addClass,
       bindFunctions,
       clear,
    diff --git a/packages/mermaid/src/diagrams/class/classRenderer-v2.ts b/packages/mermaid/src/diagrams/class/classRenderer-v2.ts
    index 5abfd769a2..97106a169d 100644
    --- a/packages/mermaid/src/diagrams/class/classRenderer-v2.ts
    +++ b/packages/mermaid/src/diagrams/class/classRenderer-v2.ts
    @@ -2,7 +2,7 @@
     import { select, curveLinear } from 'd3';
     import * as graphlib from 'dagre-d3-es/src/graphlib/index.js';
     import { log } from '../../logger.js';
    -import { getConfig } from '../../config.js';
    +import { getConfig } from '../../diagram-api/diagramAPI.js';
     import { render } from '../../dagre-wrapper/index.js';
     import utils from '../../utils.js';
     import { interpolateToCurve, getStylesFromArray } from '../../utils.js';
    @@ -231,7 +231,7 @@ export const addRelations = function (relations: ClassRelation[], g: graphlib.Gr
           //Set relationship style and line type
           classes: 'relation',
           pattern: edge.relation.lineType == 1 ? 'dashed' : 'solid',
    -      id: 'id' + cnt,
    +      id: `id_${edge.id1}_${edge.id2}_${cnt}`,
           // Set link type for rendering
           arrowhead: edge.type === 'arrow_open' ? 'none' : 'normal',
           //Set edge extra labels
    diff --git a/packages/mermaid/src/diagrams/class/classRenderer.js b/packages/mermaid/src/diagrams/class/classRenderer.js
    index 58def16c2a..8c2dab7fb8 100644
    --- a/packages/mermaid/src/diagrams/class/classRenderer.js
    +++ b/packages/mermaid/src/diagrams/class/classRenderer.js
    @@ -4,7 +4,7 @@ import * as graphlib from 'dagre-d3-es/src/graphlib/index.js';
     import { log } from '../../logger.js';
     import svgDraw from './svgDraw.js';
     import { configureSvgSize } from '../../setupGraphViewbox.js';
    -import { getConfig } from '../../config.js';
    +import { getConfig } from '../../diagram-api/diagramAPI.js';
     
     let idCache = {};
     const padding = 20;
    diff --git a/packages/mermaid/src/diagrams/class/classTypes.ts b/packages/mermaid/src/diagrams/class/classTypes.ts
    index d372feebad..e288eefde8 100644
    --- a/packages/mermaid/src/diagrams/class/classTypes.ts
    +++ b/packages/mermaid/src/diagrams/class/classTypes.ts
    @@ -1,4 +1,4 @@
    -import { getConfig } from '../../config.js';
    +import { getConfig } from '../../diagram-api/diagramAPI.js';
     import { parseGenericTypes, sanitizeText } from '../common/common.js';
     
     export interface ClassNode {
    diff --git a/packages/mermaid/src/diagrams/class/styles.js b/packages/mermaid/src/diagrams/class/styles.js
    index f12f609f91..5f99a8b913 100644
    --- a/packages/mermaid/src/diagrams/class/styles.js
    +++ b/packages/mermaid/src/diagrams/class/styles.js
    @@ -1,7 +1,6 @@
     const getStyles = (options) =>
       `g.classGroup text {
    -  fill: ${options.nodeBorder};
    -  fill: ${options.classText};
    +  fill: ${options.nodeBorder || options.classText};
       stroke: none;
       font-family: ${options.fontFamily};
       font-size: 10px;
    diff --git a/packages/mermaid/src/diagrams/common/common.spec.ts b/packages/mermaid/src/diagrams/common/common.spec.ts
    index 4dac5b33c1..9af2444061 100644
    --- a/packages/mermaid/src/diagrams/common/common.spec.ts
    +++ b/packages/mermaid/src/diagrams/common/common.spec.ts
    @@ -38,6 +38,20 @@ describe('when securityLevel is antiscript, all script must be removed', () => {
         compareRemoveScript(``, ``);
       });
     
    +  it('should detect unsecured target attribute, if value is _blank then generate a secured link', () => {
    +    compareRemoveScript(
    +      `note about mermaid`,
    +      `note about mermaid`
    +    );
    +  });
    +
    +  it('should detect unsecured target attribute from links', () => {
    +    compareRemoveScript(
    +      `note about mermaid`,
    +      `note about mermaid`
    +    );
    +  });
    +
       it('should detect iframes', () => {
         compareRemoveScript(
           `
    diff --git a/packages/mermaid/src/diagrams/common/common.ts b/packages/mermaid/src/diagrams/common/common.ts
    index e0ca2929db..caf43bc682 100644
    --- a/packages/mermaid/src/diagrams/common/common.ts
    +++ b/packages/mermaid/src/diagrams/common/common.ts
    @@ -25,7 +25,27 @@ export const getRows = (s?: string): string[] => {
      * @returns The safer text
      */
     export const removeScript = (txt: string): string => {
    -  return DOMPurify.sanitize(txt);
    +  const TEMPORARY_ATTRIBUTE = 'data-temp-href-target';
    +
    +  DOMPurify.addHook('beforeSanitizeAttributes', (node: Element) => {
    +    if (node.tagName === 'A' && node.hasAttribute('target')) {
    +      node.setAttribute(TEMPORARY_ATTRIBUTE, node.getAttribute('target') || '');
    +    }
    +  });
    +
    +  const sanitizedText = DOMPurify.sanitize(txt);
    +
    +  DOMPurify.addHook('afterSanitizeAttributes', (node: Element) => {
    +    if (node.tagName === 'A' && node.hasAttribute(TEMPORARY_ATTRIBUTE)) {
    +      node.setAttribute('target', node.getAttribute(TEMPORARY_ATTRIBUTE) || '');
    +      node.removeAttribute(TEMPORARY_ATTRIBUTE);
    +      if (node.getAttribute('target') === '_blank') {
    +        node.setAttribute('rel', 'noopener');
    +      }
    +    }
    +  });
    +
    +  return sanitizedText;
     };
     
     const sanitizeMore = (text: string, config: MermaidConfig) => {
    diff --git a/packages/mermaid/src/diagrams/er/erDb.js b/packages/mermaid/src/diagrams/er/erDb.js
    index 9a397597ee..a58b9bbc1a 100644
    --- a/packages/mermaid/src/diagrams/er/erDb.js
    +++ b/packages/mermaid/src/diagrams/er/erDb.js
    @@ -1,5 +1,5 @@
     import { log } from '../../logger.js';
    -import * as configApi from '../../config.js';
    +import { getConfig } from '../../diagram-api/diagramAPI.js';
     
     import {
       setAccTitle,
    @@ -83,7 +83,7 @@ const clear = function () {
     export default {
       Cardinality,
       Identification,
    -  getConfig: () => configApi.getConfig().er,
    +  getConfig: () => getConfig().er,
       addEntity,
       addAttributes,
       getEntities,
    diff --git a/packages/mermaid/src/diagrams/er/erRenderer.js b/packages/mermaid/src/diagrams/er/erRenderer.js
    index 0c19d491b6..e8b25d50bb 100644
    --- a/packages/mermaid/src/diagrams/er/erRenderer.js
    +++ b/packages/mermaid/src/diagrams/er/erRenderer.js
    @@ -1,7 +1,7 @@
     import * as graphlib from 'dagre-d3-es/src/graphlib/index.js';
     import { line, curveBasis, select } from 'd3';
     import { layout as dagreLayout } from 'dagre-d3-es/src/dagre/index.js';
    -import { getConfig } from '../../config.js';
    +import { getConfig } from '../../diagram-api/diagramAPI.js';
     import { log } from '../../logger.js';
     import utils from '../../utils.js';
     import erMarkers from './erMarkers.js';
    @@ -202,7 +202,7 @@ const drawAttributes = (groupNode, entityTextNode, attributes) => {
         let attribStyle = 'attributeBoxOdd'; // We will flip the style on alternate rows to achieve a banded effect
     
         attributeNodes.forEach((attributeNode) => {
    -      // Calculate the alignment y co-ordinate for the type/name of the attribute
    +      // Calculate the alignment y coordinate for the type/name of the attribute
           const alignY = heightOffset + heightPadding + attributeNode.height / 2;
     
           // Position the type attribute
    @@ -579,8 +579,8 @@ export const draw = function (text, id, _version, diagObj) {
       // 2. Make sure they are all added to the graph
       // 3. Add all the edges (relationships) to the graph as well
       // 4. Let dagre do its magic to lay out the graph.  This assigns:
    -  //    - the centre co-ordinates for each node, bearing in mind the dimensions and edge relationships
    -  //    - the path co-ordinates for each edge
    +  //    - the centre coordinates for each node, bearing in mind the dimensions and edge relationships
    +  //    - the path coordinates for each edge
       //    But it has no impact on the svg child nodes - the diagram remains with every entity rooted at 0,0
       // 5. Now assign a transform to each entity in the svg node so that it gets drawn in the correct place, as determined by
       //    its centre point, which is obtained from the graph, and it's width and height
    diff --git a/packages/mermaid/src/diagrams/er/parser/erDiagram.spec.js b/packages/mermaid/src/diagrams/er/parser/erDiagram.spec.js
    index 825af737a6..ba29ff04b9 100644
    --- a/packages/mermaid/src/diagrams/er/parser/erDiagram.spec.js
    +++ b/packages/mermaid/src/diagrams/er/parser/erDiagram.spec.js
    @@ -92,7 +92,7 @@ describe('when parsing ER diagram it...', function () {
           });
         });
     
    -    it('cannot contain % because it interfers with parsing comments', function () {
    +    it('cannot contain % because it interferes with parsing comments', function () {
           expect(() => {
             erDiagram.parser.parse(`erDiagram\n "Blo%rf"\n`);
             const entities = erDb.getEntities();
    diff --git a/packages/mermaid/src/diagrams/flowchart/elk/flowRenderer-elk.js b/packages/mermaid/src/diagrams/flowchart/elk/flowRenderer-elk.js
    index 737b492fb3..e45739524e 100644
    --- a/packages/mermaid/src/diagrams/flowchart/elk/flowRenderer-elk.js
    +++ b/packages/mermaid/src/diagrams/flowchart/elk/flowRenderer-elk.js
    @@ -561,7 +561,7 @@ export const addEdges = function (edges, diagObj, graph, svg) {
     };
     
     // 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 cant use the dagre
    +// 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
    @@ -803,8 +803,14 @@ const insertChildren = (nodeArray, parentLookupDb) => {
      */
     
     export const draw = async function (text, id, _version, diagObj) {
    +  // Add temporary render element
    +  diagObj.db.clear();
       nodeDb = {};
       portPos = {};
    +  diagObj.db.setGen('gen-2');
    +  // Parse the graph definition
    +  diagObj.parser.parse(text);
    +
       const renderEl = select('body').append('div').attr('style', 'height:400px').attr('id', 'cy');
       let graph = {
         id: 'root',
    diff --git a/packages/mermaid/src/diagrams/flowchart/flowDb.js b/packages/mermaid/src/diagrams/flowchart/flowDb.js
    index a87bf558de..9a693aabf5 100644
    --- a/packages/mermaid/src/diagrams/flowchart/flowDb.js
    +++ b/packages/mermaid/src/diagrams/flowchart/flowDb.js
    @@ -1,6 +1,6 @@
     import { select } from 'd3';
     import utils from '../../utils.js';
    -import * as configApi from '../../config.js';
    +import { getConfig, defaultConfig } from '../../diagram-api/diagramAPI.js';
     import common from '../common/common.js';
     import { log } from '../../logger.js';
     import {
    @@ -12,10 +12,11 @@ import {
       setDiagramTitle,
       getDiagramTitle,
     } from '../common/commonDb.js';
    +import errorDiagram from '../error/errorDiagram.js';
     
     const MERMAID_DOM_ID_PREFIX = 'flowchart-';
     let vertexCounter = 0;
    -let config = configApi.getConfig();
    +let config = getConfig();
     let vertices = {};
     let edges = [];
     let classes = {};
    @@ -84,7 +85,7 @@ export const addVertex = function (_id, textObj, type, style, classes, dir, prop
       }
       vertexCounter++;
       if (textObj !== undefined) {
    -    config = configApi.getConfig();
    +    config = getConfig();
         txt = sanitizeText(textObj.text.trim());
         vertices[id].labelType = textObj.type;
         // strip quotes if string starts and ends with a quote
    @@ -156,7 +157,15 @@ export const addSingleLink = function (_start, _end, type) {
         edge.stroke = type.stroke;
         edge.length = type.length;
       }
    -  edges.push(edge);
    +  if (edge?.length > 10) {
    +    edge.length = 10;
    +  }
    +  if (edges.length < 280) {
    +    log.info('abc78 pushing edge...');
    +    edges.push(edge);
    +  } else {
    +    throw new Error('Too many edges');
    +  }
     };
     export const addLink = function (_start, _end, type) {
       log.info('addLink (abc78)', _start, _end, type);
    @@ -192,6 +201,13 @@ export const updateLinkInterpolate = function (positions, interp) {
      */
     export const updateLink = function (positions, style) {
       positions.forEach(function (pos) {
    +    if (pos >= edges.length) {
    +      throw new Error(
    +        `The index ${pos} for linkStyle is out of bounds. Valid indices for linkStyle are between 0 and ${
    +          edges.length - 1
    +        }. (Help: Ensure that the index is within the range of existing edges.)`
    +      );
    +    }
         if (pos === 'default') {
           edges.defaultStyle = style;
         } else {
    @@ -277,7 +293,7 @@ const setTooltip = function (ids, tooltip) {
     const setClickFun = function (id, functionName, functionArgs) {
       let domId = lookUpDomId(id);
       // if (_id[0].match(/\d/)) id = MERMAID_DOM_ID_PREFIX + id;
    -  if (configApi.getConfig().securityLevel !== 'loose') {
    +  if (getConfig().securityLevel !== 'loose') {
         return;
       }
       if (functionName === undefined) {
    @@ -416,7 +432,7 @@ const setupToolTips = function (element) {
           tooltipElem
             .text(el.attr('title'))
             .style('left', window.scrollX + rect.left + (rect.right - rect.left) / 2 + 'px')
    -        .style('top', window.scrollY + rect.top - 14 + document.body.scrollTop + 'px');
    +        .style('top', window.scrollY + rect.bottom + 'px');
           tooltipElem.html(tooltipElem.html().replace(/<br\/>/g, '
    ')); el.classed('hover', true); }) @@ -766,7 +782,7 @@ export const lex = { firstGraph, }; export default { - defaultConfig: () => configApi.defaultConfig.flowchart, + defaultConfig: () => defaultConfig.flowchart, setAccTitle, getAccTitle, getAccDescription, diff --git a/packages/mermaid/src/diagrams/flowchart/flowDiagram-v2.ts b/packages/mermaid/src/diagrams/flowchart/flowDiagram-v2.ts index c3de4b6854..368a98ccae 100644 --- a/packages/mermaid/src/diagrams/flowchart/flowDiagram-v2.ts +++ b/packages/mermaid/src/diagrams/flowchart/flowDiagram-v2.ts @@ -4,7 +4,7 @@ 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 '../../config.js'; +import { setConfig } from '../../diagram-api/diagramAPI.js'; export const diagram = { parser: flowParser, diff --git a/packages/mermaid/src/diagrams/flowchart/flowRenderer-v2.js b/packages/mermaid/src/diagrams/flowchart/flowRenderer-v2.js index 576ee6b34d..23d43da2b0 100644 --- a/packages/mermaid/src/diagrams/flowchart/flowRenderer-v2.js +++ b/packages/mermaid/src/diagrams/flowchart/flowRenderer-v2.js @@ -1,6 +1,6 @@ import * as graphlib from 'dagre-d3-es/src/graphlib/index.js'; import { select, curveLinear, selectAll } from 'd3'; -import { getConfig } from '../../config.js'; +import { getConfig } from '../../diagram-api/diagramAPI.js'; import utils from '../../utils.js'; import { render } from '../../dagre-wrapper/index.js'; import { addHtmlLabel } from 'dagre-d3-es/src/dagre-js/label/add-html-label.js'; diff --git a/packages/mermaid/src/diagrams/flowchart/flowRenderer.js b/packages/mermaid/src/diagrams/flowchart/flowRenderer.js index 8394b41e88..142e455563 100644 --- a/packages/mermaid/src/diagrams/flowchart/flowRenderer.js +++ b/packages/mermaid/src/diagrams/flowchart/flowRenderer.js @@ -1,6 +1,6 @@ import * as graphlib from 'dagre-d3-es/src/graphlib/index.js'; import { select, curveLinear, selectAll } from 'd3'; -import { getConfig } from '../../config.js'; +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'; diff --git a/packages/mermaid/src/diagrams/flowchart/flowRenderer.spec.js b/packages/mermaid/src/diagrams/flowchart/flowRenderer.spec.js index 0e9e8c0dea..5fb2307e53 100644 --- a/packages/mermaid/src/diagrams/flowchart/flowRenderer.spec.js +++ b/packages/mermaid/src/diagrams/flowchart/flowRenderer.spec.js @@ -1,5 +1,5 @@ import { addVertices, addEdges } from './flowRenderer.js'; -import { setConfig } from '../../config.js'; +import { setConfig } from '../../diagram-api/diagramAPI.js'; setConfig({ flowchart: { diff --git a/packages/mermaid/src/diagrams/flowchart/parser/flow-style.spec.js b/packages/mermaid/src/diagrams/flowchart/parser/flow-style.spec.js index 1ab7543085..5b0f740bd9 100644 --- a/packages/mermaid/src/diagrams/flowchart/parser/flow-style.spec.js +++ b/packages/mermaid/src/diagrams/flowchart/parser/flow-style.spec.js @@ -286,6 +286,30 @@ describe('[Style] when parsing', () => { expect(edges[0].type).toBe('arrow_point'); }); + it('should handle style definitions within number of edges', function () { + expect(() => + flow.parser + .parse( + `graph TD + A-->B + linkStyle 1 stroke-width:1px;` + ) + .toThrow( + 'The index 1 for linkStyle is out of bounds. Valid indices for linkStyle are between 0 and 0. (Help: Ensure that the index is within the range of existing edges.)' + ) + ); + }); + + it('should handle style definitions within number of edges', function () { + const res = flow.parser.parse(`graph TD + A-->B + linkStyle 0 stroke-width:1px;`); + + const edges = flow.parser.yy.getEdges(); + + expect(edges[0].style[0]).toBe('stroke-width:1px'); + }); + it('should handle multi-numbered style definitions with more then 1 digit in a row', function () { const res = flow.parser.parse( 'graph TD\n' + diff --git a/packages/mermaid/src/diagrams/flowchart/parser/flow-text.spec.js b/packages/mermaid/src/diagrams/flowchart/parser/flow-text.spec.js index b127e1b65d..61eccbbc8e 100644 --- a/packages/mermaid/src/diagrams/flowchart/parser/flow-text.spec.js +++ b/packages/mermaid/src/diagrams/flowchart/parser/flow-text.spec.js @@ -535,6 +535,10 @@ describe('[Text] when parsing', () => { expect(vert['A'].text).toBe('this is an ellipse'); }); + it('should not freeze when ellipse text has a `(`', function () { + expect(() => flow.parser.parse('graph\nX(- My Text (')).toThrowError(); + }); + it('should handle text in diamond vertices with space', function () { const res = flow.parser.parse('graph TD;A(chimpansen hoppar)-->C;'); diff --git a/packages/mermaid/src/diagrams/flowchart/parser/flow.jison b/packages/mermaid/src/diagrams/flowchart/parser/flow.jison index 6dad36d25f..de23d93cb9 100644 --- a/packages/mermaid/src/diagrams/flowchart/parser/flow.jison +++ b/packages/mermaid/src/diagrams/flowchart/parser/flow.jison @@ -134,7 +134,7 @@ that id. <*>\s*\~\~[\~]+\s* return 'LINK'; [-/\)][\)] { this.popState(); return '-)'; } -[^\(\)\[\]\{\}]|-/!\)+ return "TEXT" +[^\(\)\[\]\{\}]|-\!\)+ return "TEXT" <*>"(-" { this.pushState("ellipseText"); return '(-'; } "])" { this.popState(); return 'STADIUMEND'; } diff --git a/packages/mermaid/src/diagrams/gantt/ganttDb.js b/packages/mermaid/src/diagrams/gantt/ganttDb.js index 775494e3df..1c73a13ea9 100644 --- a/packages/mermaid/src/diagrams/gantt/ganttDb.js +++ b/packages/mermaid/src/diagrams/gantt/ganttDb.js @@ -4,7 +4,7 @@ import dayjsIsoWeek from 'dayjs/plugin/isoWeek.js'; import dayjsCustomParseFormat from 'dayjs/plugin/customParseFormat.js'; import dayjsAdvancedFormat from 'dayjs/plugin/advancedFormat.js'; import { log } from '../../logger.js'; -import * as configApi from '../../config.js'; +import { getConfig } from '../../diagram-api/diagramAPI.js'; import utils from '../../utils.js'; import { @@ -603,7 +603,7 @@ const compileTasks = function () { */ export const setLink = function (ids, _linkStr) { let linkStr = _linkStr; - if (configApi.getConfig().securityLevel !== 'loose') { + if (getConfig().securityLevel !== 'loose') { linkStr = sanitizeUrl(_linkStr); } ids.split(',').forEach(function (id) { @@ -634,7 +634,7 @@ export const setClass = function (ids, className) { }; const setClickFun = function (id, functionName, functionArgs) { - if (configApi.getConfig().securityLevel !== 'loose') { + if (getConfig().securityLevel !== 'loose') { return; } if (functionName === undefined) { @@ -725,7 +725,7 @@ export const bindFunctions = function (element) { }; export default { - getConfig: () => configApi.getConfig().gantt, + getConfig: () => getConfig().gantt, clear, setDateFormat, getDateFormat, diff --git a/packages/mermaid/src/diagrams/gantt/ganttRenderer.js b/packages/mermaid/src/diagrams/gantt/ganttRenderer.js index 55b5607a28..33dbaf9efd 100644 --- a/packages/mermaid/src/diagrams/gantt/ganttRenderer.js +++ b/packages/mermaid/src/diagrams/gantt/ganttRenderer.js @@ -25,7 +25,7 @@ import { timeMonth, } from 'd3'; import common from '../common/common.js'; -import { getConfig } from '../../config.js'; +import { getConfig } from '../../diagram-api/diagramAPI.js'; import { configureSvgSize } from '../../setupGraphViewbox.js'; export const setConf = function () { diff --git a/packages/mermaid/src/diagrams/gantt/styles.js b/packages/mermaid/src/diagrams/gantt/styles.js index 8193130bbc..626ed4e0f1 100644 --- a/packages/mermaid/src/diagrams/gantt/styles.js +++ b/packages/mermaid/src/diagrams/gantt/styles.js @@ -1,9 +1,9 @@ const getStyles = (options) => ` .mermaid-main-font { - font-family: "trebuchet ms", verdana, arial, sans-serif; - font-family: var(--mermaid-font-family); + font-family: var(--mermaid-font-family, "trebuchet ms", verdana, arial, sans-serif); } + .exclude-range { fill: ${options.excludeBkgColor}; } @@ -45,11 +45,7 @@ const getStyles = (options) => .sectionTitle { text-anchor: start; - // font-size: ${options.ganttFontSize}; - // text-height: 14px; - font-family: 'trebuchet ms', verdana, arial, sans-serif; - font-family: var(--mermaid-font-family); - + font-family: var(--mermaid-font-family, "trebuchet ms", verdana, arial, sans-serif); } @@ -59,10 +55,11 @@ const getStyles = (options) => stroke: ${options.gridColor}; opacity: 0.8; shape-rendering: crispEdges; - text { - font-family: ${options.fontFamily}; - fill: ${options.textColor}; - } + } + + .grid .tick text { + font-family: ${options.fontFamily}; + fill: ${options.textColor}; } .grid path { @@ -89,33 +86,27 @@ const getStyles = (options) => .taskText { text-anchor: middle; - font-family: 'trebuchet ms', verdana, arial, sans-serif; - font-family: var(--mermaid-font-family); + font-family: var(--mermaid-font-family, "trebuchet ms", verdana, arial, sans-serif); } - // .taskText:not([font-size]) { - // font-size: ${options.ganttFontSize}; - // } - .taskTextOutsideRight { fill: ${options.taskTextDarkColor}; text-anchor: start; - // font-size: ${options.ganttFontSize}; - font-family: 'trebuchet ms', verdana, arial, sans-serif; - font-family: var(--mermaid-font-family); - + font-family: var(--mermaid-font-family, "trebuchet ms", verdana, arial, sans-serif); } .taskTextOutsideLeft { fill: ${options.taskTextDarkColor}; text-anchor: end; - // font-size: ${options.ganttFontSize}; } + /* Special case clickable */ + .task.clickable { cursor: pointer; } + .taskText.clickable { cursor: pointer; fill: ${options.taskTextClickableColor} !important; @@ -134,6 +125,7 @@ const getStyles = (options) => font-weight: bold; } + /* Specific task settings for the sections*/ .taskText0, @@ -255,9 +247,8 @@ const getStyles = (options) => .titleText { text-anchor: middle; font-size: 18px; - fill: ${options.textColor} ; - font-family: 'trebuchet ms', verdana, arial, sans-serif; - font-family: var(--mermaid-font-family); + fill: ${options.titleColor || options.textColor}; + font-family: var(--mermaid-font-family, "trebuchet ms", verdana, arial, sans-serif); } `; diff --git a/packages/mermaid/src/diagrams/git/gitGraphAst.js b/packages/mermaid/src/diagrams/git/gitGraphAst.js index abad68b22b..f8cfba6f51 100644 --- a/packages/mermaid/src/diagrams/git/gitGraphAst.js +++ b/packages/mermaid/src/diagrams/git/gitGraphAst.js @@ -1,7 +1,6 @@ import { log } from '../../logger.js'; import { random } from '../../utils.js'; -import * as configApi from '../../config.js'; -import { getConfig } from '../../config.js'; +import { getConfig } from '../../diagram-api/diagramAPI.js'; import common from '../common/common.js'; import { setAccTitle, @@ -106,9 +105,9 @@ export const getOptions = function () { export const commit = function (msg, id, type, tag) { log.debug('Entering commit:', msg, id, type, tag); - id = common.sanitizeText(id, configApi.getConfig()); - msg = common.sanitizeText(msg, configApi.getConfig()); - tag = common.sanitizeText(tag, configApi.getConfig()); + id = common.sanitizeText(id, getConfig()); + msg = common.sanitizeText(msg, getConfig()); + tag = common.sanitizeText(tag, getConfig()); const commit = { id: id ? id : seq + '-' + getId(), message: msg, @@ -125,7 +124,7 @@ export const commit = function (msg, id, type, tag) { }; export const branch = function (name, order) { - name = common.sanitizeText(name, configApi.getConfig()); + name = common.sanitizeText(name, getConfig()); if (branches[name] === undefined) { branches[name] = head != null ? head.id : null; branchesConfig[name] = { name, order: order ? parseInt(order, 10) : null }; @@ -149,8 +148,8 @@ export const branch = function (name, order) { }; export const merge = function (otherBranch, custom_id, override_type, custom_tag) { - otherBranch = common.sanitizeText(otherBranch, configApi.getConfig()); - custom_id = common.sanitizeText(custom_id, configApi.getConfig()); + otherBranch = common.sanitizeText(otherBranch, getConfig()); + custom_id = common.sanitizeText(custom_id, getConfig()); const currentCommit = commits[branches[curBranch]]; const otherCommit = commits[branches[otherBranch]]; @@ -258,9 +257,9 @@ export const merge = function (otherBranch, custom_id, override_type, custom_tag export const cherryPick = function (sourceId, targetId, tag) { log.debug('Entering cherryPick:', sourceId, targetId, tag); - sourceId = common.sanitizeText(sourceId, configApi.getConfig()); - targetId = common.sanitizeText(targetId, configApi.getConfig()); - tag = common.sanitizeText(tag, configApi.getConfig()); + sourceId = common.sanitizeText(sourceId, getConfig()); + targetId = common.sanitizeText(targetId, getConfig()); + tag = common.sanitizeText(tag, getConfig()); if (!sourceId || commits[sourceId] === undefined) { let error = new Error( @@ -338,7 +337,7 @@ export const cherryPick = function (sourceId, targetId, tag) { } }; export const checkout = function (branch) { - branch = common.sanitizeText(branch, configApi.getConfig()); + branch = common.sanitizeText(branch, getConfig()); if (branches[branch] === undefined) { let error = new Error( 'Trying to checkout branch which is not yet created. (Help try using "branch ' + branch + '")' @@ -502,7 +501,7 @@ export const commitType = { }; export default { - getConfig: () => configApi.getConfig().gitGraph, + getConfig: () => getConfig().gitGraph, setDirection, setOptions, getOptions, diff --git a/packages/mermaid/src/diagrams/git/gitGraphParserV2.spec.js b/packages/mermaid/src/diagrams/git/gitGraphParserV2.spec.js index df20a5eb5a..540ab773bf 100644 --- a/packages/mermaid/src/diagrams/git/gitGraphParserV2.spec.js +++ b/packages/mermaid/src/diagrams/git/gitGraphParserV2.spec.js @@ -6,7 +6,7 @@ describe('when parsing a gitGraph', function () { parser.yy = gitGraphAst; parser.yy.clear(); }); - it('should handle a gitGraph commit with NO pararms, get auto-genrated reandom ID', function () { + it('should handle a gitGraph commit with NO pararms, get auto-generated reandom ID', function () { const str = `gitGraph: commit `; diff --git a/packages/mermaid/src/diagrams/git/layout.js b/packages/mermaid/src/diagrams/git/layout.js index 0dbe577659..2a782a0798 100644 --- a/packages/mermaid/src/diagrams/git/layout.js +++ b/packages/mermaid/src/diagrams/git/layout.js @@ -1,4 +1,4 @@ -import { getConfig } from '../../config.js'; +import { getConfig } from '../../diagram-api/diagramAPI.js'; export default (dir, _branches) => { const config = getConfig().gitGraph; diff --git a/packages/mermaid/src/diagrams/mindmap/mindmap.spec.js b/packages/mermaid/src/diagrams/mindmap/mindmap.spec.js index 845205f9b3..c0b72060d9 100644 --- a/packages/mermaid/src/diagrams/mindmap/mindmap.spec.js +++ b/packages/mermaid/src/diagrams/mindmap/mindmap.spec.js @@ -129,7 +129,7 @@ root expect(child.nodeId).toEqual('theId'); expect(child.type).toEqual(mindmap.yy.nodeType.ROUNDED_RECT); }); - it('MMP-10 mutiple types (circle)', function () { + it('MMP-10 multiple types (circle)', function () { let str = `mindmap root((the root)) `; @@ -141,7 +141,7 @@ root expect(mm.type).toEqual(mindmap.yy.nodeType.CIRCLE); }); - it('MMP-11 mutiple types (cloud)', function () { + it('MMP-11 multiple types (cloud)', function () { let str = `mindmap root)the root( `; @@ -152,7 +152,7 @@ root expect(mm.children.length).toEqual(0); expect(mm.type).toEqual(mindmap.yy.nodeType.CLOUD); }); - it('MMP-12 mutiple types (bang)', function () { + it('MMP-12 multiple types (bang)', function () { let str = `mindmap root))the root(( `; @@ -164,7 +164,7 @@ root expect(mm.type).toEqual(mindmap.yy.nodeType.BANG); }); - it('MMP-12-a mutiple types (hexagon)', function () { + it('MMP-12-a multiple types (hexagon)', function () { let str = `mindmap root{{the root}} `; diff --git a/packages/mermaid/src/diagrams/mindmap/mindmapDb.js b/packages/mermaid/src/diagrams/mindmap/mindmapDb.js index 9413581d64..4206a4a260 100644 --- a/packages/mermaid/src/diagrams/mindmap/mindmapDb.js +++ b/packages/mermaid/src/diagrams/mindmap/mindmapDb.js @@ -1,4 +1,4 @@ -import { getConfig } from '../../config.js'; +import { getConfig } from '../../diagram-api/diagramAPI.js'; import { sanitizeText as _sanitizeText } from '../../diagrams/common/common.js'; import { log } from '../../logger.js'; diff --git a/packages/mermaid/src/diagrams/mindmap/mindmapRenderer.js b/packages/mermaid/src/diagrams/mindmap/mindmapRenderer.js index 7e741657b9..3fe9e1d510 100644 --- a/packages/mermaid/src/diagrams/mindmap/mindmapRenderer.js +++ b/packages/mermaid/src/diagrams/mindmap/mindmapRenderer.js @@ -1,7 +1,7 @@ /** Created by knut on 14-12-11. */ import { select } from 'd3'; import { log } from '../../logger.js'; -import { getConfig } from '../../config.js'; +import { getConfig } from '../../diagram-api/diagramAPI.js'; import { setupGraphViewbox } from '../../setupGraphViewbox.js'; import svgDraw from './svgDraw.js'; import cytoscape from 'cytoscape/dist/cytoscape.umd.js'; diff --git a/packages/mermaid/src/diagrams/pie/pie.spec.ts b/packages/mermaid/src/diagrams/pie/pie.spec.ts index 81184f1b1c..fb0cbeec8f 100644 --- a/packages/mermaid/src/diagrams/pie/pie.spec.ts +++ b/packages/mermaid/src/diagrams/pie/pie.spec.ts @@ -1,6 +1,6 @@ import { parser } from './pieParser.js'; import { DEFAULT_PIE_DB, db } from './pieDb.js'; -import { setConfig } from '../../config.js'; +import { setConfig } from '../../diagram-api/diagramAPI.js'; setConfig({ securityLevel: 'strict', diff --git a/packages/mermaid/src/diagrams/pie/pieRenderer.ts b/packages/mermaid/src/diagrams/pie/pieRenderer.ts index 27dac5be91..d8e204a4ae 100644 --- a/packages/mermaid/src/diagrams/pie/pieRenderer.ts +++ b/packages/mermaid/src/diagrams/pie/pieRenderer.ts @@ -3,7 +3,7 @@ import { scaleOrdinal, pie as d3pie, arc } from 'd3'; import { log } from '../../logger.js'; import { configureSvgSize } from '../../setupGraphViewbox.js'; -import { getConfig } from '../../config.js'; +import { getConfig } from '../../diagram-api/diagramAPI.js'; import { cleanAndMerge, parseFontSize } from '../../utils.js'; import type { DrawDefinition, Group, SVG } from '../../diagram-api/types.js'; import type { D3Section, PieDB, Sections } from './pieTypes.js'; diff --git a/packages/mermaid/src/diagrams/quadrant-chart/quadrantBuilder.ts b/packages/mermaid/src/diagrams/quadrant-chart/quadrantBuilder.ts index 75d6705381..380a79f190 100644 --- a/packages/mermaid/src/diagrams/quadrant-chart/quadrantBuilder.ts +++ b/packages/mermaid/src/diagrams/quadrant-chart/quadrantBuilder.ts @@ -1,4 +1,3 @@ -// @ts-ignore: TODO Fix ts errors import { scaleLinear } from 'd3'; import { log } from '../../logger.js'; import type { BaseDiagramConfig, QuadrantChartConfig } from '../../config.type.js'; diff --git a/packages/mermaid/src/diagrams/quadrant-chart/quadrantDb.ts b/packages/mermaid/src/diagrams/quadrant-chart/quadrantDb.ts index 0dad6dfdd5..c3a79c911e 100644 --- a/packages/mermaid/src/diagrams/quadrant-chart/quadrantDb.ts +++ b/packages/mermaid/src/diagrams/quadrant-chart/quadrantDb.ts @@ -1,4 +1,4 @@ -import * as configApi from '../../config.js'; +import { getConfig } from '../../diagram-api/diagramAPI.js'; import { sanitizeText } from '../common/common.js'; import { setAccTitle, @@ -11,7 +11,7 @@ import { } from '../common/commonDb.js'; import { QuadrantBuilder } from './quadrantBuilder.js'; -const config = configApi.getConfig(); +const config = getConfig(); function textSanitizer(text: string) { return sanitizeText(text.trim(), config); @@ -66,7 +66,7 @@ function setHeight(height: number) { } function getQuadrantData() { - const config = configApi.getConfig(); + const config = getConfig(); const { themeVariables, quadrantChart: quadrantChartConfig } = config; if (quadrantChartConfig) { quadrantBuilder.setConfig(quadrantChartConfig); diff --git a/packages/mermaid/src/diagrams/quadrant-chart/quadrantRenderer.ts b/packages/mermaid/src/diagrams/quadrant-chart/quadrantRenderer.ts index 9dd309b533..d272dccd4a 100644 --- a/packages/mermaid/src/diagrams/quadrant-chart/quadrantRenderer.ts +++ b/packages/mermaid/src/diagrams/quadrant-chart/quadrantRenderer.ts @@ -1,6 +1,6 @@ // @ts-nocheck - don't check until handle it import { select } from 'd3'; -import * as configApi from '../../config.js'; +import { getConfig } from '../../diagram-api/diagramAPI.js'; import { log } from '../../logger.js'; import { configureSvgSize } from '../../setupGraphViewbox.js'; import type { Diagram } from '../../Diagram.js'; @@ -27,7 +27,7 @@ export const draw = (txt: string, id: string, _version: string, diagObj: Diagram return `translate(${data.x}, ${data.y}) rotate(${data.rotation || 0})`; } - const conf = configApi.getConfig(); + const conf = getConfig(); log.debug('Rendering quadrant chart\n' + txt); diff --git a/packages/mermaid/src/diagrams/requirement/requirementDb.js b/packages/mermaid/src/diagrams/requirement/requirementDb.js index 325e95ee20..9357e2a660 100644 --- a/packages/mermaid/src/diagrams/requirement/requirementDb.js +++ b/packages/mermaid/src/diagrams/requirement/requirementDb.js @@ -1,4 +1,4 @@ -import * as configApi from '../../config.js'; +import { getConfig } from '../../diagram-api/diagramAPI.js'; import { log } from '../../logger.js'; import { @@ -144,7 +144,7 @@ export default { VerifyType, Relationships, - getConfig: () => configApi.getConfig().req, + getConfig: () => getConfig().req, addRequirement, getRequirements, diff --git a/packages/mermaid/src/diagrams/requirement/requirementRenderer.js b/packages/mermaid/src/diagrams/requirement/requirementRenderer.js index 49b7828651..2af2067ad8 100644 --- a/packages/mermaid/src/diagrams/requirement/requirementRenderer.js +++ b/packages/mermaid/src/diagrams/requirement/requirementRenderer.js @@ -5,7 +5,7 @@ import { log } from '../../logger.js'; import { configureSvgSize } from '../../setupGraphViewbox.js'; import common from '../common/common.js'; import markers from './requirementMarkers.js'; -import { getConfig } from '../../config.js'; +import { getConfig } from '../../diagram-api/diagramAPI.js'; let conf = {}; let relCnt = 0; diff --git a/packages/mermaid/src/diagrams/sankey/sankeyDB.ts b/packages/mermaid/src/diagrams/sankey/sankeyDB.ts index 8b3a22c5a0..d6fd90373c 100644 --- a/packages/mermaid/src/diagrams/sankey/sankeyDB.ts +++ b/packages/mermaid/src/diagrams/sankey/sankeyDB.ts @@ -1,4 +1,4 @@ -import * as configApi from '../../config.js'; +import { getConfig } from '../../diagram-api/diagramAPI.js'; import common from '../common/common.js'; import { setAccTitle, @@ -42,7 +42,7 @@ class SankeyNode { } const findOrCreateNode = (ID: string): SankeyNode => { - ID = common.sanitizeText(ID, configApi.getConfig()); + ID = common.sanitizeText(ID, getConfig()); if (!nodesMap[ID]) { nodesMap[ID] = new SankeyNode(ID); @@ -65,7 +65,7 @@ const getGraph = () => ({ export default { nodesMap, - getConfig: () => configApi.getConfig().sankey, + getConfig: () => getConfig().sankey, getNodes, getLinks, getGraph, diff --git a/packages/mermaid/src/diagrams/sankey/sankeyRenderer.ts b/packages/mermaid/src/diagrams/sankey/sankeyRenderer.ts index 9f5b3c1720..0179e715b7 100644 --- a/packages/mermaid/src/diagrams/sankey/sankeyRenderer.ts +++ b/packages/mermaid/src/diagrams/sankey/sankeyRenderer.ts @@ -1,5 +1,5 @@ import type { Diagram } from '../../Diagram.js'; -import * as configApi from '../../config.js'; +import { getConfig, defaultConfig } from '../../diagram-api/diagramAPI.js'; import { select as d3select, @@ -41,8 +41,8 @@ const alignmentsMap: Record< */ export const draw = function (text: string, id: string, _version: string, diagObj: Diagram): void { // Get Sankey config - const { securityLevel, sankey: conf } = configApi.getConfig(); - const defaultSankeyConfig = configApi!.defaultConfig!.sankey!; + const { securityLevel, sankey: conf } = getConfig(); + const defaultSankeyConfig = defaultConfig!.sankey!; // TODO: // This code repeats for every diagram diff --git a/packages/mermaid/src/diagrams/sequence/sequenceDb.js b/packages/mermaid/src/diagrams/sequence/sequenceDb.js index 6c3f1f64df..717cc8c09e 100644 --- a/packages/mermaid/src/diagrams/sequence/sequenceDb.js +++ b/packages/mermaid/src/diagrams/sequence/sequenceDb.js @@ -1,4 +1,4 @@ -import * as configApi from '../../config.js'; +import { getConfig } from '../../diagram-api/diagramAPI.js'; import { log } from '../../logger.js'; import { sanitizeText } from '../common/common.js'; import { @@ -196,7 +196,7 @@ export const autoWrap = () => { if (wrapEnabled !== undefined) { return wrapEnabled; } - return configApi.getConfig().sequence.wrap; + return getConfig().sequence.wrap; }; export const clear = function () { @@ -251,7 +251,7 @@ export const parseBoxData = function (str) { color: color, text: title !== undefined - ? sanitizeText(title.replace(/^:?(?:no)?wrap:/, ''), configApi.getConfig()) + ? sanitizeText(title.replace(/^:?(?:no)?wrap:/, ''), getConfig()) : undefined, wrap: title !== undefined @@ -337,7 +337,7 @@ export const addLinks = function (actorId, text) { const actor = getActor(actorId); // JSON.parse the text try { - let sanitizedText = sanitizeText(text.text, configApi.getConfig()); + let sanitizedText = sanitizeText(text.text, getConfig()); sanitizedText = sanitizedText.replace(/&/g, '&'); sanitizedText = sanitizedText.replace(/=/g, '='); const links = JSON.parse(sanitizedText); @@ -353,7 +353,7 @@ export const addALink = function (actorId, text) { const actor = getActor(actorId); try { const links = {}; - let sanitizedText = sanitizeText(text.text, configApi.getConfig()); + let sanitizedText = sanitizeText(text.text, getConfig()); var sep = sanitizedText.indexOf('@'); sanitizedText = sanitizedText.replace(/&/g, '&'); sanitizedText = sanitizedText.replace(/=/g, '='); @@ -387,7 +387,7 @@ export const addProperties = function (actorId, text) { const actor = getActor(actorId); // JSON.parse the text try { - let sanitizedText = sanitizeText(text.text, configApi.getConfig()); + let sanitizedText = sanitizeText(text.text, getConfig()); const properties = JSON.parse(sanitizedText); // add the deserialized text to the actor's property field. insertProperties(actor, properties); @@ -629,7 +629,7 @@ export default { getBoxes, getDiagramTitle, setDiagramTitle, - getConfig: () => configApi.getConfig().sequence, + getConfig: () => getConfig().sequence, clear, parseMessage, parseBoxData, diff --git a/packages/mermaid/src/diagrams/sequence/sequenceDiagram.spec.js b/packages/mermaid/src/diagrams/sequence/sequenceDiagram.spec.js index 77ac7c45cd..8a7e2281cb 100644 --- a/packages/mermaid/src/diagrams/sequence/sequenceDiagram.spec.js +++ b/packages/mermaid/src/diagrams/sequence/sequenceDiagram.spec.js @@ -1,5 +1,5 @@ import { vi } from 'vitest'; -import * as configApi from '../../config.js'; +import { setSiteConfig } from '../../diagram-api/diagramAPI.js'; import mermaidAPI from '../../mermaidAPI.js'; import { Diagram, getDiagramFromText } from '../../Diagram.js'; import { addDiagrams } from '../../diagram-api/diagram-orchestration.js'; @@ -1610,7 +1610,7 @@ describe('when rendering a sequenceDiagram APA', function () { wrap: false, mirrorActors: false, }; - configApi.setSiteConfig({ logLevel: 5, sequence: conf }); + setSiteConfig({ logLevel: 5, sequence: conf }); }); let conf; beforeEach(function () { @@ -1631,7 +1631,7 @@ describe('when rendering a sequenceDiagram APA', function () { wrap: false, mirrorActors: false, }; - configApi.setSiteConfig({ logLevel: 5, sequence: conf }); + setSiteConfig({ logLevel: 5, sequence: conf }); diagram = new Diagram(` sequenceDiagram Alice->Bob:Hello Bob, how are you? diff --git a/packages/mermaid/src/diagrams/sequence/sequenceRenderer.ts b/packages/mermaid/src/diagrams/sequence/sequenceRenderer.ts index a41c3877fe..7c38a80163 100644 --- a/packages/mermaid/src/diagrams/sequence/sequenceRenderer.ts +++ b/packages/mermaid/src/diagrams/sequence/sequenceRenderer.ts @@ -4,7 +4,7 @@ import svgDraw, { ACTOR_TYPE_WIDTH, drawText, fixLifeLineHeights } from './svgDr import { log } from '../../logger.js'; import common from '../common/common.js'; import * as svgDrawCommon from '../common/svgDrawCommon.js'; -import * as configApi from '../../config.js'; +import { getConfig } from '../../diagram-api/diagramAPI.js'; import assignWithDepth from '../../assignWithDepth.js'; import utils from '../../utils.js'; import { configureSvgSize } from '../../setupGraphViewbox.js'; @@ -91,7 +91,7 @@ export const bounds = { stopy: undefined, }; this.verticalPos = 0; - setConf(configApi.getConfig()); + setConf(getConfig()); }, updateVal: function (obj, key, val, fun) { if (obj[key] === undefined) { @@ -747,7 +747,7 @@ function adjustCreatedDestroyedData( * @param diagObj - A standard diagram containing the db and the text and type etc of the diagram */ export const draw = function (_text: string, id: string, _version: string, diagObj: Diagram) { - const { securityLevel, sequence } = configApi.getConfig(); + const { securityLevel, sequence } = getConfig(); conf = sequence; // Handle root and Document for when rendering in sandbox mode let sandboxElement; @@ -829,6 +829,11 @@ export const draw = function (_text: string, id: string, _version: string, diagO bounds.insert(activationData.startx, verticalPos - 10, activationData.stopx, verticalPos); } + log.debug('createdActors', createdActors); + log.debug('destroyedActors', destroyedActors); + + drawActors(diagram, actors, actorKeys, false); + // Draw the messages/signals let sequenceIndex = 1; let sequenceIndexStep = 1; @@ -1028,14 +1033,12 @@ export const draw = function (_text: string, id: string, _version: string, diagO } }); - log.debug('createdActors', createdActors); - log.debug('destroyedActors', destroyedActors); - - drawActors(diagram, actors, actorKeys, false); messagesToDraw.forEach((e) => drawMessage(diagram, e.messageModel, e.lineStartY, diagObj)); + if (conf.mirrorActors) { drawActors(diagram, actors, actorKeys, true); } + backgrounds.forEach((e) => svgDraw.drawBackgroundRect(diagram, e)); fixLifeLineHeights(diagram, actors, actorKeys, conf); diff --git a/packages/mermaid/src/diagrams/sequence/svgDraw.js b/packages/mermaid/src/diagrams/sequence/svgDraw.js index f81147c10c..31e6dc2a89 100644 --- a/packages/mermaid/src/diagrams/sequence/svgDraw.js +++ b/packages/mermaid/src/diagrams/sequence/svgDraw.js @@ -324,7 +324,7 @@ const drawActorTypeParticipant = function (elem, actor, conf, isFooter) { const center = actor.x + actor.width / 2; const centerY = actorY + 5; - const boxpluslineGroup = elem.append('g').lower(); + const boxpluslineGroup = elem.append('g'); var g = boxpluslineGroup; if (!isFooter) { diff --git a/packages/mermaid/src/diagrams/state/shapes.js b/packages/mermaid/src/diagrams/state/shapes.js index e82a1ad61a..b8cfe5bda1 100644 --- a/packages/mermaid/src/diagrams/state/shapes.js +++ b/packages/mermaid/src/diagrams/state/shapes.js @@ -3,7 +3,7 @@ import idCache from './id-cache.js'; import stateDb from './stateDb.js'; import utils from '../../utils.js'; import common from '../common/common.js'; -import { getConfig } from '../../config.js'; +import { getConfig } from '../../diagram-api/diagramAPI.js'; import { log } from '../../logger.js'; /** diff --git a/packages/mermaid/src/diagrams/state/stateDb.js b/packages/mermaid/src/diagrams/state/stateDb.js index 0253c5bcf0..7e5e72fe09 100644 --- a/packages/mermaid/src/diagrams/state/stateDb.js +++ b/packages/mermaid/src/diagrams/state/stateDb.js @@ -1,7 +1,7 @@ import { log } from '../../logger.js'; import { generateId } from '../../utils.js'; import common from '../common/common.js'; -import * as configApi from '../../config.js'; +import { getConfig } from '../../diagram-api/diagramAPI.js'; import { setAccTitle, getAccTitle, @@ -253,7 +253,7 @@ export const addState = function ( currentDocument.states[trimmedId].note = note; currentDocument.states[trimmedId].note.text = common.sanitizeText( currentDocument.states[trimmedId].note.text, - configApi.getConfig() + getConfig() ); } @@ -398,7 +398,7 @@ export function addRelationObjs(item1, item2, relationTitle) { currentDocument.relations.push({ id1, id2, - relationTitle: common.sanitizeText(relationTitle, configApi.getConfig()), + relationTitle: common.sanitizeText(relationTitle, getConfig()), }); } @@ -423,7 +423,7 @@ export const addRelation = function (item1, item2, title) { currentDocument.relations.push({ id1, id2, - title: common.sanitizeText(title, configApi.getConfig()), + title: common.sanitizeText(title, getConfig()), }); } }; @@ -431,7 +431,7 @@ export const addRelation = function (item1, item2, title) { export const addDescription = function (id, descr) { const theState = currentDocument.states[id]; const _descr = descr.startsWith(':') ? descr.replace(':', '').trim() : descr; - theState.descriptions.push(common.sanitizeText(_descr, configApi.getConfig())); + theState.descriptions.push(common.sanitizeText(_descr, getConfig())); }; export const cleanupLabel = function (label) { @@ -542,7 +542,7 @@ const setDirection = (dir) => { const trimColon = (str) => (str && str[0] === ':' ? str.substr(1).trim() : str.trim()); export default { - getConfig: () => configApi.getConfig().state, + getConfig: () => getConfig().state, addState, clear, getState, diff --git a/packages/mermaid/src/diagrams/state/stateDiagram.spec.js b/packages/mermaid/src/diagrams/state/stateDiagram.spec.js index 536031c815..7fcf4d0a6a 100644 --- a/packages/mermaid/src/diagrams/state/stateDiagram.spec.js +++ b/packages/mermaid/src/diagrams/state/stateDiagram.spec.js @@ -212,14 +212,14 @@ describe('state diagram, ', function () { parser.parse(str); }); - it('should handle state defintions with separation of id', function () { + it('should handle state definitions with separation of id', function () { const str = `stateDiagram\n state "Long state description" as state1 `; parser.parse(str); }); - it('should handle state defintions with separation of id', function () { + it('should handle state definitions with separation of id', function () { const str = `stateDiagram state "Not Shooting State" as NotShooting { state "Idle mode" as Idle @@ -360,7 +360,7 @@ describe('state diagram, ', function () { parser.parse(str); }); - it('should handle notes for composit states', function () { + it('should handle notes for composite states', function () { const str = `stateDiagram\n [*] --> NotShooting diff --git a/packages/mermaid/src/diagrams/state/stateRenderer-v2.js b/packages/mermaid/src/diagrams/state/stateRenderer-v2.js index 0d3117b206..482e37caee 100644 --- a/packages/mermaid/src/diagrams/state/stateRenderer-v2.js +++ b/packages/mermaid/src/diagrams/state/stateRenderer-v2.js @@ -1,6 +1,6 @@ import * as graphlib from 'dagre-d3-es/src/graphlib/index.js'; import { select } from 'd3'; -import { getConfig } from '../../config.js'; +import { getConfig } from '../../diagram-api/diagramAPI.js'; import { render } from '../../dagre-wrapper/index.js'; import { log } from '../../logger.js'; import { configureSvgSize } from '../../setupGraphViewbox.js'; diff --git a/packages/mermaid/src/diagrams/state/stateRenderer.js b/packages/mermaid/src/diagrams/state/stateRenderer.js index 1b3e0f27ed..17b674cb5f 100644 --- a/packages/mermaid/src/diagrams/state/stateRenderer.js +++ b/packages/mermaid/src/diagrams/state/stateRenderer.js @@ -4,7 +4,7 @@ import * as graphlib from 'dagre-d3-es/src/graphlib/index.js'; import { log } from '../../logger.js'; import common from '../common/common.js'; import { drawState, addTitleAndBox, drawEdge } from './shapes.js'; -import { getConfig } from '../../config.js'; +import { getConfig } from '../../diagram-api/diagramAPI.js'; import { configureSvgSize } from '../../setupGraphViewbox.js'; // TODO Move conf object to main conf in mermaidAPI diff --git a/packages/mermaid/src/diagrams/timeline/timelineRenderer.ts b/packages/mermaid/src/diagrams/timeline/timelineRenderer.ts index ee351d905b..2f1f156899 100644 --- a/packages/mermaid/src/diagrams/timeline/timelineRenderer.ts +++ b/packages/mermaid/src/diagrams/timeline/timelineRenderer.ts @@ -3,7 +3,7 @@ import type { Selection } from 'd3'; import { select } from 'd3'; import svgDraw from './svgDraw.js'; import { log } from '../../logger.js'; -import { getConfig } from '../../config.js'; +import { getConfig } from '../../diagram-api/diagramAPI.js'; import { setupGraphViewbox } from '../../setupGraphViewbox.js'; import type { Diagram } from '../../Diagram.js'; import type { MermaidConfig } from '../../config.type.js'; diff --git a/packages/mermaid/src/diagrams/user-journey/journeyDb.js b/packages/mermaid/src/diagrams/user-journey/journeyDb.js index 4d71c2e9dc..a2b95b899a 100644 --- a/packages/mermaid/src/diagrams/user-journey/journeyDb.js +++ b/packages/mermaid/src/diagrams/user-journey/journeyDb.js @@ -1,4 +1,4 @@ -import * as configApi from '../../config.js'; +import { getConfig } from '../../diagram-api/diagramAPI.js'; import { setAccTitle, getAccTitle, @@ -113,7 +113,7 @@ const getActors = function () { }; export default { - getConfig: () => configApi.getConfig().journey, + getConfig: () => getConfig().journey, clear, setDiagramTitle, getDiagramTitle, diff --git a/packages/mermaid/src/diagrams/user-journey/journeyRenderer.ts b/packages/mermaid/src/diagrams/user-journey/journeyRenderer.ts index 28c83f19d7..13eb31a024 100644 --- a/packages/mermaid/src/diagrams/user-journey/journeyRenderer.ts +++ b/packages/mermaid/src/diagrams/user-journey/journeyRenderer.ts @@ -1,7 +1,7 @@ // @ts-nocheck TODO: fix file import { select } from 'd3'; import svgDraw from './svgDraw.js'; -import { getConfig } from '../../config.js'; +import { getConfig } from '../../diagram-api/diagramAPI.js'; import { configureSvgSize } from '../../setupGraphViewbox.js'; export const setConf = function (cnf) { diff --git a/packages/mermaid/src/diagrams/xychart/chartBuilder/components/axis/bandAxis.ts b/packages/mermaid/src/diagrams/xychart/chartBuilder/components/axis/bandAxis.ts new file mode 100644 index 0000000000..864ef1316e --- /dev/null +++ b/packages/mermaid/src/diagrams/xychart/chartBuilder/components/axis/bandAxis.ts @@ -0,0 +1,45 @@ +import type { ScaleBand } from 'd3'; +import { scaleBand } from 'd3'; +import { log } from '../../../../../logger.js'; +import type { TextDimensionCalculator } from '../../textDimensionCalculator.js'; +import { BaseAxis } from './baseAxis.js'; +import type { XYChartAxisThemeConfig, XYChartAxisConfig } from '../../interfaces.js'; + +export class BandAxis extends BaseAxis { + private scale: ScaleBand; + private categories: string[]; + + constructor( + axisConfig: XYChartAxisConfig, + axisThemeConfig: XYChartAxisThemeConfig, + categories: string[], + title: string, + textDimensionCalculator: TextDimensionCalculator + ) { + super(axisConfig, title, textDimensionCalculator, axisThemeConfig); + this.categories = categories; + this.scale = scaleBand().domain(this.categories).range(this.getRange()); + } + + setRange(range: [number, number]): void { + super.setRange(range); + } + + recalculateScale(): void { + this.scale = scaleBand() + .domain(this.categories) + .range(this.getRange()) + .paddingInner(1) + .paddingOuter(0) + .align(0.5); + log.trace('BandAxis axis final categories, range: ', this.categories, this.getRange()); + } + + getTickValues(): (string | number)[] { + return this.categories; + } + + getScaleValue(value: string): number { + return this.scale(value) || this.getRange()[0]; + } +} diff --git a/packages/mermaid/src/diagrams/xychart/chartBuilder/components/axis/baseAxis.ts b/packages/mermaid/src/diagrams/xychart/chartBuilder/components/axis/baseAxis.ts new file mode 100644 index 0000000000..c3240a4a7b --- /dev/null +++ b/packages/mermaid/src/diagrams/xychart/chartBuilder/components/axis/baseAxis.ts @@ -0,0 +1,422 @@ +import type { + BoundingRect, + Dimension, + DrawableElem, + Point, + XYChartAxisConfig, + XYChartAxisThemeConfig, +} from '../../interfaces.js'; +import type { TextDimensionCalculator } from '../../textDimensionCalculator.js'; +import type { Axis, AxisPosition } from './index.js'; + +const BAR_WIDTH_TO_TICK_WIDTH_RATIO = 0.7; +const MAX_OUTER_PADDING_PERCENT_FOR_WRT_LABEL = 0.2; + +export abstract class BaseAxis implements Axis { + protected boundingRect: BoundingRect = { x: 0, y: 0, width: 0, height: 0 }; + protected axisPosition: AxisPosition = 'left'; + private range: [number, number]; + protected showTitle = false; + protected showLabel = false; + protected showTick = false; + protected showAxisLine = false; + protected outerPadding = 0; + protected titleTextHeight = 0; + protected labelTextHeight = 0; + + constructor( + protected axisConfig: XYChartAxisConfig, + protected title: string, + protected textDimensionCalculator: TextDimensionCalculator, + protected axisThemeConfig: XYChartAxisThemeConfig + ) { + this.range = [0, 10]; + this.boundingRect = { x: 0, y: 0, width: 0, height: 0 }; + this.axisPosition = 'left'; + } + + setRange(range: [number, number]): void { + this.range = range; + if (this.axisPosition === 'left' || this.axisPosition === 'right') { + this.boundingRect.height = range[1] - range[0]; + } else { + this.boundingRect.width = range[1] - range[0]; + } + this.recalculateScale(); + } + + getRange(): [number, number] { + return [this.range[0] + this.outerPadding, this.range[1] - this.outerPadding]; + } + + setAxisPosition(axisPosition: AxisPosition): void { + this.axisPosition = axisPosition; + this.setRange(this.range); + } + + abstract getScaleValue(value: number | string): number; + + abstract recalculateScale(): void; + + abstract getTickValues(): Array; + + getTickDistance(): number { + const range = this.getRange(); + return Math.abs(range[0] - range[1]) / this.getTickValues().length; + } + + getAxisOuterPadding(): number { + return this.outerPadding; + } + + private getLabelDimension(): Dimension { + return this.textDimensionCalculator.getMaxDimension( + this.getTickValues().map((tick) => tick.toString()), + this.axisConfig.labelFontSize + ); + } + + recalculateOuterPaddingToDrawBar(): void { + if (BAR_WIDTH_TO_TICK_WIDTH_RATIO * this.getTickDistance() > this.outerPadding * 2) { + this.outerPadding = Math.floor((BAR_WIDTH_TO_TICK_WIDTH_RATIO * this.getTickDistance()) / 2); + } + this.recalculateScale(); + } + + private calculateSpaceIfDrawnHorizontally(availableSpace: Dimension) { + let availableHeight = availableSpace.height; + if (this.axisConfig.showAxisLine && availableHeight > this.axisConfig.axisLineWidth) { + availableHeight -= this.axisConfig.axisLineWidth; + this.showAxisLine = true; + } + if (this.axisConfig.showLabel) { + const spaceRequired = this.getLabelDimension(); + const maxPadding = MAX_OUTER_PADDING_PERCENT_FOR_WRT_LABEL * availableSpace.width; + this.outerPadding = Math.min(spaceRequired.width / 2, maxPadding); + + const heightRequired = spaceRequired.height + this.axisConfig.labelPadding * 2; + this.labelTextHeight = spaceRequired.height; + if (heightRequired <= availableHeight) { + availableHeight -= heightRequired; + this.showLabel = true; + } + } + if (this.axisConfig.showTick && availableHeight >= this.axisConfig.tickLength) { + this.showTick = true; + availableHeight -= this.axisConfig.tickLength; + } + if (this.axisConfig.showTitle && this.title) { + const spaceRequired = this.textDimensionCalculator.getMaxDimension( + [this.title], + this.axisConfig.titleFontSize + ); + const heightRequired = spaceRequired.height + this.axisConfig.titlePadding * 2; + this.titleTextHeight = spaceRequired.height; + if (heightRequired <= availableHeight) { + availableHeight -= heightRequired; + this.showTitle = true; + } + } + this.boundingRect.width = availableSpace.width; + this.boundingRect.height = availableSpace.height - availableHeight; + } + + private calculateSpaceIfDrawnVertical(availableSpace: Dimension) { + let availableWidth = availableSpace.width; + if (this.axisConfig.showAxisLine && availableWidth > this.axisConfig.axisLineWidth) { + availableWidth -= this.axisConfig.axisLineWidth; + this.showAxisLine = true; + } + if (this.axisConfig.showLabel) { + const spaceRequired = this.getLabelDimension(); + const maxPadding = MAX_OUTER_PADDING_PERCENT_FOR_WRT_LABEL * availableSpace.height; + this.outerPadding = Math.min(spaceRequired.height / 2, maxPadding); + const widthRequired = spaceRequired.width + this.axisConfig.labelPadding * 2; + if (widthRequired <= availableWidth) { + availableWidth -= widthRequired; + this.showLabel = true; + } + } + if (this.axisConfig.showTick && availableWidth >= this.axisConfig.tickLength) { + this.showTick = true; + availableWidth -= this.axisConfig.tickLength; + } + if (this.axisConfig.showTitle && this.title) { + const spaceRequired = this.textDimensionCalculator.getMaxDimension( + [this.title], + this.axisConfig.titleFontSize + ); + const widthRequired = spaceRequired.height + this.axisConfig.titlePadding * 2; + this.titleTextHeight = spaceRequired.height; + if (widthRequired <= availableWidth) { + availableWidth -= widthRequired; + this.showTitle = true; + } + } + this.boundingRect.width = availableSpace.width - availableWidth; + this.boundingRect.height = availableSpace.height; + } + + calculateSpace(availableSpace: Dimension): Dimension { + if (this.axisPosition === 'left' || this.axisPosition === 'right') { + this.calculateSpaceIfDrawnVertical(availableSpace); + } else { + this.calculateSpaceIfDrawnHorizontally(availableSpace); + } + this.recalculateScale(); + return { + width: this.boundingRect.width, + height: this.boundingRect.height, + }; + } + + setBoundingBoxXY(point: Point): void { + this.boundingRect.x = point.x; + this.boundingRect.y = point.y; + } + + private getDrawableElementsForLeftAxis(): DrawableElem[] { + const drawableElement: DrawableElem[] = []; + if (this.showAxisLine) { + const x = this.boundingRect.x + this.boundingRect.width - this.axisConfig.axisLineWidth / 2; + drawableElement.push({ + type: 'path', + groupTexts: ['left-axis', 'axisl-line'], + data: [ + { + path: `M ${x},${this.boundingRect.y} L ${x},${ + this.boundingRect.y + this.boundingRect.height + } `, + strokeFill: this.axisThemeConfig.axisLineColor, + strokeWidth: this.axisConfig.axisLineWidth, + }, + ], + }); + } + if (this.showLabel) { + drawableElement.push({ + type: 'text', + groupTexts: ['left-axis', 'label'], + data: this.getTickValues().map((tick) => ({ + text: tick.toString(), + x: + this.boundingRect.x + + this.boundingRect.width - + (this.showLabel ? this.axisConfig.labelPadding : 0) - + (this.showTick ? this.axisConfig.tickLength : 0) - + (this.showAxisLine ? this.axisConfig.axisLineWidth : 0), + y: this.getScaleValue(tick), + fill: this.axisThemeConfig.labelColor, + fontSize: this.axisConfig.labelFontSize, + rotation: 0, + verticalPos: 'middle', + horizontalPos: 'right', + })), + }); + } + if (this.showTick) { + const x = + this.boundingRect.x + + this.boundingRect.width - + (this.showAxisLine ? this.axisConfig.axisLineWidth : 0); + drawableElement.push({ + type: 'path', + groupTexts: ['left-axis', 'ticks'], + data: this.getTickValues().map((tick) => ({ + path: `M ${x},${this.getScaleValue(tick)} L ${ + x - this.axisConfig.tickLength + },${this.getScaleValue(tick)}`, + strokeFill: this.axisThemeConfig.tickColor, + strokeWidth: this.axisConfig.tickWidth, + })), + }); + } + if (this.showTitle) { + drawableElement.push({ + type: 'text', + groupTexts: ['left-axis', 'title'], + data: [ + { + text: this.title, + x: this.boundingRect.x + this.axisConfig.titlePadding, + y: this.boundingRect.y + this.boundingRect.height / 2, + fill: this.axisThemeConfig.titleColor, + fontSize: this.axisConfig.titleFontSize, + rotation: 270, + verticalPos: 'top', + horizontalPos: 'center', + }, + ], + }); + } + return drawableElement; + } + private getDrawableElementsForBottomAxis(): DrawableElem[] { + const drawableElement: DrawableElem[] = []; + if (this.showAxisLine) { + const y = this.boundingRect.y + this.axisConfig.axisLineWidth / 2; + drawableElement.push({ + type: 'path', + groupTexts: ['bottom-axis', 'axis-line'], + data: [ + { + path: `M ${this.boundingRect.x},${y} L ${ + this.boundingRect.x + this.boundingRect.width + },${y}`, + strokeFill: this.axisThemeConfig.axisLineColor, + strokeWidth: this.axisConfig.axisLineWidth, + }, + ], + }); + } + if (this.showLabel) { + drawableElement.push({ + type: 'text', + groupTexts: ['bottom-axis', 'label'], + data: this.getTickValues().map((tick) => ({ + text: tick.toString(), + x: this.getScaleValue(tick), + y: + this.boundingRect.y + + this.axisConfig.labelPadding + + (this.showTick ? this.axisConfig.tickLength : 0) + + (this.showAxisLine ? this.axisConfig.axisLineWidth : 0), + fill: this.axisThemeConfig.labelColor, + fontSize: this.axisConfig.labelFontSize, + rotation: 0, + verticalPos: 'top', + horizontalPos: 'center', + })), + }); + } + if (this.showTick) { + const y = this.boundingRect.y + (this.showAxisLine ? this.axisConfig.axisLineWidth : 0); + drawableElement.push({ + type: 'path', + groupTexts: ['bottom-axis', 'ticks'], + data: this.getTickValues().map((tick) => ({ + path: `M ${this.getScaleValue(tick)},${y} L ${this.getScaleValue(tick)},${ + y + this.axisConfig.tickLength + }`, + strokeFill: this.axisThemeConfig.tickColor, + strokeWidth: this.axisConfig.tickWidth, + })), + }); + } + if (this.showTitle) { + drawableElement.push({ + type: 'text', + groupTexts: ['bottom-axis', 'title'], + data: [ + { + text: this.title, + x: this.range[0] + (this.range[1] - this.range[0]) / 2, + y: + this.boundingRect.y + + this.boundingRect.height - + this.axisConfig.titlePadding - + this.titleTextHeight, + fill: this.axisThemeConfig.titleColor, + fontSize: this.axisConfig.titleFontSize, + rotation: 0, + verticalPos: 'top', + horizontalPos: 'center', + }, + ], + }); + } + return drawableElement; + } + private getDrawableElementsForTopAxis(): DrawableElem[] { + const drawableElement: DrawableElem[] = []; + if (this.showAxisLine) { + const y = this.boundingRect.y + this.boundingRect.height - this.axisConfig.axisLineWidth / 2; + drawableElement.push({ + type: 'path', + groupTexts: ['top-axis', 'axis-line'], + data: [ + { + path: `M ${this.boundingRect.x},${y} L ${ + this.boundingRect.x + this.boundingRect.width + },${y}`, + strokeFill: this.axisThemeConfig.axisLineColor, + strokeWidth: this.axisConfig.axisLineWidth, + }, + ], + }); + } + if (this.showLabel) { + drawableElement.push({ + type: 'text', + groupTexts: ['top-axis', 'label'], + data: this.getTickValues().map((tick) => ({ + text: tick.toString(), + x: this.getScaleValue(tick), + y: + this.boundingRect.y + + (this.showTitle ? this.titleTextHeight + this.axisConfig.titlePadding * 2 : 0) + + this.axisConfig.labelPadding, + fill: this.axisThemeConfig.labelColor, + fontSize: this.axisConfig.labelFontSize, + rotation: 0, + verticalPos: 'top', + horizontalPos: 'center', + })), + }); + } + if (this.showTick) { + const y = this.boundingRect.y; + drawableElement.push({ + type: 'path', + groupTexts: ['top-axis', 'ticks'], + data: this.getTickValues().map((tick) => ({ + path: `M ${this.getScaleValue(tick)},${ + y + this.boundingRect.height - (this.showAxisLine ? this.axisConfig.axisLineWidth : 0) + } L ${this.getScaleValue(tick)},${ + y + + this.boundingRect.height - + this.axisConfig.tickLength - + (this.showAxisLine ? this.axisConfig.axisLineWidth : 0) + }`, + strokeFill: this.axisThemeConfig.tickColor, + strokeWidth: this.axisConfig.tickWidth, + })), + }); + } + if (this.showTitle) { + drawableElement.push({ + type: 'text', + groupTexts: ['top-axis', 'title'], + data: [ + { + text: this.title, + x: this.boundingRect.x + this.boundingRect.width / 2, + y: this.boundingRect.y + this.axisConfig.titlePadding, + fill: this.axisThemeConfig.titleColor, + fontSize: this.axisConfig.titleFontSize, + rotation: 0, + verticalPos: 'top', + horizontalPos: 'center', + }, + ], + }); + } + return drawableElement; + } + + getDrawableElements(): DrawableElem[] { + if (this.axisPosition === 'left') { + return this.getDrawableElementsForLeftAxis(); + } + if (this.axisPosition === 'right') { + throw Error('Drawing of right axis is not implemented'); + } + if (this.axisPosition === 'bottom') { + return this.getDrawableElementsForBottomAxis(); + } + if (this.axisPosition === 'top') { + return this.getDrawableElementsForTopAxis(); + } + return []; + } +} diff --git a/packages/mermaid/src/diagrams/xychart/chartBuilder/components/axis/index.ts b/packages/mermaid/src/diagrams/xychart/chartBuilder/components/axis/index.ts new file mode 100644 index 0000000000..3f1eca5476 --- /dev/null +++ b/packages/mermaid/src/diagrams/xychart/chartBuilder/components/axis/index.ts @@ -0,0 +1,47 @@ +import type { Group } from '../../../../../diagram-api/types.js'; +import type { + AxisDataType, + ChartComponent, + XYChartAxisConfig, + XYChartAxisThemeConfig, +} from '../../interfaces.js'; +import { isBandAxisData } from '../../interfaces.js'; +import { TextDimensionCalculatorWithFont } from '../../textDimensionCalculator.js'; +import { BandAxis } from './bandAxis.js'; +import { LinearAxis } from './linearAxis.js'; + +export type AxisPosition = 'left' | 'right' | 'top' | 'bottom'; + +export interface Axis extends ChartComponent { + getScaleValue(value: string | number): number; + setAxisPosition(axisPosition: AxisPosition): void; + getAxisOuterPadding(): number; + getTickDistance(): number; + recalculateOuterPaddingToDrawBar(): void; + setRange(range: [number, number]): void; +} + +export function getAxis( + data: AxisDataType, + axisConfig: XYChartAxisConfig, + axisThemeConfig: XYChartAxisThemeConfig, + tmpSVGGroup: Group +): Axis { + const textDimansionCalculator = new TextDimensionCalculatorWithFont(tmpSVGGroup); + if (isBandAxisData(data)) { + return new BandAxis( + axisConfig, + axisThemeConfig, + data.categories, + data.title, + textDimansionCalculator + ); + } + return new LinearAxis( + axisConfig, + axisThemeConfig, + [data.min, data.max], + data.title, + textDimansionCalculator + ); +} diff --git a/packages/mermaid/src/diagrams/xychart/chartBuilder/components/axis/linearAxis.ts b/packages/mermaid/src/diagrams/xychart/chartBuilder/components/axis/linearAxis.ts new file mode 100644 index 0000000000..8107732d93 --- /dev/null +++ b/packages/mermaid/src/diagrams/xychart/chartBuilder/components/axis/linearAxis.ts @@ -0,0 +1,38 @@ +import type { ScaleLinear } from 'd3'; +import { scaleLinear } from 'd3'; +import type { TextDimensionCalculator } from '../../textDimensionCalculator.js'; +import { BaseAxis } from './baseAxis.js'; +import type { XYChartAxisThemeConfig, XYChartAxisConfig } from '../../interfaces.js'; + +export class LinearAxis extends BaseAxis { + private scale: ScaleLinear; + private domain: [number, number]; + + constructor( + axisConfig: XYChartAxisConfig, + axisThemeConfig: XYChartAxisThemeConfig, + domain: [number, number], + title: string, + textDimensionCalculator: TextDimensionCalculator + ) { + super(axisConfig, title, textDimensionCalculator, axisThemeConfig); + this.domain = domain; + this.scale = scaleLinear().domain(this.domain).range(this.getRange()); + } + + getTickValues(): (string | number)[] { + return this.scale.ticks(); + } + + recalculateScale(): void { + const domain = [...this.domain]; // copy the array so if reverse is called two times it should not cancel the reverse effect + if (this.axisPosition === 'left') { + domain.reverse(); // since y-axis in svg start from top + } + this.scale = scaleLinear().domain(domain).range(this.getRange()); + } + + getScaleValue(value: number): number { + return this.scale(value); + } +} diff --git a/packages/mermaid/src/diagrams/xychart/chartBuilder/components/chartTitle.ts b/packages/mermaid/src/diagrams/xychart/chartBuilder/components/chartTitle.ts new file mode 100644 index 0000000000..bbab56bdc1 --- /dev/null +++ b/packages/mermaid/src/diagrams/xychart/chartBuilder/components/chartTitle.ts @@ -0,0 +1,91 @@ +import type { Group } from '../../../../diagram-api/types.js'; +import type { + BoundingRect, + ChartComponent, + Dimension, + DrawableElem, + Point, + XYChartData, + XYChartThemeConfig, + XYChartConfig, +} from '../interfaces.js'; +import type { TextDimensionCalculator } from '../textDimensionCalculator.js'; +import { TextDimensionCalculatorWithFont } from '../textDimensionCalculator.js'; + +export class ChartTitle implements ChartComponent { + private boundingRect: BoundingRect; + private showChartTitle: boolean; + constructor( + private textDimensionCalculator: TextDimensionCalculator, + private chartConfig: XYChartConfig, + private chartData: XYChartData, + private chartThemeConfig: XYChartThemeConfig + ) { + this.boundingRect = { + x: 0, + y: 0, + width: 0, + height: 0, + }; + this.showChartTitle = false; + } + setBoundingBoxXY(point: Point): void { + this.boundingRect.x = point.x; + this.boundingRect.y = point.y; + } + calculateSpace(availableSpace: Dimension): Dimension { + const titleDimension = this.textDimensionCalculator.getMaxDimension( + [this.chartData.title], + this.chartConfig.titleFontSize + ); + const widthRequired = Math.max(titleDimension.width, availableSpace.width); + const heightRequired = titleDimension.height + 2 * this.chartConfig.titlePadding; + if ( + titleDimension.width <= widthRequired && + titleDimension.height <= heightRequired && + this.chartConfig.showTitle && + this.chartData.title + ) { + this.boundingRect.width = widthRequired; + this.boundingRect.height = heightRequired; + this.showChartTitle = true; + } + + return { + width: this.boundingRect.width, + height: this.boundingRect.height, + }; + } + getDrawableElements(): DrawableElem[] { + const drawableElem: DrawableElem[] = []; + if (this.showChartTitle) { + drawableElem.push({ + groupTexts: ['chart-title'], + type: 'text', + data: [ + { + fontSize: this.chartConfig.titleFontSize, + text: this.chartData.title, + verticalPos: 'middle', + horizontalPos: 'center', + x: this.boundingRect.x + this.boundingRect.width / 2, + y: this.boundingRect.y + this.boundingRect.height / 2, + fill: this.chartThemeConfig.titleColor, + rotation: 0, + }, + ], + }); + } + return drawableElem; + } +} + +export function getChartTitleComponent( + chartConfig: XYChartConfig, + chartData: XYChartData, + chartThemeConfig: XYChartThemeConfig, + tmpSVGGroup: Group +): ChartComponent { + const textDimensionCalculator = new TextDimensionCalculatorWithFont(tmpSVGGroup); + return new ChartTitle(textDimensionCalculator, chartConfig, chartData, chartThemeConfig); +} diff --git a/packages/mermaid/src/diagrams/xychart/chartBuilder/components/plot/barPlot.ts b/packages/mermaid/src/diagrams/xychart/chartBuilder/components/plot/barPlot.ts new file mode 100644 index 0000000000..cf7d4e516d --- /dev/null +++ b/packages/mermaid/src/diagrams/xychart/chartBuilder/components/plot/barPlot.ts @@ -0,0 +1,60 @@ +import type { BarPlotData, BoundingRect, DrawableElem, XYChartConfig } from '../../interfaces.js'; +import type { Axis } from '../axis/index.js'; + +export class BarPlot { + constructor( + private barData: BarPlotData, + private boundingRect: BoundingRect, + private xAxis: Axis, + private yAxis: Axis, + private orientation: XYChartConfig['chartOrientation'], + private plotIndex: number + ) {} + + getDrawableElement(): DrawableElem[] { + const finalData: [number, number][] = this.barData.data.map((d) => [ + this.xAxis.getScaleValue(d[0]), + this.yAxis.getScaleValue(d[1]), + ]); + + const barPaddingPercent = 0.05; + + const barWidth = + Math.min(this.xAxis.getAxisOuterPadding() * 2, this.xAxis.getTickDistance()) * + (1 - barPaddingPercent); + const barWidthHalf = barWidth / 2; + + if (this.orientation === 'horizontal') { + return [ + { + groupTexts: ['plot', `bar-plot-${this.plotIndex}`], + type: 'rect', + data: finalData.map((data) => ({ + x: this.boundingRect.x, + y: data[0] - barWidthHalf, + height: barWidth, + width: data[1] - this.boundingRect.x, + fill: this.barData.fill, + strokeWidth: 0, + strokeFill: this.barData.fill, + })), + }, + ]; + } + return [ + { + groupTexts: ['plot', `bar-plot-${this.plotIndex}`], + type: 'rect', + data: finalData.map((data) => ({ + x: data[0] - barWidthHalf, + y: data[1], + width: barWidth, + height: this.boundingRect.y + this.boundingRect.height - data[1], + fill: this.barData.fill, + strokeWidth: 0, + strokeFill: this.barData.fill, + })), + }, + ]; + } +} diff --git a/packages/mermaid/src/diagrams/xychart/chartBuilder/components/plot/index.ts b/packages/mermaid/src/diagrams/xychart/chartBuilder/components/plot/index.ts new file mode 100644 index 0000000000..2a7b4a2838 --- /dev/null +++ b/packages/mermaid/src/diagrams/xychart/chartBuilder/components/plot/index.ts @@ -0,0 +1,97 @@ +import type { + XYChartData, + Dimension, + BoundingRect, + DrawableElem, + Point, + XYChartThemeConfig, + XYChartConfig, +} from '../../interfaces.js'; +import type { Axis } from '../axis/index.js'; +import type { ChartComponent } from '../../interfaces.js'; +import { LinePlot } from './linePlot.js'; +import { BarPlot } from './barPlot.js'; + +export interface Plot extends ChartComponent { + setAxes(xAxis: Axis, yAxis: Axis): void; +} + +export class BasePlot implements Plot { + private boundingRect: BoundingRect; + private xAxis?: Axis; + private yAxis?: Axis; + + constructor( + private chartConfig: XYChartConfig, + private chartData: XYChartData, + private chartThemeConfig: XYChartThemeConfig + ) { + this.boundingRect = { + x: 0, + y: 0, + width: 0, + height: 0, + }; + } + setAxes(xAxis: Axis, yAxis: Axis) { + this.xAxis = xAxis; + this.yAxis = yAxis; + } + setBoundingBoxXY(point: Point): void { + this.boundingRect.x = point.x; + this.boundingRect.y = point.y; + } + calculateSpace(availableSpace: Dimension): Dimension { + this.boundingRect.width = availableSpace.width; + this.boundingRect.height = availableSpace.height; + + return { + width: this.boundingRect.width, + height: this.boundingRect.height, + }; + } + getDrawableElements(): DrawableElem[] { + if (!(this.xAxis && this.yAxis)) { + throw Error('Axes must be passed to render Plots'); + } + const drawableElem: DrawableElem[] = []; + for (const [i, plot] of this.chartData.plots.entries()) { + switch (plot.type) { + case 'line': + { + const linePlot = new LinePlot( + plot, + this.xAxis, + this.yAxis, + this.chartConfig.chartOrientation, + i + ); + drawableElem.push(...linePlot.getDrawableElement()); + } + break; + case 'bar': + { + const barPlot = new BarPlot( + plot, + this.boundingRect, + this.xAxis, + this.yAxis, + this.chartConfig.chartOrientation, + i + ); + drawableElem.push(...barPlot.getDrawableElement()); + } + break; + } + } + return drawableElem; + } +} + +export function getPlotComponent( + chartConfig: XYChartConfig, + chartData: XYChartData, + chartThemeConfig: XYChartThemeConfig +): Plot { + return new BasePlot(chartConfig, chartData, chartThemeConfig); +} diff --git a/packages/mermaid/src/diagrams/xychart/chartBuilder/components/plot/linePlot.ts b/packages/mermaid/src/diagrams/xychart/chartBuilder/components/plot/linePlot.ts new file mode 100644 index 0000000000..d8d0666de3 --- /dev/null +++ b/packages/mermaid/src/diagrams/xychart/chartBuilder/components/plot/linePlot.ts @@ -0,0 +1,47 @@ +import { line } from 'd3'; +import type { DrawableElem, LinePlotData, XYChartConfig } from '../../interfaces.js'; +import type { Axis } from '../axis/index.js'; + +export class LinePlot { + constructor( + private plotData: LinePlotData, + private xAxis: Axis, + private yAxis: Axis, + private orientation: XYChartConfig['chartOrientation'], + private plotIndex: number + ) {} + + getDrawableElement(): DrawableElem[] { + const finalData: [number, number][] = this.plotData.data.map((d) => [ + this.xAxis.getScaleValue(d[0]), + this.yAxis.getScaleValue(d[1]), + ]); + + let path: string | null; + if (this.orientation === 'horizontal') { + path = line() + .y((d) => d[0]) + .x((d) => d[1])(finalData); + } else { + path = line() + .x((d) => d[0]) + .y((d) => d[1])(finalData); + } + if (!path) { + return []; + } + return [ + { + groupTexts: ['plot', `line-plot-${this.plotIndex}`], + type: 'path', + data: [ + { + path, + strokeFill: this.plotData.strokeFill, + strokeWidth: this.plotData.strokeWidth, + }, + ], + }, + ]; + } +} diff --git a/packages/mermaid/src/diagrams/xychart/chartBuilder/index.ts b/packages/mermaid/src/diagrams/xychart/chartBuilder/index.ts new file mode 100644 index 0000000000..192eb47f62 --- /dev/null +++ b/packages/mermaid/src/diagrams/xychart/chartBuilder/index.ts @@ -0,0 +1,15 @@ +import type { Group } from '../../../diagram-api/types.js'; +import type { DrawableElem, XYChartConfig, XYChartData, XYChartThemeConfig } from './interfaces.js'; +import { Orchestrator } from './orchestrator.js'; + +export class XYChartBuilder { + static build( + config: XYChartConfig, + chartData: XYChartData, + chartThemeConfig: XYChartThemeConfig, + tmpSVGGroup: Group + ): DrawableElem[] { + const orchestrator = new Orchestrator(config, chartData, chartThemeConfig, tmpSVGGroup); + return orchestrator.getDrawableElement(); + } +} diff --git a/packages/mermaid/src/diagrams/xychart/chartBuilder/interfaces.ts b/packages/mermaid/src/diagrams/xychart/chartBuilder/interfaces.ts new file mode 100644 index 0000000000..3d188895f2 --- /dev/null +++ b/packages/mermaid/src/diagrams/xychart/chartBuilder/interfaces.ts @@ -0,0 +1,163 @@ +export interface XYChartAxisThemeConfig { + titleColor: string; + labelColor: string; + tickColor: string; + axisLineColor: string; +} + +export interface XYChartThemeConfig { + backgroundColor: string; + titleColor: string; + xAxisLabelColor: string; + xAxisTitleColor: string; + xAxisTickColor: string; + xAxisLineColor: string; + yAxisLabelColor: string; + yAxisTitleColor: string; + yAxisTickColor: string; + yAxisLineColor: string; + plotColorPalette: string; +} + +export interface ChartComponent { + calculateSpace(availableSpace: Dimension): Dimension; + setBoundingBoxXY(point: Point): void; + getDrawableElements(): DrawableElem[]; +} + +export type SimplePlotDataType = [string, number][]; + +export interface LinePlotData { + type: 'line'; + strokeFill: string; + strokeWidth: number; + data: SimplePlotDataType; +} + +export interface BarPlotData { + type: 'bar'; + fill: string; + data: SimplePlotDataType; +} + +export type PlotData = LinePlotData | BarPlotData; + +export function isBarPlot(data: PlotData): data is BarPlotData { + return data.type === 'bar'; +} + +export interface BandAxisDataType { + type: 'band'; + title: string; + categories: string[]; +} + +export interface LinearAxisDataType { + type: 'linear'; + title: string; + min: number; + max: number; +} + +export type AxisDataType = LinearAxisDataType | BandAxisDataType; + +export function isBandAxisData(data: AxisDataType): data is BandAxisDataType { + return data.type === 'band'; +} + +export function isLinearAxisData(data: AxisDataType): data is LinearAxisDataType { + return data.type === 'linear'; +} + +/** + * For now we are keeping this configs as we are removing the required fields while generating the config.type.ts file + * we should remove `XYChartAxisConfig` and `XYChartConfig` after we started using required fields + */ +export interface XYChartAxisConfig { + showLabel: boolean; + labelFontSize: number; + labelPadding: number; + showTitle: boolean; + titleFontSize: number; + titlePadding: number; + showTick: boolean; + tickLength: number; + tickWidth: number; + showAxisLine: boolean; + axisLineWidth: number; +} + +export interface XYChartConfig { + width: number; + height: number; + titleFontSize: number; + titlePadding: number; + showTitle: boolean; + xAxis: XYChartAxisConfig; + yAxis: XYChartAxisConfig; + chartOrientation: 'vertical' | 'horizontal'; + plotReservedSpacePercent: number; +} + +export interface XYChartData { + xAxis: AxisDataType; + yAxis: AxisDataType; + title: string; + plots: PlotData[]; +} + +export interface Dimension { + width: number; + height: number; +} + +export interface BoundingRect extends Point, Dimension {} + +export interface Point { + x: number; + y: number; +} + +export type TextHorizontalPos = 'left' | 'center' | 'right'; +export type TextVerticalPos = 'top' | 'middle'; + +export interface RectElem extends Point { + width: number; + height: number; + fill: string; + strokeWidth: number; + strokeFill: string; +} + +export interface TextElem extends Point { + text: string; + fill: string; + verticalPos: TextVerticalPos; + horizontalPos: TextHorizontalPos; + fontSize: number; + rotation: number; +} + +export interface PathElem { + path: string; + fill?: string; + strokeWidth: number; + strokeFill: string; +} + +export type DrawableElem = + | { + groupTexts: string[]; + type: 'rect'; + data: RectElem[]; + } + | { + groupTexts: string[]; + type: 'text'; + data: TextElem[]; + } + | { + groupTexts: string[]; + type: 'path'; + data: PathElem[]; + }; diff --git a/packages/mermaid/src/diagrams/xychart/chartBuilder/orchestrator.ts b/packages/mermaid/src/diagrams/xychart/chartBuilder/orchestrator.ts new file mode 100644 index 0000000000..8338d4f411 --- /dev/null +++ b/packages/mermaid/src/diagrams/xychart/chartBuilder/orchestrator.ts @@ -0,0 +1,192 @@ +import type { + ChartComponent, + DrawableElem, + XYChartConfig, + XYChartData, + XYChartThemeConfig, +} from './interfaces.js'; +import { isBarPlot } from './interfaces.js'; +import type { Axis } from './components/axis/index.js'; +import { getAxis } from './components/axis/index.js'; +import { getChartTitleComponent } from './components/chartTitle.js'; +import type { Plot } from './components/plot/index.js'; +import { getPlotComponent } from './components/plot/index.js'; +import type { Group } from '../../../diagram-api/types.js'; + +export class Orchestrator { + private componentStore: { + title: ChartComponent; + plot: Plot; + xAxis: Axis; + yAxis: Axis; + }; + constructor( + private chartConfig: XYChartConfig, + private chartData: XYChartData, + chartThemeConfig: XYChartThemeConfig, + tmpSVGGroup: Group + ) { + this.componentStore = { + title: getChartTitleComponent(chartConfig, chartData, chartThemeConfig, tmpSVGGroup), + plot: getPlotComponent(chartConfig, chartData, chartThemeConfig), + xAxis: getAxis( + chartData.xAxis, + chartConfig.xAxis, + { + titleColor: chartThemeConfig.xAxisTitleColor, + labelColor: chartThemeConfig.xAxisLabelColor, + tickColor: chartThemeConfig.xAxisTickColor, + axisLineColor: chartThemeConfig.xAxisLineColor, + }, + tmpSVGGroup + ), + yAxis: getAxis( + chartData.yAxis, + chartConfig.yAxis, + { + titleColor: chartThemeConfig.yAxisTitleColor, + labelColor: chartThemeConfig.yAxisLabelColor, + tickColor: chartThemeConfig.yAxisTickColor, + axisLineColor: chartThemeConfig.yAxisLineColor, + }, + tmpSVGGroup + ), + }; + } + + private calculateVerticalSpace() { + let availableWidth = this.chartConfig.width; + let availableHeight = this.chartConfig.height; + let plotX = 0; + let plotY = 0; + let chartWidth = Math.floor((availableWidth * this.chartConfig.plotReservedSpacePercent) / 100); + let chartHeight = Math.floor( + (availableHeight * this.chartConfig.plotReservedSpacePercent) / 100 + ); + let spaceUsed = this.componentStore.plot.calculateSpace({ + width: chartWidth, + height: chartHeight, + }); + availableWidth -= spaceUsed.width; + availableHeight -= spaceUsed.height; + + spaceUsed = this.componentStore.title.calculateSpace({ + width: this.chartConfig.width, + height: availableHeight, + }); + plotY = spaceUsed.height; + availableHeight -= spaceUsed.height; + this.componentStore.xAxis.setAxisPosition('bottom'); + spaceUsed = this.componentStore.xAxis.calculateSpace({ + width: availableWidth, + height: availableHeight, + }); + availableHeight -= spaceUsed.height; + this.componentStore.yAxis.setAxisPosition('left'); + spaceUsed = this.componentStore.yAxis.calculateSpace({ + width: availableWidth, + height: availableHeight, + }); + plotX = spaceUsed.width; + availableWidth -= spaceUsed.width; + if (availableWidth > 0) { + chartWidth += availableWidth; + availableWidth = 0; + } + if (availableHeight > 0) { + chartHeight += availableHeight; + availableHeight = 0; + } + this.componentStore.plot.calculateSpace({ + width: chartWidth, + height: chartHeight, + }); + + this.componentStore.plot.setBoundingBoxXY({ x: plotX, y: plotY }); + this.componentStore.xAxis.setRange([plotX, plotX + chartWidth]); + this.componentStore.xAxis.setBoundingBoxXY({ x: plotX, y: plotY + chartHeight }); + this.componentStore.yAxis.setRange([plotY, plotY + chartHeight]); + this.componentStore.yAxis.setBoundingBoxXY({ x: 0, y: plotY }); + if (this.chartData.plots.some((p) => isBarPlot(p))) { + this.componentStore.xAxis.recalculateOuterPaddingToDrawBar(); + } + } + + private calculateHorizonatalSpace() { + let availableWidth = this.chartConfig.width; + let availableHeight = this.chartConfig.height; + let titleYEnd = 0; + let plotX = 0; + let plotY = 0; + let chartWidth = Math.floor((availableWidth * this.chartConfig.plotReservedSpacePercent) / 100); + let chartHeight = Math.floor( + (availableHeight * this.chartConfig.plotReservedSpacePercent) / 100 + ); + let spaceUsed = this.componentStore.plot.calculateSpace({ + width: chartWidth, + height: chartHeight, + }); + availableWidth -= spaceUsed.width; + availableHeight -= spaceUsed.height; + + spaceUsed = this.componentStore.title.calculateSpace({ + width: this.chartConfig.width, + height: availableHeight, + }); + titleYEnd = spaceUsed.height; + availableHeight -= spaceUsed.height; + this.componentStore.xAxis.setAxisPosition('left'); + spaceUsed = this.componentStore.xAxis.calculateSpace({ + width: availableWidth, + height: availableHeight, + }); + availableWidth -= spaceUsed.width; + plotX = spaceUsed.width; + this.componentStore.yAxis.setAxisPosition('top'); + spaceUsed = this.componentStore.yAxis.calculateSpace({ + width: availableWidth, + height: availableHeight, + }); + availableHeight -= spaceUsed.height; + plotY = titleYEnd + spaceUsed.height; + if (availableWidth > 0) { + chartWidth += availableWidth; + availableWidth = 0; + } + if (availableHeight > 0) { + chartHeight += availableHeight; + availableHeight = 0; + } + this.componentStore.plot.calculateSpace({ + width: chartWidth, + height: chartHeight, + }); + + this.componentStore.plot.setBoundingBoxXY({ x: plotX, y: plotY }); + this.componentStore.yAxis.setRange([plotX, plotX + chartWidth]); + this.componentStore.yAxis.setBoundingBoxXY({ x: plotX, y: titleYEnd }); + this.componentStore.xAxis.setRange([plotY, plotY + chartHeight]); + this.componentStore.xAxis.setBoundingBoxXY({ x: 0, y: plotY }); + if (this.chartData.plots.some((p) => isBarPlot(p))) { + this.componentStore.xAxis.recalculateOuterPaddingToDrawBar(); + } + } + + private calculateSpace() { + if (this.chartConfig.chartOrientation === 'horizontal') { + this.calculateHorizonatalSpace(); + } else { + this.calculateVerticalSpace(); + } + } + + getDrawableElement() { + this.calculateSpace(); + const drawableElem: DrawableElem[] = []; + this.componentStore.plot.setAxes(this.componentStore.xAxis, this.componentStore.yAxis); + for (const component of Object.values(this.componentStore)) { + drawableElem.push(...component.getDrawableElements()); + } + return drawableElem; + } +} diff --git a/packages/mermaid/src/diagrams/xychart/chartBuilder/textDimensionCalculator.ts b/packages/mermaid/src/diagrams/xychart/chartBuilder/textDimensionCalculator.ts new file mode 100644 index 0000000000..8049bf5272 --- /dev/null +++ b/packages/mermaid/src/diagrams/xychart/chartBuilder/textDimensionCalculator.ts @@ -0,0 +1,39 @@ +import type { Dimension } from './interfaces.js'; +import { computeDimensionOfText } from '../../../rendering-util/createText.js'; +import type { Group } from '../../../diagram-api/types.js'; + +export interface TextDimensionCalculator { + getMaxDimension(texts: string[], fontSize: number): Dimension; +} + +export class TextDimensionCalculatorWithFont implements TextDimensionCalculator { + constructor(private parentGroup: Group) {} + getMaxDimension(texts: string[], fontSize: number): Dimension { + if (!this.parentGroup) { + return { + width: texts.reduce((acc, cur) => Math.max(cur.length, acc), 0) * fontSize, + height: fontSize, + }; + } + + const dimension: Dimension = { + width: 0, + height: 0, + }; + + const elem = this.parentGroup + .append('g') + .attr('visibility', 'hidden') + .attr('font-size', fontSize); + + for (const t of texts) { + const bbox = computeDimensionOfText(elem, 1, t); + const width = bbox ? bbox.width : t.length * fontSize; + const height = bbox ? bbox.height : fontSize; + dimension.width = Math.max(dimension.width, width); + dimension.height = Math.max(dimension.height, height); + } + elem.remove(); + return dimension; + } +} diff --git a/packages/mermaid/src/diagrams/xychart/parser/xychart.jison b/packages/mermaid/src/diagrams/xychart/parser/xychart.jison new file mode 100644 index 0000000000..987132d17a --- /dev/null +++ b/packages/mermaid/src/diagrams/xychart/parser/xychart.jison @@ -0,0 +1,171 @@ +%lex +%options case-insensitive + +%x string +%x md_string +%x title +%x acc_title +%x acc_descr +%x acc_descr_multiline +%s axis_data +%s axis_band_data +%s data +%s data_inner +%% +\%\%(?!\{)[^\n]* /* skip comments */ +[^\}]\%\%[^\n]* /* skip comments */ +(\r?\n) { this.popState(); return 'NEWLINE'; } +(\r?\n) { this.popState(); return 'NEWLINE'; } +[\n\r]+ return 'NEWLINE'; +\%\%[^\n]* /* do nothing */ + +"title" { return 'title'; } + +"accTitle"\s*":"\s* { this.pushState("acc_title");return 'acc_title'; } +(?!\n|;|#)*[^\n]* { this.popState(); return "acc_title_value"; } +"accDescr"\s*":"\s* { this.pushState("acc_descr");return 'acc_descr'; } +(?!\n|;|#)*[^\n]* { this.popState(); return "acc_descr_value"; } +"accDescr"\s*"{"\s* { this.pushState("acc_descr_multiline");} +"{" { this.popState(); } +[^\}]* { return "acc_descr_multiline_value"; } + +"xychart-beta" {return 'XYCHART';} +(?:"vertical"|"horizontal") {return 'CHART_ORIENTATION'} + +"x-axis" { this.pushState("axis_data"); return "X_AXIS"; } +"y-axis" { this.pushState("axis_data"); return "Y_AXIS"; } +"[" { this.pushState("axis_band_data"); return 'SQUARE_BRACES_START'; } +"-->" { return 'ARROW_DELIMITER'; } + + +"line" { this.pushState("data"); return 'LINE'; } +"bar" { this.pushState("data"); return 'BAR'; } +"[" { this.pushState("data_inner"); return 'SQUARE_BRACES_START'; } +[+-]?(?:\d+(?:\.\d+)?|\.\d+) { return 'NUMBER_WITH_DECIMAL'; } +"]" { this.popState(); return 'SQUARE_BRACES_END'; } + + + + +(?:"`) { this.pushState("md_string"); } +(?:(?!`\").)+ { return "MD_STR"; } +(?:`") { this.popState(); } +["] this.pushState("string"); +["] this.popState(); +[^"]* return "STR"; + + +"[" return 'SQUARE_BRACES_START' +"]" return 'SQUARE_BRACES_END' +[A-Za-z]+ return 'ALPHA'; +":" return 'COLON'; +\+ return 'PLUS'; +"," return 'COMMA'; +\= return 'EQUALS'; +"*" return 'MULT'; +\# return 'BRKT'; +[\_] return 'UNDERSCORE'; +"." return 'DOT'; +"&" return 'AMP'; +\- return 'MINUS'; +[0-9]+ return 'NUM'; +\s+ /* skip */ +";" return 'SEMI'; +<> return 'EOF'; + +/lex + +%start start + +%% /* language grammar */ + +start + : eol start + | XYCHART chartConfig start + | XYCHART start + | document + ; + +chartConfig + : CHART_ORIENTATION { yy.setOrientation($1); } + ; + +document + : /* empty */ + | document statement + ; + +statement + : statement eol + | title text { yy.setDiagramTitle($text.text.trim()); } + | X_AXIS parseXAxis + | Y_AXIS parseYAxis + | LINE plotData { yy.setLineData({text: '', type: 'text'}, $plotData); } + | LINE text plotData { yy.setLineData($text, $plotData); } + | BAR plotData { yy.setBarData({text: '', type: 'text'}, $plotData); } + | BAR text plotData { yy.setBarData($text, $plotData); } + | acc_title acc_title_value { $$=$acc_title_value.trim();yy.setAccTitle($$); } + | acc_descr acc_descr_value { $$=$acc_descr_value.trim();yy.setAccDescription($$); } + | acc_descr_multiline_value { $$=$acc_descr_multiline_value.trim();yy.setAccDescription($$); } + ; + +plotData + : SQUARE_BRACES_START commaSeparatedNumbers SQUARE_BRACES_END { $$ = $commaSeparatedNumbers } + ; + +commaSeparatedNumbers + : NUMBER_WITH_DECIMAL COMMA commaSeparatedNumbers { $$ = [Number($NUMBER_WITH_DECIMAL), ...$commaSeparatedNumbers] } + | NUMBER_WITH_DECIMAL { $$ = [Number($NUMBER_WITH_DECIMAL)] } + ; + +parseXAxis + : text {yy.setXAxisTitle($text);} + | text xAxisData {yy.setXAxisTitle($text);} + | xAxisData {yy.setXAxisTitle({type: 'text', text: ''});} + ; + +xAxisData + : bandData {yy.setXAxisBand($bandData);} + | NUMBER_WITH_DECIMAL ARROW_DELIMITER NUMBER_WITH_DECIMAL {yy.setXAxisRangeData(Number($NUMBER_WITH_DECIMAL1), Number($NUMBER_WITH_DECIMAL2));} + ; + +bandData + : SQUARE_BRACES_START commaSeparatedTexts SQUARE_BRACES_END {$$ = $commaSeparatedTexts} + ; + +commaSeparatedTexts + : text COMMA commaSeparatedTexts { $$ = [$text, ...$commaSeparatedTexts] } + | text { $$ = [$text] } + ; + +parseYAxis + : text {yy.setYAxisTitle($text);} + | text yAxisData {yy.setYAxisTitle($text);} + | yAxisData {yy.setYAxisTitle({type: "text", text: ""});} + ; + +yAxisData + : NUMBER_WITH_DECIMAL ARROW_DELIMITER NUMBER_WITH_DECIMAL {yy.setYAxisRangeData(Number($NUMBER_WITH_DECIMAL1), Number($NUMBER_WITH_DECIMAL2));} + ; + +eol + : NEWLINE + | SEMI + | EOF + ; + + +text: alphaNum { $$={text:$alphaNum, type: 'text'};} + | STR { $$={text: $STR, type: 'text'};} + | MD_STR { $$={text: $MD_STR, type: 'markdown'};} + ; + +alphaNum + : alphaNumToken {$$=$alphaNumToken;} + | alphaNum alphaNumToken {$$=$alphaNum+''+$alphaNumToken;} + ; + + +alphaNumToken : AMP | NUM | ALPHA | PLUS | EQUALS | MULT | DOT | BRKT| MINUS | UNDERSCORE ; + +%% diff --git a/packages/mermaid/src/diagrams/xychart/parser/xychart.jison.spec.ts b/packages/mermaid/src/diagrams/xychart/parser/xychart.jison.spec.ts new file mode 100644 index 0000000000..d113250aa4 --- /dev/null +++ b/packages/mermaid/src/diagrams/xychart/parser/xychart.jison.spec.ts @@ -0,0 +1,448 @@ +// @ts-ignore: Jison doesn't support type. +import { parser } from './xychart.jison'; +import type { Mock } from 'vitest'; +import { vi } from 'vitest'; + +const parserFnConstructor = (str: string) => { + return () => { + parser.parse(str); + }; +}; + +const mockDB: Record> = { + setOrientation: vi.fn(), + setDiagramTitle: vi.fn(), + setXAxisTitle: vi.fn(), + setXAxisRangeData: vi.fn(), + setXAxisBand: vi.fn(), + setYAxisTitle: vi.fn(), + setYAxisRangeData: vi.fn(), + setLineData: vi.fn(), + setBarData: vi.fn(), +}; + +function clearMocks() { + for (const key in mockDB) { + mockDB[key].mockRestore(); + } +} + +describe('Testing xychart jison file', () => { + beforeEach(() => { + parser.yy = mockDB; + clearMocks(); + }); + + it('should throw error if xychart-beta text is not there', () => { + const str = 'xychart-beta-1'; + expect(parserFnConstructor(str)).toThrow(); + }); + + it('should not throw error if only xychart is there', () => { + const str = 'xychart-beta'; + expect(parserFnConstructor(str)).not.toThrow(); + }); + + it('parse title of the chart within "', () => { + const str = 'xychart-beta \n title "This is a title"'; + expect(parserFnConstructor(str)).not.toThrow(); + expect(mockDB.setDiagramTitle).toHaveBeenCalledWith('This is a title'); + }); + it('parse title of the chart without "', () => { + const str = 'xychart-beta \n title oneLinertitle'; + expect(parserFnConstructor(str)).not.toThrow(); + expect(mockDB.setDiagramTitle).toHaveBeenCalledWith('oneLinertitle'); + }); + + it('parse chart orientation', () => { + const str = 'xychart-beta vertical'; + expect(parserFnConstructor(str)).not.toThrow(); + expect(mockDB.setOrientation).toHaveBeenCalledWith('vertical'); + }); + + it('parse chart orientation with spaces', () => { + let str = 'xychart-beta horizontal '; + expect(parserFnConstructor(str)).not.toThrow(); + expect(mockDB.setOrientation).toHaveBeenCalledWith('horizontal'); + + str = 'xychart-beta abc'; + expect(parserFnConstructor(str)).toThrow(); + }); + + it('parse x-axis', () => { + const str = 'xychart-beta \nx-axis xAxisName\n'; + expect(parserFnConstructor(str)).not.toThrow(); + expect(mockDB.setXAxisTitle).toHaveBeenCalledWith({ + text: 'xAxisName', + type: 'text', + }); + }); + + it('parse x-axis with axis name without "', () => { + const str = 'xychart-beta \nx-axis xAxisName \n'; + expect(parserFnConstructor(str)).not.toThrow(); + expect(mockDB.setXAxisTitle).toHaveBeenCalledWith({ + text: 'xAxisName', + type: 'text', + }); + }); + + it('parse x-axis with axis name with "', () => { + const str = 'xychart-beta \n x-axis "xAxisName has space"\n'; + expect(parserFnConstructor(str)).not.toThrow(); + expect(mockDB.setXAxisTitle).toHaveBeenCalledWith({ + text: 'xAxisName has space', + type: 'text', + }); + }); + + it('parse x-axis with axis name with " with spaces', () => { + const str = 'xychart-beta \n x-axis " xAxisName has space " \n'; + expect(parserFnConstructor(str)).not.toThrow(); + expect(mockDB.setXAxisTitle).toHaveBeenCalledWith({ + text: ' xAxisName has space ', + type: 'text', + }); + }); + + it('parse x-axis with axis name and range data', () => { + const str = 'xychart-beta \nx-axis xAxisName 45.5 --> 33 \n'; + expect(parserFnConstructor(str)).not.toThrow(); + expect(mockDB.setXAxisTitle).toHaveBeenCalledWith({ + text: 'xAxisName', + type: 'text', + }); + expect(mockDB.setXAxisRangeData).toHaveBeenCalledWith(45.5, 33); + }); + it('parse x-axis throw error for invalid range data', () => { + const str = 'xychart-beta \nx-axis xAxisName aaa --> 33 \n'; + expect(parserFnConstructor(str)).toThrow(); + }); + it('parse x-axis with axis name and range data with only decimal part', () => { + const str = 'xychart-beta \nx-axis xAxisName 45.5 --> .34 \n'; + expect(parserFnConstructor(str)).not.toThrow(); + expect(mockDB.setXAxisTitle).toHaveBeenCalledWith({ + text: 'xAxisName', + type: 'text', + }); + expect(mockDB.setXAxisRangeData).toHaveBeenCalledWith(45.5, 0.34); + }); + + it('parse x-axis without axisname and range data', () => { + const str = 'xychart-beta \nx-axis 45.5 --> 1.34 \n'; + expect(parserFnConstructor(str)).not.toThrow(); + expect(mockDB.setXAxisTitle).toHaveBeenCalledWith({ + text: '', + type: 'text', + }); + expect(mockDB.setXAxisRangeData).toHaveBeenCalledWith(45.5, 1.34); + }); + + it('parse x-axis with axis name and category data', () => { + const str = 'xychart-beta \nx-axis xAxisName [ "cat1" , cat2a ] \n '; + expect(parserFnConstructor(str)).not.toThrow(); + expect(mockDB.setXAxisTitle).toHaveBeenCalledWith({ + text: 'xAxisName', + type: 'text', + }); + expect(mockDB.setXAxisBand).toHaveBeenCalledWith([ + { + text: 'cat1', + type: 'text', + }, + { text: 'cat2a', type: 'text' }, + ]); + }); + + it('parse x-axis without axisname and category data', () => { + const str = 'xychart-beta \nx-axis [ "cat1" , cat2a ] \n '; + expect(parserFnConstructor(str)).not.toThrow(); + expect(mockDB.setXAxisTitle).toHaveBeenCalledWith({ + text: '', + type: 'text', + }); + expect(mockDB.setXAxisBand).toHaveBeenCalledWith([ + { + text: 'cat1', + type: 'text', + }, + { text: 'cat2a', type: 'text' }, + ]); + }); + + it('parse x-axis throw error if unbalanced bracket', () => { + let str = 'xychart-beta \nx-axis xAxisName [ "cat1" [ cat2a ] \n '; + expect(parserFnConstructor(str)).toThrow(); + str = 'xychart-beta \nx-axis xAxisName [ "cat1" , cat2a ] ] \n '; + expect(parserFnConstructor(str)).toThrow(); + }); + + it('parse x-axis complete variant 1', () => { + const str = `xychart-beta \n x-axis "this is x axis" [category1, "category 2", category3]\n`; + expect(parserFnConstructor(str)).not.toThrow(); + expect(mockDB.setXAxisTitle).toHaveBeenCalledWith({ text: 'this is x axis', type: 'text' }); + expect(mockDB.setXAxisBand).toHaveBeenCalledWith([ + { text: 'category1', type: 'text' }, + { text: 'category 2', type: 'text' }, + { text: 'category3', type: 'text' }, + ]); + }); + + it('parse x-axis complete variant 2', () => { + const str = + 'xychart-beta \nx-axis xAxisName [ "cat1 with space" , cat2 , cat3] \n '; + expect(parserFnConstructor(str)).not.toThrow(); + expect(mockDB.setXAxisTitle).toHaveBeenCalledWith({ text: 'xAxisName', type: 'text' }); + expect(mockDB.setXAxisBand).toHaveBeenCalledWith([ + { text: 'cat1 with space', type: 'text' }, + { text: 'cat2', type: 'text' }, + { text: 'cat3', type: 'text' }, + ]); + }); + + it('parse x-axis complete variant 3', () => { + const str = + 'xychart-beta \nx-axis xAxisName [ "cat1 with space" , cat2 asdf , cat3] \n '; + expect(parserFnConstructor(str)).not.toThrow(); + expect(mockDB.setXAxisTitle).toHaveBeenCalledWith({ text: 'xAxisName', type: 'text' }); + expect(mockDB.setXAxisBand).toHaveBeenCalledWith([ + { text: 'cat1 with space', type: 'text' }, + { text: 'cat2asdf', type: 'text' }, + { text: 'cat3', type: 'text' }, + ]); + }); + + it('parse y-axis with axis name', () => { + const str = 'xychart-beta \ny-axis yAxisName\n'; + expect(parserFnConstructor(str)).not.toThrow(); + expect(mockDB.setYAxisTitle).toHaveBeenCalledWith({ text: 'yAxisName', type: 'text' }); + }); + it('parse y-axis with axis name with spaces', () => { + const str = 'xychart-beta \ny-axis yAxisName \n'; + expect(parserFnConstructor(str)).not.toThrow(); + expect(mockDB.setYAxisTitle).toHaveBeenCalledWith({ text: 'yAxisName', type: 'text' }); + }); + it('parse y-axis with axis name with "', () => { + const str = 'xychart-beta \n y-axis "yAxisName has space"\n'; + expect(parserFnConstructor(str)).not.toThrow(); + expect(mockDB.setYAxisTitle).toHaveBeenCalledWith({ + text: 'yAxisName has space', + type: 'text', + }); + }); + it('parse y-axis with axis name with " and spaces', () => { + const str = 'xychart-beta \n y-axis " yAxisName has space " \n'; + expect(parserFnConstructor(str)).not.toThrow(); + expect(mockDB.setYAxisTitle).toHaveBeenCalledWith({ + text: ' yAxisName has space ', + type: 'text', + }); + }); + it('parse y-axis with axis name with range data', () => { + const str = 'xychart-beta \ny-axis yAxisName 45.5 --> 33 \n'; + expect(parserFnConstructor(str)).not.toThrow(); + expect(mockDB.setYAxisTitle).toHaveBeenCalledWith({ text: 'yAxisName', type: 'text' }); + expect(mockDB.setYAxisRangeData).toHaveBeenCalledWith(45.5, 33); + }); + it('parse y-axis without axisname with range data', () => { + const str = 'xychart-beta \ny-axis 45.5 --> 33 \n'; + expect(parserFnConstructor(str)).not.toThrow(); + expect(mockDB.setYAxisTitle).toHaveBeenCalledWith({ text: '', type: 'text' }); + expect(mockDB.setYAxisRangeData).toHaveBeenCalledWith(45.5, 33); + }); + it('parse y-axis with axis name with range data with only decimal part', () => { + const str = 'xychart-beta \ny-axis yAxisName 45.5 --> .33 \n'; + expect(parserFnConstructor(str)).not.toThrow(); + expect(mockDB.setYAxisTitle).toHaveBeenCalledWith({ text: 'yAxisName', type: 'text' }); + expect(mockDB.setYAxisRangeData).toHaveBeenCalledWith(45.5, 0.33); + }); + it('parse y-axis throw error for invalid number in range data', () => { + const str = 'xychart-beta \ny-axis yAxisName 45.5 --> abc \n'; + expect(parserFnConstructor(str)).toThrow(); + }); + it('parse y-axis throws error if range data is passed', () => { + const str = 'xychart-beta \ny-axis yAxisName [ 45.3, 33 ] \n'; + expect(parserFnConstructor(str)).toThrow(); + }); + it('parse both axis at once', () => { + const str = 'xychart-beta\nx-axis xAxisName\ny-axis yAxisName\n'; + expect(parserFnConstructor(str)).not.toThrow(); + expect(mockDB.setXAxisTitle).toHaveBeenCalledWith({ text: 'xAxisName', type: 'text' }); + expect(mockDB.setYAxisTitle).toHaveBeenCalledWith({ text: 'yAxisName', type: 'text' }); + }); + it('parse line Data', () => { + const str = 'xychart-beta\nx-axis xAxisName\ny-axis yAxisName\n line lineTitle [23, 45, 56.6]'; + expect(parserFnConstructor(str)).not.toThrow(); + expect(mockDB.setLineData).toHaveBeenCalledWith( + { text: 'lineTitle', type: 'text' }, + [23, 45, 56.6] + ); + expect(mockDB.setXAxisTitle).toHaveBeenCalledWith({ text: 'xAxisName', type: 'text' }); + expect(mockDB.setYAxisTitle).toHaveBeenCalledWith({ text: 'yAxisName', type: 'text' }); + }); + it('parse line Data with spaces and +,- symbols', () => { + const str = + 'xychart-beta\nx-axis xAxisName\ny-axis yAxisName\n line "lineTitle with space" [ +23 , -45 , 56.6 ] '; + expect(parserFnConstructor(str)).not.toThrow(); + expect(mockDB.setYAxisTitle).toHaveBeenCalledWith({ text: 'yAxisName', type: 'text' }); + expect(mockDB.setXAxisTitle).toHaveBeenCalledWith({ text: 'xAxisName', type: 'text' }); + expect(mockDB.setLineData).toHaveBeenCalledWith( + { text: 'lineTitle with space', type: 'text' }, + [23, -45, 56.6] + ); + }); + it('parse line Data without title', () => { + const str = + 'xychart-beta\nx-axis xAxisName\ny-axis yAxisName\n line [ +23 , -45 , 56.6 , .33] '; + expect(parserFnConstructor(str)).not.toThrow(); + expect(mockDB.setYAxisTitle).toHaveBeenCalledWith({ text: 'yAxisName', type: 'text' }); + expect(mockDB.setXAxisTitle).toHaveBeenCalledWith({ text: 'xAxisName', type: 'text' }); + expect(mockDB.setLineData).toHaveBeenCalledWith( + { text: '', type: 'text' }, + [23, -45, 56.6, 0.33] + ); + }); + it('parse line Data throws error unbalanced brackets', () => { + let str = + 'xychart-beta\nx-axis xAxisName\ny-axis yAxisName\n line "lineTitle with space" [ +23 [ -45 , 56.6 ] '; + expect(parserFnConstructor(str)).toThrow(); + str = + 'xychart-beta\nx-axis xAxisName\ny-axis yAxisName\n line "lineTitle with space" [ +23 , -45 ] 56.6 ] '; + expect(parserFnConstructor(str)).toThrow(); + }); + it('parse line Data throws error if data is not provided', () => { + const str = 'xychart-beta\nx-axis xAxisName\ny-axis yAxisName\n line "lineTitle with space" '; + expect(parserFnConstructor(str)).toThrow(); + }); + it('parse line Data throws error if data is empty', () => { + const str = + 'xychart-beta\nx-axis xAxisName\ny-axis yAxisName\n line "lineTitle with space" [ ] '; + expect(parserFnConstructor(str)).toThrow(); + }); + it('parse line Data throws error if , is not in proper', () => { + const str = + 'xychart-beta\nx-axis xAxisName\ny-axis yAxisName\n line "lineTitle with space" [ +23 , , -45 , 56.6 ] '; + expect(parserFnConstructor(str)).toThrow(); + }); + it('parse line Data throws error if not number', () => { + const str = + 'xychart-beta\nx-axis xAxisName\ny-axis yAxisName\n line "lineTitle with space" [ +23 , -4aa5 , 56.6 ] '; + expect(parserFnConstructor(str)).toThrow(); + }); + it('parse bar Data', () => { + const str = + 'xychart-beta\nx-axis xAxisName\ny-axis yAxisName\n bar barTitle [23, 45, 56.6, .22]'; + expect(parserFnConstructor(str)).not.toThrow(); + expect(mockDB.setYAxisTitle).toHaveBeenCalledWith({ text: 'yAxisName', type: 'text' }); + expect(mockDB.setXAxisTitle).toHaveBeenCalledWith({ text: 'xAxisName', type: 'text' }); + expect(mockDB.setBarData).toHaveBeenCalledWith( + { text: 'barTitle', type: 'text' }, + [23, 45, 56.6, 0.22] + ); + }); + it('parse bar Data spaces and +,- symbol', () => { + const str = + 'xychart-beta\nx-axis xAxisName\ny-axis yAxisName\n bar "barTitle with space" [ +23 , -45 , 56.6 ] '; + expect(parserFnConstructor(str)).not.toThrow(); + expect(mockDB.setYAxisTitle).toHaveBeenCalledWith({ text: 'yAxisName', type: 'text' }); + expect(mockDB.setXAxisTitle).toHaveBeenCalledWith({ text: 'xAxisName', type: 'text' }); + expect(mockDB.setBarData).toHaveBeenCalledWith( + { text: 'barTitle with space', type: 'text' }, + [23, -45, 56.6] + ); + }); + it('parse bar Data without plot title', () => { + const str = + 'xychart-beta\nx-axis xAxisName\ny-axis yAxisName\n bar [ +23 , -45 , 56.6 ] '; + expect(parserFnConstructor(str)).not.toThrow(); + expect(mockDB.setYAxisTitle).toHaveBeenCalledWith({ text: 'yAxisName', type: 'text' }); + expect(mockDB.setXAxisTitle).toHaveBeenCalledWith({ text: 'xAxisName', type: 'text' }); + expect(mockDB.setBarData).toHaveBeenCalledWith({ text: '', type: 'text' }, [23, -45, 56.6]); + }); + it('parse bar should throw for unbalanced brackets', () => { + let str = + 'xychart-beta\nx-axis xAxisName\ny-axis yAxisName\n bar "barTitle with space" [ +23 [ -45 , 56.6 ] '; + expect(parserFnConstructor(str)).toThrow(); + str = + 'xychart-beta\nx-axis xAxisName\ny-axis yAxisName\n bar "barTitle with space" [ +23 , -45 ] 56.6 ] '; + expect(parserFnConstructor(str)).toThrow(); + }); + it('parse bar should throw error if data is not provided', () => { + const str = 'xychart-beta\nx-axis xAxisName\ny-axis yAxisName\n bar "barTitle with space" '; + expect(parserFnConstructor(str)).toThrow(); + }); + it('parse bar should throw error if data is empty', () => { + const str = + 'xychart-beta\nx-axis xAxisName\ny-axis yAxisName\n bar "barTitle with space" [ ] '; + expect(parserFnConstructor(str)).toThrow(); + }); + it('parse bar should throw error if comma is not proper', () => { + const str = + 'xychart-beta\nx-axis xAxisName\ny-axis yAxisName\n bar "barTitle with space" [ +23 , , -45 , 56.6 ] '; + expect(parserFnConstructor(str)).toThrow(); + }); + it('parse bar should throw error if number is not passed', () => { + const str = + 'xychart-beta\nx-axis xAxisName\ny-axis yAxisName\n bar "barTitle with space" [ +23 , -4aa5 , 56.6 ] '; + expect(parserFnConstructor(str)).toThrow(); + }); + it('parse multiple bar and line variant 1', () => { + const str = + 'xychart-beta\nx-axis xAxisName\ny-axis yAxisName\n bar barTitle1 [23, 45, 56.6] \n line lineTitle1 [11, 45.5, 67, 23] \n bar barTitle2 [13, 42, 56.89] \n line lineTitle2 [45, 99, 012]'; + expect(parserFnConstructor(str)).not.toThrow(); + expect(mockDB.setYAxisTitle).toHaveBeenCalledWith({ text: 'yAxisName', type: 'text' }); + expect(mockDB.setXAxisTitle).toHaveBeenCalledWith({ text: 'xAxisName', type: 'text' }); + expect(mockDB.setBarData).toHaveBeenCalledWith( + { text: 'barTitle1', type: 'text' }, + [23, 45, 56.6] + ); + expect(mockDB.setBarData).toHaveBeenCalledWith( + { text: 'barTitle2', type: 'text' }, + [13, 42, 56.89] + ); + expect(mockDB.setLineData).toHaveBeenCalledWith( + { text: 'lineTitle1', type: 'text' }, + [11, 45.5, 67, 23] + ); + expect(mockDB.setLineData).toHaveBeenCalledWith( + { text: 'lineTitle2', type: 'text' }, + [45, 99, 12] + ); + }); + it('parse multiple bar and line variant 2', () => { + const str = ` + xychart-beta horizontal + title Basic xychart + x-axis "this is x axis" [category1, "category 2", category3] + y-axis yaxisText 10 --> 150 + bar barTitle1 [23, 45, 56.6] + line lineTitle1 [11, 45.5, 67, 23] + bar barTitle2 [13, 42, 56.89] + line lineTitle2 [45, 99, 012]`; + expect(parserFnConstructor(str)).not.toThrow(); + expect(mockDB.setYAxisTitle).toHaveBeenCalledWith({ text: 'yaxisText', type: 'text' }); + expect(mockDB.setYAxisRangeData).toHaveBeenCalledWith(10, 150); + expect(mockDB.setXAxisTitle).toHaveBeenCalledWith({ text: 'this is x axis', type: 'text' }); + expect(mockDB.setXAxisBand).toHaveBeenCalledWith([ + { text: 'category1', type: 'text' }, + { text: 'category 2', type: 'text' }, + { text: 'category3', type: 'text' }, + ]); + expect(mockDB.setBarData).toHaveBeenCalledWith( + { text: 'barTitle1', type: 'text' }, + [23, 45, 56.6] + ); + expect(mockDB.setBarData).toHaveBeenCalledWith( + { text: 'barTitle2', type: 'text' }, + [13, 42, 56.89] + ); + expect(mockDB.setLineData).toHaveBeenCalledWith( + { text: 'lineTitle1', type: 'text' }, + [11, 45.5, 67, 23] + ); + expect(mockDB.setLineData).toHaveBeenCalledWith( + { text: 'lineTitle2', type: 'text' }, + [45, 99, 12] + ); + }); +}); diff --git a/packages/mermaid/src/diagrams/xychart/xychartDb.ts b/packages/mermaid/src/diagrams/xychart/xychartDb.ts new file mode 100644 index 0000000000..637477f28b --- /dev/null +++ b/packages/mermaid/src/diagrams/xychart/xychartDb.ts @@ -0,0 +1,229 @@ +import { + clear as commonClear, + getAccDescription, + getAccTitle, + getDiagramTitle, + setAccDescription, + setAccTitle, + setDiagramTitle, +} from '../common/commonDb.js'; +import * as configApi from '../../config.js'; +import defaultConfig from '../../defaultConfig.js'; +import { getThemeVariables } from '../../themes/theme-default.js'; +import { cleanAndMerge } from '../../utils.js'; +import { sanitizeText } from '../common/common.js'; +import { XYChartBuilder } from './chartBuilder/index.js'; +import type { + DrawableElem, + SimplePlotDataType, + XYChartConfig, + XYChartData, + XYChartThemeConfig, +} from './chartBuilder/interfaces.js'; +import { isBandAxisData, isLinearAxisData } from './chartBuilder/interfaces.js'; +import type { Group } from '../../diagram-api/types.js'; + +let plotIndex = 0; + +let tmpSVGGroup: Group; + +let xyChartConfig: XYChartConfig = getChartDefaultConfig(); +let xyChartThemeConfig: XYChartThemeConfig = getChartDefaultThemeConfig(); +let xyChartData: XYChartData = getChartDefaultData(); +let plotColorPalette = xyChartThemeConfig.plotColorPalette.split(',').map((color) => color.trim()); +let hasSetXAxis = false; +let hasSetYAxis = false; + +interface NormalTextType { + type: 'text'; + text: string; +} + +function getChartDefaultThemeConfig(): XYChartThemeConfig { + const defaultThemeVariables = getThemeVariables(); + const config = configApi.getConfig(); + return cleanAndMerge(defaultThemeVariables.xyChart, config.themeVariables.xyChart); +} +function getChartDefaultConfig(): XYChartConfig { + const config = configApi.getConfig(); + return cleanAndMerge( + defaultConfig.xyChart as XYChartConfig, + config.xyChart as XYChartConfig + ); +} + +function getChartDefaultData(): XYChartData { + return { + yAxis: { + type: 'linear', + title: '', + min: Infinity, + max: -Infinity, + }, + xAxis: { + type: 'band', + title: '', + categories: [], + }, + title: '', + plots: [], + }; +} + +function textSanitizer(text: string) { + const config = configApi.getConfig(); + return sanitizeText(text.trim(), config); +} + +function setTmpSVGG(SVGG: Group) { + tmpSVGGroup = SVGG; +} +function setOrientation(orientation: string) { + if (orientation === 'horizontal') { + xyChartConfig.chartOrientation = 'horizontal'; + } else { + xyChartConfig.chartOrientation = 'vertical'; + } +} +function setXAxisTitle(title: NormalTextType) { + xyChartData.xAxis.title = textSanitizer(title.text); +} +function setXAxisRangeData(min: number, max: number) { + xyChartData.xAxis = { type: 'linear', title: xyChartData.xAxis.title, min, max }; + hasSetXAxis = true; +} +function setXAxisBand(categories: NormalTextType[]) { + xyChartData.xAxis = { + type: 'band', + title: xyChartData.xAxis.title, + categories: categories.map((c) => textSanitizer(c.text)), + }; + hasSetXAxis = true; +} +function setYAxisTitle(title: NormalTextType) { + xyChartData.yAxis.title = textSanitizer(title.text); +} +function setYAxisRangeData(min: number, max: number) { + xyChartData.yAxis = { type: 'linear', title: xyChartData.yAxis.title, min, max }; + hasSetYAxis = true; +} + +// this function does not set `hasSetYAxis` as there can be multiple data so we should calculate the range accordingly +function setYAxisRangeFromPlotData(data: number[]) { + const minValue = Math.min(...data); + const maxValue = Math.max(...data); + const prevMinValue = isLinearAxisData(xyChartData.yAxis) ? xyChartData.yAxis.min : Infinity; + const prevMaxValue = isLinearAxisData(xyChartData.yAxis) ? xyChartData.yAxis.max : -Infinity; + xyChartData.yAxis = { + type: 'linear', + title: xyChartData.yAxis.title, + min: Math.min(prevMinValue, minValue), + max: Math.max(prevMaxValue, maxValue), + }; +} + +function transformDataWithoutCategory(data: number[]): SimplePlotDataType { + let retData: SimplePlotDataType = []; + if (data.length === 0) { + return retData; + } + if (!hasSetXAxis) { + const prevMinValue = isLinearAxisData(xyChartData.xAxis) ? xyChartData.xAxis.min : Infinity; + const prevMaxValue = isLinearAxisData(xyChartData.xAxis) ? xyChartData.xAxis.max : -Infinity; + setXAxisRangeData(Math.min(prevMinValue, 1), Math.max(prevMaxValue, data.length)); + } + if (!hasSetYAxis) { + setYAxisRangeFromPlotData(data); + } + + if (isBandAxisData(xyChartData.xAxis)) { + retData = xyChartData.xAxis.categories.map((c, i) => [c, data[i]]); + } + + if (isLinearAxisData(xyChartData.xAxis)) { + const min = xyChartData.xAxis.min; + const max = xyChartData.xAxis.max; + const step = (max - min + 1) / data.length; + const categories: string[] = []; + for (let i = min; i <= max; i += step) { + categories.push(`${i}`); + } + retData = categories.map((c, i) => [c, data[i]]); + } + + return retData; +} + +function getPlotColorFromPalette(plotIndex: number): string { + return plotColorPalette[plotIndex === 0 ? 0 : plotIndex % plotColorPalette.length]; +} + +function setLineData(title: NormalTextType, data: number[]) { + const plotData = transformDataWithoutCategory(data); + xyChartData.plots.push({ + type: 'line', + strokeFill: getPlotColorFromPalette(plotIndex), + strokeWidth: 2, + data: plotData, + }); + plotIndex++; +} + +function setBarData(title: NormalTextType, data: number[]) { + const plotData = transformDataWithoutCategory(data); + xyChartData.plots.push({ + type: 'bar', + fill: getPlotColorFromPalette(plotIndex), + data: plotData, + }); + plotIndex++; +} + +function getDrawableElem(): DrawableElem[] { + if (xyChartData.plots.length === 0) { + throw Error('No Plot to render, please provide a plot with some data'); + } + xyChartData.title = getDiagramTitle(); + return XYChartBuilder.build(xyChartConfig, xyChartData, xyChartThemeConfig, tmpSVGGroup); +} + +function getChartThemeConfig() { + return xyChartThemeConfig; +} + +function getChartConfig() { + return xyChartConfig; +} + +const clear = function () { + commonClear(); + plotIndex = 0; + xyChartConfig = getChartDefaultConfig(); + xyChartData = getChartDefaultData(); + xyChartThemeConfig = getChartDefaultThemeConfig(); + plotColorPalette = xyChartThemeConfig.plotColorPalette.split(',').map((color) => color.trim()); + hasSetXAxis = false; + hasSetYAxis = false; +}; + +export default { + getDrawableElem, + clear, + setAccTitle, + getAccTitle, + setDiagramTitle, + getDiagramTitle, + getAccDescription, + setAccDescription, + setOrientation, + setXAxisTitle, + setXAxisRangeData, + setXAxisBand, + setYAxisTitle, + setYAxisRangeData, + setLineData, + setBarData, + setTmpSVGG, + getChartThemeConfig, + getChartConfig, +}; diff --git a/packages/mermaid/src/diagrams/xychart/xychartDetector.ts b/packages/mermaid/src/diagrams/xychart/xychartDetector.ts new file mode 100644 index 0000000000..08be05b01e --- /dev/null +++ b/packages/mermaid/src/diagrams/xychart/xychartDetector.ts @@ -0,0 +1,24 @@ +import type { + DiagramDetector, + DiagramLoader, + ExternalDiagramDefinition, +} from '../../diagram-api/types.js'; + +const id = 'xychart'; + +const detector: DiagramDetector = (txt) => { + return /^\s*xychart-beta/.test(txt); +}; + +const loader: DiagramLoader = async () => { + const { diagram } = await import('./xychartDiagram.js'); + return { id, diagram }; +}; + +const plugin: ExternalDiagramDefinition = { + id, + detector, + loader, +}; + +export default plugin; diff --git a/packages/mermaid/src/diagrams/xychart/xychartDiagram.ts b/packages/mermaid/src/diagrams/xychart/xychartDiagram.ts new file mode 100644 index 0000000000..2f09c10a24 --- /dev/null +++ b/packages/mermaid/src/diagrams/xychart/xychartDiagram.ts @@ -0,0 +1,11 @@ +import type { DiagramDefinition } from '../../diagram-api/types.js'; +// @ts-ignore: Jison doesn't support types. +import parser from './parser/xychart.jison'; +import db from './xychartDb.js'; +import renderer from './xychartRenderer.js'; + +export const diagram: DiagramDefinition = { + parser, + db, + renderer, +}; diff --git a/packages/mermaid/src/diagrams/xychart/xychartRenderer.ts b/packages/mermaid/src/diagrams/xychart/xychartRenderer.ts new file mode 100644 index 0000000000..1f4d36e8ab --- /dev/null +++ b/packages/mermaid/src/diagrams/xychart/xychartRenderer.ts @@ -0,0 +1,123 @@ +import type { Diagram } from '../../Diagram.js'; +import { log } from '../../logger.js'; +import { selectSvgElement } from '../../rendering-util/selectSvgElement.js'; +import { configureSvgSize } from '../../setupGraphViewbox.js'; +import type { + DrawableElem, + TextElem, + TextHorizontalPos, + TextVerticalPos, +} from './chartBuilder/interfaces.js'; +import type XYChartDB from './xychartDb.js'; + +export const draw = (txt: string, id: string, _version: string, diagObj: Diagram) => { + const db = diagObj.db as typeof XYChartDB; + const themeConfig = db.getChartThemeConfig(); + const chartConfig = db.getChartConfig(); + function getDominantBaseLine(horizontalPos: TextVerticalPos) { + return horizontalPos === 'top' ? 'text-before-edge' : 'middle'; + } + + function getTextAnchor(verticalPos: TextHorizontalPos) { + return verticalPos === 'left' ? 'start' : verticalPos === 'right' ? 'end' : 'middle'; + } + + function getTextTransformation(data: TextElem) { + return `translate(${data.x}, ${data.y}) rotate(${data.rotation || 0})`; + } + + log.debug('Rendering xychart chart\n' + txt); + + const svg = selectSvgElement(id); + const group = svg.append('g').attr('class', 'main'); + const background = group + .append('rect') + .attr('width', chartConfig.width) + .attr('height', chartConfig.height) + .attr('class', 'background'); + + // @ts-ignore: TODO Fix ts errors + configureSvgSize(svg, chartConfig.height, chartConfig.width, true); + + svg.attr('viewBox', `0 0 ${chartConfig.width} ${chartConfig.height}`); + + background.attr('fill', themeConfig.backgroundColor); + + db.setTmpSVGG(svg.append('g').attr('class', 'mermaid-tmp-group')); + + const shapes: DrawableElem[] = db.getDrawableElem(); + + const groups: Record = {}; + + function getGroup(gList: string[]) { + let elem = group; + let prefix = ''; + for (const [i] of gList.entries()) { + let parent = group; + if (i > 0 && groups[prefix]) { + parent = groups[prefix]; + } + prefix += gList[i]; + elem = groups[prefix]; + if (!elem) { + elem = groups[prefix] = parent.append('g').attr('class', gList[i]); + } + } + return elem; + } + + for (const shape of shapes) { + if (shape.data.length === 0) { + continue; + } + + const shapeGroup = getGroup(shape.groupTexts); + + switch (shape.type) { + case 'rect': + shapeGroup + .selectAll('rect') + .data(shape.data) + .enter() + .append('rect') + .attr('x', (data) => data.x) + .attr('y', (data) => data.y) + .attr('width', (data) => data.width) + .attr('height', (data) => data.height) + .attr('fill', (data) => data.fill) + .attr('stroke', (data) => data.strokeFill) + .attr('stroke-width', (data) => data.strokeWidth); + break; + case 'text': + shapeGroup + .selectAll('text') + .data(shape.data) + .enter() + .append('text') + .attr('x', 0) + .attr('y', 0) + .attr('fill', (data) => data.fill) + .attr('font-size', (data) => data.fontSize) + .attr('dominant-baseline', (data) => getDominantBaseLine(data.verticalPos)) + .attr('text-anchor', (data) => getTextAnchor(data.horizontalPos)) + .attr('transform', (data) => getTextTransformation(data)) + .text((data) => data.text); + break; + case 'path': + shapeGroup + .selectAll('path') + .data(shape.data) + .enter() + .append('path') + .attr('d', (data) => data.path) + .attr('fill', (data) => (data.fill ? data.fill : 'none')) + .attr('stroke', (data) => data.strokeFill) + .attr('stroke-width', (data) => data.strokeWidth); + break; + } + } +}; + +export default { + draw, +}; diff --git a/packages/mermaid/src/docs/.vitepress/config.ts b/packages/mermaid/src/docs/.vitepress/config.ts index a950c7f011..691ca05651 100644 --- a/packages/mermaid/src/docs/.vitepress/config.ts +++ b/packages/mermaid/src/docs/.vitepress/config.ts @@ -91,7 +91,7 @@ function nav() { items: [ { text: 'Changelog', - link: 'https://github.com/mermaid-js/mermaid/blob/develop/CHANGELOG.md', + link: 'https://github.com/mermaid-js/mermaid/releases', }, { text: 'Contributing', @@ -150,6 +150,7 @@ function sidebarSyntax() { { text: 'Timeline 🔥', link: '/syntax/timeline' }, { text: 'Zenuml 🔥', link: '/syntax/zenuml' }, { text: 'Sankey 🔥', link: '/syntax/sankey' }, + { text: 'XYChart 🔥', link: '/syntax/xyChart' }, { text: 'Other Examples', link: '/syntax/examples' }, ], }, diff --git a/packages/mermaid/src/docs/.vitepress/theme/index.ts b/packages/mermaid/src/docs/.vitepress/theme/index.ts index ae626558db..6561578106 100644 --- a/packages/mermaid/src/docs/.vitepress/theme/index.ts +++ b/packages/mermaid/src/docs/.vitepress/theme/index.ts @@ -6,8 +6,8 @@ import Mermaid from './Mermaid.vue'; import Contributors from '../components/Contributors.vue'; // @ts-ignore import HomePage from '../components/HomePage.vue'; -// @ts-ignore -import TopBar from '../components/TopBar.vue'; +// // @ts-ignore +// import TopBar from '../components/TopBar.vue'; import { getRedirect } from './redirect.js'; @@ -21,7 +21,8 @@ export default { ...DefaultTheme, Layout() { return h(Theme.Layout, null, { - 'home-hero-before': () => h(TopBar), + // Keeping this as comment as it took a lot of time to figure out how to add a component to the top bar. + // 'home-hero-before': () => h(TopBar), 'home-features-after': () => h(HomePage), }); }, diff --git a/packages/mermaid/src/docs/community/questions-and-suggestions.md b/packages/mermaid/src/docs/community/questions-and-suggestions.md index 6d6f80fb6d..386e3753a1 100644 --- a/packages/mermaid/src/docs/community/questions-and-suggestions.md +++ b/packages/mermaid/src/docs/community/questions-and-suggestions.md @@ -4,9 +4,8 @@ ## First search to see if someone has already asked (and hopefully been answered) or suggested the same thing. -- Search in Discussions -- Search in open Issues -- Search in closed Issues +- [Search in Discussions](https://github.com/orgs/mermaid-js/discussions) +- [Search in Issues (Open & Closed)](https://github.com/mermaid-js/mermaid/issues?q=is%3Aissue) If you find an open issue or discussion thread that is similar to your question but isn't answered, you can let us know that you are also interested in it. Use the GitHub reactions to add a thumbs-up to the issue or discussion thread. diff --git a/packages/mermaid/src/docs/config/accessibility.md b/packages/mermaid/src/docs/config/accessibility.md index 67fb090b80..559c739874 100644 --- a/packages/mermaid/src/docs/config/accessibility.md +++ b/packages/mermaid/src/docs/config/accessibility.md @@ -91,7 +91,7 @@ See [the accTitle and accDescr usage examples](#acctitle-and-accdescr-usage-exam graph LR accTitle: Big Decisions accDescr: Bob's Burgers process for making big decisions - A[Identify Big Descision] --> B{Make Big Decision} + A[Identify Big Decision] --> B{Make Big Decision} B --> D[Be done] ``` @@ -123,7 +123,7 @@ Here is the HTML generated for the SVG element: _(Note that some of the SVG attr for making very, very big decisions. This is actually a very simple flow: identify the big decision and then make the big decision. } - A[Identify Big Descision] --> B{Make Big Decision} + A[Identify Big Decision] --> B{Make Big Decision} B --> D[Be done] ``` diff --git a/packages/mermaid/src/docs/config/configuration.md b/packages/mermaid/src/docs/config/configuration.md index dcbdcf8757..1eb7836a69 100644 --- a/packages/mermaid/src/docs/config/configuration.md +++ b/packages/mermaid/src/docs/config/configuration.md @@ -4,8 +4,8 @@ When mermaid starts, configuration is extracted to determine a configuration to - The default configuration - Overrides at the site level are set by the initialize call, and will be applied to all diagrams in the site/app. The term for this is the **siteConfig**. -- Frontmatter (v10.5.0+) - diagram authors can update select configuration parameters in the frontmatter of the diagram. These are applied to the render config. -- Directives (Deprecated by Frontmatter) - diagram authors can update select configuration parameters directly in the diagram code via directives. These are applied to the render config. +- Frontmatter (v10.5.0+) - diagram authors can update selected configuration parameters in the frontmatter of the diagram. These are applied to the render config. +- Directives (Deprecated by Frontmatter) - diagram authors can update selected configuration parameters directly in the diagram code via directives. These are applied to the render config. **The render config** is configuration that is used when rendering by applying these configurations. diff --git a/packages/mermaid/src/docs/ecosystem/integrations-community.md b/packages/mermaid/src/docs/ecosystem/integrations-community.md index 3a3a20de8f..f6ffd908fb 100644 --- a/packages/mermaid/src/docs/ecosystem/integrations-community.md +++ b/packages/mermaid/src/docs/ecosystem/integrations-community.md @@ -33,12 +33,19 @@ Below are a list of community plugins and integrations created with Mermaid. - [Notion](https://notion.so) ✅ - [Observable](https://observablehq.com/@observablehq/mermaid) ✅ - [Obsidian](https://help.obsidian.md/Editing+and+formatting/Advanced+formatting+syntax#Diagram) ✅ +- [NotesHub](https://noteshub.app) ✅ - [GitBook](https://gitbook.com) - [Mermaid Plugin](https://github.com/JozoVilcek/gitbook-plugin-mermaid) - [Markdown with Mermaid CLI](https://github.com/miao1007/gitbook-plugin-mermaid-cli) - [Mermaid plugin for GitBook](https://github.com/wwformat/gitbook-plugin-mermaid-pdf) - [LiveBook](https://livebook.dev) ✅ - [Atlassian Products](https://www.atlassian.com) + - [Mermaid for Confluence](https://marketplace.atlassian.com/apps/1224722/mermaid-for-confluence?hosting=cloud&tab=overview) + - [Mermaid Integration for Confluence](https://marketplace.atlassian.com/apps/1222792/mermaid-integration-for-confluence?hosting=cloud&tab=overview) + - [Mermaid Diagrams for Confluence](https://marketplace.atlassian.com/apps/1226945/mermaid-diagrams-for-confluence?hosting=cloud&tab=overview) + - [Mermaid Macro for Confluence](https://marketplace.atlassian.com/apps/1231150/mermaid-macro-for-confluence?hosting=cloud&tab=overview) + - [EliteSoft Mermaid Charts and Diagrams](https://marketplace.atlassian.com/apps/1227286/elitesoft-mermaid-charts-and-diagrams?hosting=cloud&tab=overview) + - [Mermaid for Jira Cloud - Draw UML diagrams easily](https://marketplace.atlassian.com/apps/1223053/mermaid-for-jira-cloud-draw-uml-diagrams-easily?hosting=cloud&tab=overview) - [Mermaid Charts & Diagrams for Confluence](https://marketplace.atlassian.com/apps/1222572/) - [Mermaid Charts & Diagrams for Jira](https://marketplace.atlassian.com/apps/1224537/) - [Mermaid Live Editor for Confluence Cloud](https://marketplace.atlassian.com/apps/1231571/mermaid-live-editor-for-confluence?hosting=cloud&tab=overview) @@ -97,6 +104,8 @@ Communication tools and platforms - [phpbb-ext-mermaid](https://github.com/AlfredoRamos/phpbb-ext-mermaid) - [NodeBB](https://nodebb.org) - [Mermaid Plugin](https://www.npmjs.com/package/nodebb-plugin-mermaid) +- [Slack](https://slack.com) + - [Mermaid for Slack](https://github.com/JackuB/mermaid-for-slack) ### Wikis @@ -167,13 +176,14 @@ Communication tools and platforms ### Document Generation +- [Docusaurus](https://docusaurus.io/docs/markdown-features/diagrams) ✅ - [Swimm - Up-to-date diagrams with Swimm, the knowledge management tool for code](https://docs.swimm.io/Features/diagrams-and-charts) - [Sphinx](https://www.sphinx-doc.org/en/master/) - [sphinxcontrib-mermaid](https://github.com/mgaitan/sphinxcontrib-mermaid) - [remark](https://remark.js.org/) - [remark-mermaidjs](https://github.com/remcohaszing/remark-mermaidjs) - [rehype](https://github.com/rehypejs/rehype) - - [rehype-mermaidjs](https://github.com/remcohaszing/rehype-mermaidjs) + - [rehype-mermaid](https://github.com/remcohaszing/rehype-mermaid) - [Gatsby](https://www.gatsbyjs.com/) - [gatsby-remark-mermaid](https://github.com/remcohaszing/gatsby-remark-mermaid) - [JSDoc](https://jsdoc.app/) diff --git a/packages/mermaid/src/docs/index.md b/packages/mermaid/src/docs/index.md index 378e9dfaba..218757b104 100644 --- a/packages/mermaid/src/docs/index.md +++ b/packages/mermaid/src/docs/index.md @@ -31,7 +31,7 @@ features: - title: 🏆 Award winning! details: 2019 JavaScript Open Source Award winner for "The Most Exciting Use of Technology". link: https://osawards.com/javascript/2019 - - title: 🎉 We are on Product Hunt! - details: We would love any and all support from the Mermaid community! - link: https://www.producthunt.com/posts/mermaid-chart + - title: 🥰 Mermaid + Mermaid Chart + details: Mermaid Chart is a major supporter of the Mermaid project. + link: https://www.mermaidchart.com/ --- diff --git a/packages/mermaid/src/docs/intro/examples.md b/packages/mermaid/src/docs/intro/examples.md index 7dda288dcb..978edb2b7a 100644 --- a/packages/mermaid/src/docs/intro/examples.md +++ b/packages/mermaid/src/docs/intro/examples.md @@ -117,3 +117,14 @@ quadrantChart Campaign E: [0.40, 0.34] Campaign F: [0.35, 0.78] ``` + +### [XY Chart](../syntax/xyChart.md) + +```mermaid-example +xychart-beta + title "Sales Revenue" + x-axis [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] + line [5000, 6000, 7500, 8200, 9500, 10500, 11000, 10200, 9200, 8500, 7000, 6000] +``` diff --git a/packages/mermaid/src/docs/intro/index.md b/packages/mermaid/src/docs/intro/index.md index bd45ffaf1a..535ee3a3dc 100644 --- a/packages/mermaid/src/docs/intro/index.md +++ b/packages/mermaid/src/docs/intro/index.md @@ -97,7 +97,7 @@ To Deploy Mermaid: ``` -**Doing so commands the mermaid parser to look for the `
    ` or `
    ` tags with `class="mermaid"`. From these tags, mermaid tries read the diagram/chart definitions and render them into SVG charts.**
    +**Doing so commands the mermaid parser to look for the `
    ` or `
    ` tags with `class="mermaid"`. From these tags, mermaid tries to read the diagram/chart definitions and render them into SVG charts.**
     
     **Examples can be found in** [Other examples](../syntax/examples.md)
     
    diff --git a/packages/mermaid/src/docs/package.json b/packages/mermaid/src/docs/package.json
    index ff8a03d5df..87777eb9fe 100644
    --- a/packages/mermaid/src/docs/package.json
    +++ b/packages/mermaid/src/docs/package.json
    @@ -22,17 +22,17 @@
       },
       "devDependencies": {
         "@iconify-json/carbon": "^1.1.16",
    -    "@unocss/reset": "^0.56.0",
    +    "@unocss/reset": "^0.57.0",
         "@vite-pwa/vitepress": "^0.2.0",
         "@vitejs/plugin-vue": "^4.2.1",
         "fast-glob": "^3.2.12",
         "https-localhost": "^4.7.1",
         "pathe": "^1.1.0",
    -    "unocss": "^0.56.0",
    +    "unocss": "^0.57.0",
         "unplugin-vue-components": "^0.25.0",
         "vite": "^4.3.9",
         "vite-plugin-pwa": "^0.16.0",
    -    "vitepress": "1.0.0-rc.20",
    +    "vitepress": "1.0.0-rc.25",
         "workbox-window": "^7.0.0"
       }
     }
    diff --git a/packages/mermaid/src/docs/syntax/c4.md b/packages/mermaid/src/docs/syntax/c4.md
    index be13323ea9..b6ee5fb795 100644
    --- a/packages/mermaid/src/docs/syntax/c4.md
    +++ b/packages/mermaid/src/docs/syntax/c4.md
    @@ -257,7 +257,7 @@ UpdateRelStyle(customerA, bankA, $offsetY="60")
         title Component diagram for Internet Banking System - API Application
     
         Container(spa, "Single Page Application", "javascript and angular", "Provides all the internet banking functionality to customers via their web browser.")
    -    Container(ma, "Mobile App", "Xamarin", "Provides a limited subset ot the internet banking functionality to customers via their mobile mobile device.")
    +    Container(ma, "Mobile App", "Xamarin", "Provides a limited subset to the internet banking functionality to customers via their mobile mobile device.")
         ContainerDb(db, "Database", "Relational Database Schema", "Stores user registration information, hashed authentication credentials, access logs, etc.")
         System_Ext(mbs, "Mainframe Banking System", "Stores all of the core banking information about customers, accounts, transactions, etc.")
     
    diff --git a/packages/mermaid/src/docs/syntax/classDiagram.md b/packages/mermaid/src/docs/syntax/classDiagram.md
    index 4b0cd49def..f02ae67be9 100644
    --- a/packages/mermaid/src/docs/syntax/classDiagram.md
    +++ b/packages/mermaid/src/docs/syntax/classDiagram.md
    @@ -281,8 +281,6 @@ And `Link` can be one of:
     
     A namespace groups classes.
     
    -Code:
    -
     ```mermaid-example
     classDiagram
     namespace BaseShapes {
    diff --git a/packages/mermaid/src/docs/syntax/flowchart.md b/packages/mermaid/src/docs/syntax/flowchart.md
    index d06e75c22b..23306acdd7 100644
    --- a/packages/mermaid/src/docs/syntax/flowchart.md
    +++ b/packages/mermaid/src/docs/syntax/flowchart.md
    @@ -293,7 +293,7 @@ flowchart TB
         A & B--> C & D
     ```
     
    -If you describe the same diagram using the the basic syntax, it will take four lines. A
    +If you describe the same diagram using the basic syntax, it will take four lines. A
     word of warning, one could go overboard with this making the flowchart harder to read in
     markdown form. The Swedish word `lagom` comes to mind. It means, not too much and not too little.
     This goes for expressive syntaxes as well.
    @@ -487,7 +487,7 @@ flowchart LR
         end
         %% ^ These subgraphs are identical, except for the links to them:
     
    -    %% Link *to* subgraph1: subgraph1 direction is mantained
    +    %% Link *to* subgraph1: subgraph1 direction is maintained
         outside --> subgraph1
         %% Link *within* subgraph2:
         %% subgraph2 inherits the direction of the top-level graph (LR)
    @@ -757,7 +757,7 @@ flowchart TD
         B-->E(A fa:fa-camera-retro perhaps?)
     ```
     
    -Mermaid is compatible with Font Awesome up to verion 5, Free icons only. Check that the icons you use are from the [supported set of icons](https://fontawesome.com/v5/search?o=r&m=free).
    +Mermaid is compatible with Font Awesome up to version 5, Free icons only. Check that the icons you use are from the [supported set of icons](https://fontawesome.com/v5/search?o=r&m=free).
     
     ## Graph declarations with spaces between vertices and link and without semicolon
     
    diff --git a/packages/mermaid/src/docs/syntax/gitgraph.md b/packages/mermaid/src/docs/syntax/gitgraph.md
    index 87f43afdde..5fa09cb225 100644
    --- a/packages/mermaid/src/docs/syntax/gitgraph.md
    +++ b/packages/mermaid/src/docs/syntax/gitgraph.md
    @@ -513,18 +513,25 @@ Here, we have changed the default main branch name to `MetroLine1`.
     
     ## Orientation (v10.3.0+)
     
    -In Mermaid, the default orientation is Left to Right. The branches are lined vertically.
    +Mermaid supports two graph orientations: **Left-to-Right** (default) and **Top-to-Bottom**.
    +
    +You can set this with either `LR:` (for [**Left-to-Right**](#left-to-right-default-lr)) or `TB:` (for [**Top-to-Bottom**](#top-to-bottom-tb)) after `gitGraph`.
    +
    +### Left to Right (default, `LR:`)
    +
    +In Mermaid, the default orientation is for commits to run from left to right and for branches to be stacked on top of one another.
    +
    +However, you can set this explicitly with `LR:` after `gitGraph`.
     
     Usage example:
     
     ```mermaid-example
    -    gitGraph
    +    gitGraph LR:
            commit
            commit
            branch develop
            commit
            commit
    -       commit
            checkout main
            commit
            commit
    @@ -533,9 +540,11 @@ Usage example:
            commit
     ```
     
    -Sometimes we may want to change the orientation. Currently, Mermaid supports two orientations: **Left to Right**(default) and **Top to Bottom**.
    +### Top to Bottom (`TB:`)
     
    -In order to change the orientation from top to bottom i.e. branches lined horizontally, you need to add `TB` along with `gitGraph`.
    +In `TB` (**Top-to-Bottom**) orientation, the commits run from top to bottom of the graph and branches are arranged side-by-side.
    +
    +To orient the graph this way, you need to add `TB:` after gitGraph.
     
     Usage example:
     
    @@ -546,7 +555,6 @@ Usage example:
            branch develop
            commit
            commit
    -       commit
            checkout main
            commit
            commit
    diff --git a/packages/mermaid/src/docs/syntax/quadrantChart.md b/packages/mermaid/src/docs/syntax/quadrantChart.md
    index 835bbfa32f..d6793aea61 100644
    --- a/packages/mermaid/src/docs/syntax/quadrantChart.md
    +++ b/packages/mermaid/src/docs/syntax/quadrantChart.md
    @@ -83,26 +83,26 @@ Points are used to plot a circle inside the quadrantChart. The syntax is `
     
     ## Chart Configurations
     
    -| Parameter                         | Description                                                                                       | Default value |
    -| --------------------------------- | ------------------------------------------------------------------------------------------------- | :-----------: |
    -| chartWidth                        | Width of the chart                                                                                |      500      |
    -| chartHeight                       | Height of the chart                                                                               |      500      |
    -| titlePadding                      | Top and Bottom padding of the title                                                               |      10       |
    -| titleFontSize                     | Title font size                                                                                   |      20       |
    -| quadrantPadding                   | Padding outside all the quadrants                                                                 |       5       |
    -| quadrantTextTopPadding            | Quadrant text top padding when text is drawn on top ( not data points are there)                  |       5       |
    -| quadrantLabelFontSize             | Quadrant text font size                                                                           |      16       |
    -| quadrantInternalBorderStrokeWidth | Border stroke width inside the quadrants                                                          |       1       |
    -| quadrantExternalBorderStrokeWidth | Quadrant external border stroke width                                                             |       2       |
    -| xAxisLabelPadding                 | Top and bottom padding of x-axis text                                                             |       5       |
    -| xAxisLabelFontSize                | X-axis texts font size                                                                            |      16       |
    -| xAxisPosition                     | Position of x-axis (top , bottom) if there are points the x-axis will alway be rendered in bottom |     'top'     |
    -| yAxisLabelPadding                 | Left and Right padding of y-axis text                                                             |       5       |
    -| yAxisLabelFontSize                | Y-axis texts font size                                                                            |      16       |
    -| yAxisPosition                     | Position of y-axis (left , right)                                                                 |    'left'     |
    -| pointTextPadding                  | Padding between point and the below text                                                          |       5       |
    -| pointLabelFontSize                | Point text font size                                                                              |      12       |
    -| pointRadius                       | Radius of the point to be drawn                                                                   |       5       |
    +| Parameter                         | Description                                                                                        | Default value |
    +| --------------------------------- | -------------------------------------------------------------------------------------------------- | :-----------: |
    +| chartWidth                        | Width of the chart                                                                                 |      500      |
    +| chartHeight                       | Height of the chart                                                                                |      500      |
    +| titlePadding                      | Top and Bottom padding of the title                                                                |      10       |
    +| titleFontSize                     | Title font size                                                                                    |      20       |
    +| quadrantPadding                   | Padding outside all the quadrants                                                                  |       5       |
    +| quadrantTextTopPadding            | Quadrant text top padding when text is drawn on top ( not data points are there)                   |       5       |
    +| quadrantLabelFontSize             | Quadrant text font size                                                                            |      16       |
    +| quadrantInternalBorderStrokeWidth | Border stroke width inside the quadrants                                                           |       1       |
    +| quadrantExternalBorderStrokeWidth | Quadrant external border stroke width                                                              |       2       |
    +| xAxisLabelPadding                 | Top and bottom padding of x-axis text                                                              |       5       |
    +| xAxisLabelFontSize                | X-axis texts font size                                                                             |      16       |
    +| xAxisPosition                     | Position of x-axis (top , bottom) if there are points the x-axis will always be rendered in bottom |     'top'     |
    +| yAxisLabelPadding                 | Left and Right padding of y-axis text                                                              |       5       |
    +| yAxisLabelFontSize                | Y-axis texts font size                                                                             |      16       |
    +| yAxisPosition                     | Position of y-axis (left , right)                                                                  |    'left'     |
    +| pointTextPadding                  | Padding between point and the below text                                                           |       5       |
    +| pointLabelFontSize                | Point text font size                                                                               |      12       |
    +| pointRadius                       | Radius of the point to be drawn                                                                    |       5       |
     
     ## Chart Theme Variables
     
    diff --git a/packages/mermaid/src/docs/syntax/timeline.md b/packages/mermaid/src/docs/syntax/timeline.md
    index 201ab6b16f..eeff9b1353 100644
    --- a/packages/mermaid/src/docs/syntax/timeline.md
    +++ b/packages/mermaid/src/docs/syntax/timeline.md
    @@ -143,7 +143,7 @@ Note that there are no sections defined, and each time period and its correspond
     
     2. Disable the multiColor option using the `disableMultiColor` option. This will make all time periods and events follow the same color scheme.
     
    -You will need to add this option either via mermaid.intialize function or directives.
    +You will need to add this option either via mermaid.initialize function or directives.
     
     ```javascript
     mermaid.initialize({
    diff --git a/packages/mermaid/src/docs/syntax/xyChart.md b/packages/mermaid/src/docs/syntax/xyChart.md
    new file mode 100644
    index 0000000000..8edfecbea3
    --- /dev/null
    +++ b/packages/mermaid/src/docs/syntax/xyChart.md
    @@ -0,0 +1,165 @@
    +# XY Chart
    +
    +> In the context of mermaid-js, the XY chart is a comprehensive charting module that encompasses various types of charts that utilize both x-axis and y-axis for data representation. Presently, it includes two fundamental chart types: the bar chart and the line chart. These charts are designed to visually display and analyze data that involve two numerical variables.
    +
    +> It's important to note that while the current implementation of mermaid-js includes these two chart types, the framework is designed to be dynamic and adaptable. Therefore, it has the capacity for expansion and the inclusion of additional chart types in the future. This means that users can expect an evolving suite of charting options within the XY chart module, catering to various data visualization needs as new chart types are introduced over time.
    +
    +## Example
    +
    +```mermaid-example
    +xychart-beta
    +    title "Sales Revenue"
    +    x-axis [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]
    +    line [5000, 6000, 7500, 8200, 9500, 10500, 11000, 10200, 9200, 8500, 7000, 6000]
    +```
    +
    +## Syntax
    +
    +```note
    +All text values that contain only one word can be written without `"`. If a text value has many words in it, specifically if it contains spaces, enclose the value in `"`
    +```
    +
    +### Orientations
    +
    +The chart can be drawn horizontal or vertical, default value is vertical.
    +
    +```
    +xychart-beta horizontal
    +...
    +```
    +
    +### Title
    +
    +The title is a short description of the chart and it will always render on top of the chart.
    +
    +#### Example
    +
    +```
    +xychart-beta
    +    title "This is a simple example"
    +    ...
    +```
    +
    +```note
    +If the title is a single word one no need to use `"`, but if it has space `"` is needed
    +```
    +
    +### x-axis
    +
    +The x-axis primarily serves as a categorical value, although it can also function as a numeric range value when needed.
    +
    +#### Example
    +
    +1. `x-axis title min --> max` x-axis will function as numeric with the given range
    +2. `x-axis "title with space" [cat1, "cat2 with space", cat3]` x-axis if categorical, categories are text type
    +
    +### y-axis
    +
    +The y-axis is employed to represent numerical range values, it cannot have categorical values.
    +
    +#### Example
    +
    +1. `y-axis title min --> max`
    +2. `y-axis title` it will only add the title, the range will be auto generated from data.
    +
    +```note
    +Both x and y axis are optional if not provided we will try to create the range
    +```
    +
    +### Line chart
    +
    +A line chart offers the capability to graphically depict lines.
    +
    +#### Example
    +
    +1. `line [2.3, 45, .98, -3.4]` it can have all valid numeric values.
    +
    +### Bar chart
    +
    +A bar chart offers the capability to graphically depict bars.
    +
    +#### Example
    +
    +1. `bar [2.3, 45, .98, -3.4]` it can have all valid numeric values.
    +
    +#### Simplest example
    +
    +The only two things required are the chart name (`xychart-beta`) and one data set. So you will be able to draw a chart with a simple config like
    +
    +```
    +xychart-beta
    +    line [+1.3, .6, 2.4, -.34]
    +```
    +
    +## Chart Configurations
    +
    +| Parameter                | Description                                    | Default value |
    +| ------------------------ | ---------------------------------------------- | :-----------: |
    +| width                    | Width of the chart                             |      700      |
    +| height                   | Height of the chart                            |      500      |
    +| titlePadding             | Top and Bottom padding of the title            |      10       |
    +| titleFontSize            | Title font size                                |      20       |
    +| showTitle                | Title to be shown or not                       |     true      |
    +| xAxis                    | xAxis configuration                            |  AxisConfig   |
    +| yAxis                    | yAxis configuration                            |  AxisConfig   |
    +| chartOrientation         | 'vertical' or 'horizontal'                     |  'vertical'   |
    +| plotReservedSpacePercent | Minimum space plots will take inside the chart |      50       |
    +
    +### AxisConfig
    +
    +| Parameter     | Description                          | Default value |
    +| ------------- | ------------------------------------ | :-----------: |
    +| showLabel     | Show axis labels or tick values      |     true      |
    +| labelFontSize | Font size of the label to be drawn   |      14       |
    +| labelPadding  | Top and Bottom padding of the label  |       5       |
    +| showTitle     | Axis title to be shown or not        |     true      |
    +| titleFontSize | Axis title font size                 |      16       |
    +| titlePadding  | Top and Bottom padding of Axis title |       5       |
    +| showTick      | Tick to be shown or not              |     true      |
    +| tickLength    | How long the tick will be            |       5       |
    +| tickWidth     | How width the tick will be           |       2       |
    +| showAxisLine  | Axis line to be shown or not         |     true      |
    +| axisLineWidth | Thickness of the axis line           |       2       |
    +
    +## Chart Theme Variables
    +
    +```note
    +Themes for xychart resides inside xychart attribute so to set the variables use this syntax
    +%%{init: { "themeVariables": {"xyChart": {"titleColor": "#ff0000"} } }}%%
    +```
    +
    +| Parameter        | Description                                               |
    +| ---------------- | --------------------------------------------------------- |
    +| backgroundColor  | Background color of the whole chart                       |
    +| titleColor       | Color of the Title text                                   |
    +| xAxisLableColor  | Color of the x-axis labels                                |
    +| xAxisTitleColor  | Color of the x-axis title                                 |
    +| xAxisTickColor   | Color of the x-axis tick                                  |
    +| xAxisLineColor   | Color of the x-axis line                                  |
    +| yAxisLableColor  | Color of the y-axis labels                                |
    +| yAxisTitleColor  | Color of the y-axis title                                 |
    +| yAxisTickColor   | Color of the y-axis tick                                  |
    +| yAxisLineColor   | Color of the y-axis line                                  |
    +| plotColorPalette | String of colors separated by comma e.g. "#f3456, #43445" |
    +
    +## Example on config and theme
    +
    +```mermaid-example
    +---
    +config:
    +    xyChart:
    +        width: 900
    +        height: 600
    +    themeVariables:
    +        xyChart:
    +            titleColor: "#ff0000"
    +---
    +xychart-beta
    +    title "Sales Revenue"
    +    x-axis [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]
    +    line [5000, 6000, 7500, 8200, 9500, 10500, 11000, 10200, 9200, 8500, 7000, 6000]
    +```
    diff --git a/packages/mermaid/src/mermaidAPI.spec.ts b/packages/mermaid/src/mermaidAPI.spec.ts
    index ce84212610..4a64e9c03f 100644
    --- a/packages/mermaid/src/mermaidAPI.spec.ts
    +++ b/packages/mermaid/src/mermaidAPI.spec.ts
    @@ -38,8 +38,6 @@ import type { MermaidConfig } from './config.type.js';
     
     import mermaidAPI, { removeExistingElements } from './mermaidAPI.js';
     import {
    -  encodeEntities,
    -  decodeEntities,
       createCssStyles,
       createUserStyles,
       appendDivSvgG,
    @@ -68,6 +66,7 @@ vi.mock('stylis', () => {
       };
     });
     import { compile, serialize } from 'stylis';
    +import { decodeEntities, encodeEntities } from './utils.js';
     
     /**
      * @see https://vitest.dev/guide/mocking.html Mock part of a module
    diff --git a/packages/mermaid/src/mermaidAPI.ts b/packages/mermaid/src/mermaidAPI.ts
    index 5250f0b190..09d0077803 100644
    --- a/packages/mermaid/src/mermaidAPI.ts
    +++ b/packages/mermaid/src/mermaidAPI.ts
    @@ -30,6 +30,7 @@ import isEmpty from 'lodash-es/isEmpty.js';
     import { setA11yDiagramInfo, addSVGa11yTitleDescription } from './accessibility.js';
     import type { DiagramStyleClassDef } from './diagram-api/types.js';
     import { preprocessDiagram } from './preprocess.js';
    +import { decodeEntities } from './utils.js';
     
     const MAX_TEXTLENGTH = 50_000;
     const MAX_TEXTLENGTH_EXCEEDED_MSG =
    @@ -110,43 +111,6 @@ async function parse(text: string, parseOptions?: ParseOptions): Promise=12.13.0'}
         dependencies:
    -      '@applitools/image': 1.0.34
    -      '@applitools/logger': 2.0.2
    -      '@applitools/req': 1.3.0
    -      '@applitools/utils': 1.3.37
    +      '@applitools/image': 1.1.2
    +      '@applitools/logger': 2.0.7
    +      '@applitools/req': 1.5.2
    +      '@applitools/utils': 1.5.0
           abort-controller: 3.0.0
           throat: 6.0.2
         transitivePeerDependencies:
           - supports-color
         dev: true
     
    -  /@applitools/core@3.2.1(typescript@5.1.3):
    -    resolution: {integrity: sha512-3Sq4r5cWrp/0hLxgb/AR+XdPTN+8AClZQGhqw4gze38BXeJO10FtxYhakOQ8iVuhUESwIHo86OCmLEGREZAroA==}
    +  /@applitools/core@3.9.0(typescript@5.1.6):
    +    resolution: {integrity: sha512-fKea8ew6iyLhZskUtngcyCdlJ59Nrb+8R9fU6Y6fXT0xMBQESkU1r9Z+Dt3XUL/CzRE9NW4DWenhd52EFApxYg==}
         engines: {node: '>=12.13.0'}
         hasBin: true
         dependencies:
    -      '@applitools/core-base': 1.1.56
    +      '@applitools/core-base': 1.5.0
           '@applitools/dom-capture': 11.2.2
    -      '@applitools/dom-snapshot': 4.7.9
    -      '@applitools/driver': 1.12.1
    -      '@applitools/ec-client': 1.2.32(typescript@5.1.3)
    -      '@applitools/logger': 2.0.2
    -      '@applitools/nml-client': 1.3.55
    -      '@applitools/req': 1.3.0
    -      '@applitools/screenshoter': 3.7.46
    -      '@applitools/snippets': 2.4.21
    -      '@applitools/socket': 1.1.1
    -      '@applitools/spec-driver-webdriver': 1.0.33(webdriver@7.30.0)
    -      '@applitools/ufg-client': 1.2.18
    -      '@applitools/utils': 1.3.37
    -      '@types/ws': 8.5.4
    +      '@applitools/dom-snapshot': 4.7.10
    +      '@applitools/driver': 1.13.4
    +      '@applitools/ec-client': 1.7.4(typescript@5.1.6)
    +      '@applitools/logger': 2.0.7
    +      '@applitools/nml-client': 1.5.7
    +      '@applitools/req': 1.5.2
    +      '@applitools/screenshoter': 3.8.7
    +      '@applitools/snippets': 2.4.22
    +      '@applitools/socket': 1.1.7
    +      '@applitools/spec-driver-webdriver': 1.0.41(webdriver@7.30.0)
    +      '@applitools/ufg-client': 1.7.0
    +      '@applitools/utils': 1.5.0
    +      '@types/ws': 8.5.5
           abort-controller: 3.0.0
           chalk: 4.1.2
    -      node-fetch: 2.6.7(encoding@0.1.13)
    -      webdriver: 7.30.0(typescript@5.1.3)
    -      ws: 8.12.0
    -      yargs: 17.6.2
    +      node-fetch: 2.6.7
    +      webdriver: 7.30.0(typescript@5.1.6)
    +      ws: 8.13.0
    +      yargs: 17.7.2
         transitivePeerDependencies:
           - bufferutil
    +      - canvas
           - encoding
           - supports-color
           - typescript
    @@ -790,8 +795,8 @@ packages:
         engines: {node: '>=8.9.0'}
         dev: true
     
    -  /@applitools/dom-snapshot@4.7.9:
    -    resolution: {integrity: sha512-lY1tkNwNQUBM7snYUwVZ80EisgIYdNZxIBtbsRU0R60wKTQc8ccBPGo9e3TBbS4Z9XqQYVAupKQjZMlcMVEiwQ==}
    +  /@applitools/dom-snapshot@4.7.10:
    +    resolution: {integrity: sha512-QhX0p6irvQE48eeauNHIfEm76L8QY8mDO8Tk4YOzzBRKcGpKphQUR/5GRCR9S3jx5wwJAwjF/aMW/W7Cwdaztw==}
         engines: {node: '>=8.9.0'}
         dependencies:
           '@applitools/dom-shared': 1.0.10
    @@ -800,41 +805,41 @@ packages:
           pako: 1.0.11
         dev: true
     
    -  /@applitools/driver@1.12.1:
    -    resolution: {integrity: sha512-/ku686w2BqCOVL1brBAgrGNGiFqPp+o9vUxmfNXTvsmrAYCsefyZ9R8APoRX0tQgNKr203F9UgaNDRRRmr+Uug==}
    +  /@applitools/driver@1.13.4:
    +    resolution: {integrity: sha512-LdATkjMoTZKUDHmuIfV0uh0ZiRe+yNNIehTqmjV6LnQrNvm73ddzF2Tn+LM8Vg/CflwFhw+TVKPaywTmi35wUg==}
         engines: {node: '>=12.13.0'}
         dependencies:
    -      '@applitools/logger': 2.0.2
    -      '@applitools/snippets': 2.4.21
    -      '@applitools/utils': 1.3.37
    -      semver: 7.3.7
    +      '@applitools/logger': 2.0.7
    +      '@applitools/snippets': 2.4.22
    +      '@applitools/utils': 1.5.0
    +      semver: 7.5.4
         transitivePeerDependencies:
           - supports-color
         dev: true
     
    -  /@applitools/ec-client@1.2.32(typescript@5.1.3):
    -    resolution: {integrity: sha512-03+qK41eza2808hXuc1D1UlJG+89oCF0EjJmP1Wkp2zq1nx6Za+R3NGVRTeUJhHhjOp6xdeqAHIyD8x1whCkmA==}
    +  /@applitools/ec-client@1.7.4(typescript@5.1.6):
    +    resolution: {integrity: sha512-vFY5O9WXQ905hUcw4ot1BuKAAFq6F+hyQfX/obsQsUPUvJXqHhiFjMiI/x3cyrPr40EVLQM8edZCa71m4k2WyQ==}
         engines: {node: '>=12.13.0'}
         hasBin: true
         dependencies:
    -      '@applitools/core-base': 1.1.56
    -      '@applitools/driver': 1.12.1
    -      '@applitools/execution-grid-tunnel': 2.1.0
    -      '@applitools/logger': 2.0.2
    -      '@applitools/req': 1.3.0
    -      '@applitools/socket': 1.1.1
    -      '@applitools/spec-driver-webdriver': 1.0.33(webdriver@7.30.0)
    -      '@applitools/utils': 1.3.37
    +      '@applitools/core-base': 1.5.0
    +      '@applitools/driver': 1.13.4
    +      '@applitools/logger': 2.0.7
    +      '@applitools/req': 1.5.2
    +      '@applitools/socket': 1.1.7
    +      '@applitools/spec-driver-webdriver': 1.0.41(webdriver@7.30.0)
    +      '@applitools/tunnel-client': 1.1.3
    +      '@applitools/utils': 1.5.0
           abort-controller: 3.0.0
    -      webdriver: 7.30.0(typescript@5.1.3)
    -      yargs: 17.6.2
    +      webdriver: 7.30.0(typescript@5.1.6)
    +      yargs: 17.7.2
         transitivePeerDependencies:
           - supports-color
           - typescript
         dev: true
     
    -  /@applitools/eg-frpc@1.0.3:
    -    resolution: {integrity: sha512-16CrVdq2onkN5j6wpHxd8dfapJehHJq7GnYEg86QGFZPhTgZI5WukKrp9ryM0EblUJUjdTwEPymc5B8cffuZZQ==}
    +  /@applitools/eg-frpc@1.0.5:
    +    resolution: {integrity: sha512-9qUNiCK3R3VKxIAaLr5HO5QnUx6TioLFkJ2JcpU1ZqefApt1X2bdfS7eA4TGDXDWv/a0OIl2Lddzuo5/h3vbTw==}
         engines: {node: '>=12.13.0'}
         dev: true
     
    @@ -846,12 +851,12 @@ packages:
           is-localhost-ip: 2.0.0
         dev: true
     
    -  /@applitools/execution-grid-tunnel@2.1.0:
    -    resolution: {integrity: sha512-OooDCcS93+reh1hIpalDZVMhAsSaG/h+T5jGn7WzIYL4wVdi/GLyibvzxlY1hLFGi0EhUJGckpMoHiTZ6EuV2w==}
    +  /@applitools/execution-grid-tunnel@2.1.8:
    +    resolution: {integrity: sha512-MRjh2q9ZNGdW2CJX4w3xB7yNX4mLoTRBG1VxYD+U3n7bNdeAFwiHZAgkIRgLDMHHJJXoNh0xEIPe6aB+9KuCIg==}
         engines: {node: '>=12.13.0'}
         hasBin: true
         dependencies:
    -      '@applitools/eg-frpc': 1.0.3
    +      '@applitools/eg-frpc': 1.0.5
           '@applitools/eg-socks5-proxy-server': 0.5.4
           '@applitools/logger': 1.1.53
           dotenv: 16.3.1
    @@ -861,23 +866,23 @@ packages:
           find-process: 1.4.7
           ini: 3.0.1
           node-cleanup: 2.1.2
    -      node-fetch: 2.6.7(encoding@0.1.13)
    +      node-fetch: 2.6.12(encoding@0.1.13)
           p-retry: 4.6.2
           teen_process: 1.16.0
         transitivePeerDependencies:
           - supports-color
         dev: true
     
    -  /@applitools/eyes-cypress@3.33.1(typescript@5.1.3):
    -    resolution: {integrity: sha512-hIvXiz/8xLarvnx78QuMm2NCkLYXgF/CC/X77dAAQO54tFgDmSsDaB9ak1cE5bDMCLxP84+roRghgJUQxpersQ==}
    +  /@applitools/eyes-cypress@3.36.2(typescript@5.1.6):
    +    resolution: {integrity: sha512-GA1KP7i3Zr7sbc8yscw7fay1XbQ46LxEAH4dLqjJhOmvvpZuDlmgRyVMuvTmobDXKKHtdVpfoXtJQURrxs7fvA==}
         engines: {node: '>=12.13.0'}
         hasBin: true
         dependencies:
    -      '@applitools/core': 3.2.1(typescript@5.1.3)
    -      '@applitools/eyes': 1.2.14(typescript@5.1.3)
    +      '@applitools/core': 3.9.0(typescript@5.1.6)
    +      '@applitools/eyes': 1.7.2(typescript@5.1.6)
           '@applitools/functional-commons': 1.6.0
    -      '@applitools/logger': 2.0.2
    -      '@applitools/utils': 1.3.37
    +      '@applitools/logger': 2.0.7
    +      '@applitools/utils': 1.5.0
           boxen: 5.1.2
           chalk: 3.0.0
           semver: 7.3.8
    @@ -885,21 +890,23 @@ packages:
           ws: 8.5.0
         transitivePeerDependencies:
           - bufferutil
    +      - canvas
           - encoding
           - supports-color
           - typescript
           - utf-8-validate
         dev: true
     
    -  /@applitools/eyes@1.2.14(typescript@5.1.3):
    -    resolution: {integrity: sha512-pH9g3h2pQCGfrduT+gAZTyQ/iM+NxSmw3e8UjI+2/Zj0VnnlhprX18x5mT062tEZA6WExn59mLWsdZQcu1cDnA==}
    +  /@applitools/eyes@1.7.2(typescript@5.1.6):
    +    resolution: {integrity: sha512-RNwPMp19xmQ+oEEZH66wGbnBbOqsrHaLBwnCZ9qJ9294aN4DuYHJNXdzHtLFSwjHNc8UmyDRU2Enn4825hqV6w==}
         engines: {node: '>=12.13.0'}
         dependencies:
    -      '@applitools/core': 3.2.1(typescript@5.1.3)
    -      '@applitools/logger': 2.0.2
    -      '@applitools/utils': 1.3.37
    +      '@applitools/core': 3.9.0(typescript@5.1.6)
    +      '@applitools/logger': 2.0.7
    +      '@applitools/utils': 1.5.0
         transitivePeerDependencies:
           - bufferutil
    +      - canvas
           - encoding
           - supports-color
           - typescript
    @@ -911,53 +918,17 @@ packages:
         engines: {node: '>=8.0.0'}
         dev: true
     
    -  /@applitools/image@1.0.34:
    -    resolution: {integrity: sha512-PkgsC0bMrQl5XSsnWBz4rPxI1GeLr5OkV2X6Wr//HJPl/RpvKJYWVrgdwfA6Z+T10RETRqcxJAX5grvfytymog==}
    +  /@applitools/image@1.1.2:
    +    resolution: {integrity: sha512-Cy1oKCB2vIpHT47Y1tictsRS2RLBVI4XzxYWvKnXx+ZsbL364IiDzwWxYKgvA7/6t1Ako648n4+BWKoUi5Ewbg==}
         engines: {node: '>=12.13.0'}
         dependencies:
    -      '@applitools/utils': 1.3.37
    +      '@applitools/utils': 1.5.0
           bmpimagejs: 1.0.4
           jpeg-js: 0.4.4
    +      omggif: 1.0.10
           png-async: 0.9.4
         dev: true
     
    -  /@applitools/jsdom@1.0.4:
    -    resolution: {integrity: sha512-JtjNfTJtphJYHEkicW4xlwtYuRP3TRvjoszfkrcpxTNMCbGkbop8ed9MuUfR83dAZj5NY9begbmEqJohLJco6w==}
    -    engines: {node: '>=12'}
    -    dependencies:
    -      abab: 2.0.6
    -      acorn: 8.10.0
    -      acorn-globals: 6.0.0
    -      cssom: 0.5.0
    -      cssstyle: 2.3.0
    -      data-urls: 3.0.2
    -      decimal.js: 10.4.3
    -      domexception: 4.0.0
    -      escodegen: 2.1.0
    -      form-data: 4.0.0
    -      html-encoding-sniffer: 3.0.0
    -      http-proxy-agent: 5.0.0
    -      https-proxy-agent: 5.0.1
    -      is-potential-custom-element-name: 1.0.1
    -      nwsapi: 2.2.7
    -      parse5: 6.0.1
    -      saxes: 5.0.1
    -      symbol-tree: 3.2.4
    -      tough-cookie: 4.1.3
    -      w3c-hr-time: 1.0.2
    -      w3c-xmlserializer: 3.0.0
    -      webidl-conversions: 7.0.0
    -      whatwg-encoding: 2.0.0
    -      whatwg-mimetype: 3.0.0
    -      whatwg-url: 10.0.0
    -      ws: 8.12.0
    -      xml-name-validator: 4.0.0
    -    transitivePeerDependencies:
    -      - bufferutil
    -      - supports-color
    -      - utf-8-validate
    -    dev: true
    -
       /@applitools/logger@1.1.53:
         resolution: {integrity: sha512-4mlzYxc0MgM3WIxEwKqIjn9W7G7kMtQc2bFRxozViKOXypTfr72j8iODs88wcetP0GsXtplhZQ5/6aZN5WY9ug==}
         engines: {node: '>=12.13.0'}
    @@ -969,33 +940,33 @@ packages:
           - supports-color
         dev: true
     
    -  /@applitools/logger@2.0.2:
    -    resolution: {integrity: sha512-tprqwPbP0q4bQ1wPJaLCb2nmFywRPJlwEADWJJ5oKJBhbvpqEKo6542GNp4DIebugLy99HUvkCTQS8ngKG0jdg==}
    +  /@applitools/logger@2.0.7:
    +    resolution: {integrity: sha512-dmX2nWWixMYsOdhl1MANv7wr8cKzYUOaHxQp9CdokVbJy+NGwWAzK6qVeKjogn7D6eXHzgn3R3OzplYSq/fK/g==}
         engines: {node: '>=12.13.0'}
         dependencies:
    -      '@applitools/utils': 1.3.37
    +      '@applitools/utils': 1.5.0
           chalk: 4.1.2
    -      debug: 4.3.3
    +      debug: 4.3.4(supports-color@8.1.1)
         transitivePeerDependencies:
           - supports-color
         dev: true
     
    -  /@applitools/nml-client@1.3.55:
    -    resolution: {integrity: sha512-r9C2fByQ0ce228LguFl/Q1RHjyhZZM+BOxOQosjt2NI5fW2NEByrjyfBrIU2JnZzArQjv8j6tX4ZgrpOXVpXxQ==}
    +  /@applitools/nml-client@1.5.7:
    +    resolution: {integrity: sha512-jAG2/4JSqX7FSrdyIyjdXcPy2l2/t8KRlw554nL1ABgtqTV8yCI3DxVUk7RCNi39f1uY5pA2pfiZVGFImnSFKg==}
         engines: {node: '>=12.13.0'}
         dependencies:
    -      '@applitools/logger': 2.0.2
    -      '@applitools/req': 1.3.0
    -      '@applitools/utils': 1.3.37
    +      '@applitools/logger': 2.0.7
    +      '@applitools/req': 1.5.2
    +      '@applitools/utils': 1.5.0
         transitivePeerDependencies:
           - supports-color
         dev: true
     
    -  /@applitools/req@1.3.0:
    -    resolution: {integrity: sha512-DwoWjMuP9VKpdCMqg91ItrQNKsjpGuYGAqvFitOpu4bs1kP0lcZHOHBuiHyNj97oUccJ1TT3RRrzrkA+byWseg==}
    +  /@applitools/req@1.5.2:
    +    resolution: {integrity: sha512-evuikeiCYudhxSQ2kisO7DdywPqshaaN+BiDu4P3eTz5R9VmqhXwWP9bS88G8bzzz0FeNQMY9a7TjQ7d5jMRrA==}
         engines: {node: '>=12.13.0'}
         dependencies:
    -      '@applitools/utils': 1.3.37
    +      '@applitools/utils': 1.5.0
           abort-controller: 3.0.0
           http-proxy-agent: 5.0.0
           https-proxy-agent: 5.0.1
    @@ -1004,63 +975,79 @@ packages:
           - supports-color
         dev: true
     
    -  /@applitools/screenshoter@3.7.46:
    -    resolution: {integrity: sha512-VOqzXm3BM4yKWIHxOZGFd+3NVyK9CgEYCxoywFqO+2G5dhYl2r51r60VKF3hS4A8ySXXqkJ878kaRceI3s8YNQ==}
    +  /@applitools/screenshoter@3.8.7:
    +    resolution: {integrity: sha512-G576fLyTTAJEnhFZBeD57+1JDXGTDcTlrg0n32ujtYTFswUAf5XnXmeO6s2WqeHKQl74e2xwhBmdtU/CrVOkig==}
         engines: {node: '>=12.13.0'}
         dependencies:
    -      '@applitools/image': 1.0.34
    -      '@applitools/logger': 2.0.2
    -      '@applitools/snippets': 2.4.21
    -      '@applitools/utils': 1.3.37
    -      jpeg-js: 0.4.4
    -      png-async: 0.9.4
    +      '@applitools/image': 1.1.2
    +      '@applitools/logger': 2.0.7
    +      '@applitools/snippets': 2.4.22
    +      '@applitools/utils': 1.5.0
         transitivePeerDependencies:
           - supports-color
         dev: true
     
    -  /@applitools/snippets@2.4.21:
    -    resolution: {integrity: sha512-3l+6pR0cJJjpG5subgqoI55vFFZ94//CS7jdhWuVwUtQJzeq6QSD6m/H+qfa8A7Sg9mzXnbyUenWQ7fpUk/5Fg==}
    +  /@applitools/snippets@2.4.22:
    +    resolution: {integrity: sha512-bv4GzMf6k4mAyMxo3PVR3HKEPkf4h0O6+xNo6UO78cw+MpkT4Sr7JDk3mLq8H/WRDKk7x4VKpJeoDHd5dBxT3g==}
         engines: {node: '>=12.13.0'}
         dev: true
     
    -  /@applitools/socket@1.1.1:
    -    resolution: {integrity: sha512-lQjglDC8r2QOVAQmZ+dqsTrtHMQ/9uAGsoMpFQo9amf+/DOipIwLKo0kGx+bMYmfDnAChlifukmMvrl9o1NcTQ==}
    +  /@applitools/socket@1.1.7:
    +    resolution: {integrity: sha512-SpP+Zw5B9VJ3K+xW+wSYDwfrOQ1U9/95h5G3rszKaVleX2FTUW0JgmASuSlwgr7veU3qlcNzt3vas/tQM3/z/g==}
         engines: {node: '>=12.13.0'}
         dependencies:
    -      '@applitools/logger': 2.0.2
    -      '@applitools/utils': 1.3.37
    +      '@applitools/logger': 2.0.7
    +      '@applitools/utils': 1.5.0
         transitivePeerDependencies:
           - supports-color
         dev: true
     
    -  /@applitools/spec-driver-webdriver@1.0.33(webdriver@7.30.0):
    -    resolution: {integrity: sha512-5QbP945AyS5CjvqQ9N0d8UfnVKDjOsEhennm++W3xNBaBAGIQia2yP5xT3DFuEBH2fZCxJE+Cd/X1X4IbqGdfg==}
    +  /@applitools/spec-driver-webdriver@1.0.41(webdriver@7.30.0):
    +    resolution: {integrity: sha512-0kUHiFmr3FiOlfTnfS1k8pvJXgnEM8bP36teiDJvmAJnk8aJG5nmqaQj1KkeLUVVZ1wfYlg/+iDtGUdP4a4Zxw==}
         engines: {node: '>=12.13.0'}
         peerDependencies:
           webdriver: '>=7.27.0'
         dependencies:
    -      '@applitools/driver': 1.12.1
    -      '@applitools/utils': 1.3.37
    +      '@applitools/driver': 1.13.4
    +      '@applitools/utils': 1.5.0
           http-proxy-agent: 5.0.0
           https-proxy-agent: 5.0.1
    -      webdriver: 7.30.0(typescript@5.1.3)
    +      webdriver: 7.30.0(typescript@5.1.6)
    +    transitivePeerDependencies:
    +      - supports-color
    +    dev: true
    +
    +  /@applitools/tunnel-client@1.1.3:
    +    resolution: {integrity: sha512-xe0HqznqnuhsZYIY//NnjszEkIcYgdZkwVR4GOwzVCOCcxIGoi+kMpDUuC2Xcd8+UiVbfZfZTeo7rXITlzzqAw==}
    +    engines: {node: '>=12.13.0'}
    +    hasBin: true
    +    dependencies:
    +      '@applitools/execution-grid-tunnel': 2.1.8
    +      '@applitools/logger': 2.0.7
    +      '@applitools/req': 1.5.2
    +      '@applitools/socket': 1.1.7
    +      '@applitools/utils': 1.5.0
    +      abort-controller: 3.0.0
    +      yargs: 17.7.2
         transitivePeerDependencies:
           - supports-color
         dev: true
     
    -  /@applitools/ufg-client@1.2.18:
    -    resolution: {integrity: sha512-2JzvCfJVtzyI4++jE6QDwm3nwIpuhaXVP1+BuqRnWFrwVFdwDwg24MkrF/WMX2gq6+x8tad7KXZ0rkQFhrQw5Q==}
    +  /@applitools/ufg-client@1.7.0:
    +    resolution: {integrity: sha512-BMFLuWGq8YVs0/z5VBl8CAKzOMXbagHFxyR4oGdg31nDsuKgKfhlnbGji5qek243ex8vEbEqFwsH2K0ZXYGIeQ==}
         engines: {node: '>=12.13.0'}
         dependencies:
    -      '@applitools/jsdom': 1.0.4
    -      '@applitools/logger': 2.0.2
    -      '@applitools/req': 1.3.0
    -      '@applitools/utils': 1.3.37
    +      '@applitools/image': 1.1.2
    +      '@applitools/logger': 2.0.7
    +      '@applitools/req': 1.5.2
    +      '@applitools/utils': 1.5.0
           abort-controller: 3.0.0
           css-tree: 2.3.1
    -      throat: 6.0.1
    +      jsdom: 19.0.0
    +      throat: 6.0.2
         transitivePeerDependencies:
           - bufferutil
    +      - canvas
           - supports-color
           - utf-8-validate
         dev: true
    @@ -1070,8 +1057,8 @@ packages:
         engines: {node: '>=12.13.0'}
         dev: true
     
    -  /@applitools/utils@1.3.37:
    -    resolution: {integrity: sha512-w9RNmLoVg3BwOYCFOykqmM6pPgpgq3QuI3mUMSK3VLnOmkaz2bLqozTY7xOLnW4NACg/1hdpbVbrb8mygHYoEA==}
    +  /@applitools/utils@1.5.0:
    +    resolution: {integrity: sha512-BZk8YolP0G+/Srjkhf+pFp4zY7bU41L63hDN9gtwrD1xZOfWXJbOCD3gFQAGRB2qsozHMkPNTt+xw7RJjIjGQg==}
         engines: {node: '>=12.13.0'}
         dev: true
     
    @@ -1088,6 +1075,14 @@ packages:
           chalk: 2.4.2
         dev: true
     
    +  /@babel/code-frame@7.22.13:
    +    resolution: {integrity: sha512-XktuhWlJ5g+3TJXc5upd9Ks1HutSArik6jf2eAjYFyIOf4ej3RN+184cZbzDvbPnuTJIUhPKKJE3cIsYTiAT3w==}
    +    engines: {node: '>=6.9.0'}
    +    dependencies:
    +      '@babel/highlight': 7.22.20
    +      chalk: 2.4.2
    +    dev: true
    +
       /@babel/compat-data@7.22.9:
         resolution: {integrity: sha512-5UamI7xkUcJ3i9qVDS+KFDEK8/7oJ55/sJMB1Ge7IEapr7KfdfV/HErR+koZwOfd+SgtFKOKRhRakdg++DcJpQ==}
         engines: {node: '>=6.9.0'}
    @@ -1103,10 +1098,10 @@ packages:
           '@babel/helper-compilation-targets': 7.22.10
           '@babel/helper-module-transforms': 7.22.9(@babel/core@7.22.10)
           '@babel/helpers': 7.22.10
    -      '@babel/parser': 7.22.10
    +      '@babel/parser': 7.23.0
           '@babel/template': 7.22.5
    -      '@babel/traverse': 7.22.10
    -      '@babel/types': 7.22.10
    +      '@babel/traverse': 7.23.2
    +      '@babel/types': 7.23.0
           convert-source-map: 1.9.0
           debug: 4.3.4(supports-color@8.1.1)
           gensync: 1.0.0-beta.2
    @@ -1120,7 +1115,17 @@ packages:
         resolution: {integrity: sha512-79KIf7YiWjjdZ81JnLujDRApWtl7BxTqWD88+FFdQEIOG8LJ0etDOM7CXuIgGJa55sGOwZVwuEsaLEm0PJ5/+A==}
         engines: {node: '>=6.9.0'}
         dependencies:
    -      '@babel/types': 7.22.10
    +      '@babel/types': 7.23.0
    +      '@jridgewell/gen-mapping': 0.3.3
    +      '@jridgewell/trace-mapping': 0.3.19
    +      jsesc: 2.5.2
    +    dev: true
    +
    +  /@babel/generator@7.23.0:
    +    resolution: {integrity: sha512-lN85QRR+5IbYrMWM6Y4pE/noaQtg4pNiqeNGX60eqOfo6gtEj6uw/JagelB8vVztSd7R6M5n1+PQkDbHbBRU4g==}
    +    engines: {node: '>=6.9.0'}
    +    dependencies:
    +      '@babel/types': 7.23.0
           '@jridgewell/gen-mapping': 0.3.3
           '@jridgewell/trace-mapping': 0.3.19
           jsesc: 2.5.2
    @@ -1130,14 +1135,14 @@ packages:
         resolution: {integrity: sha512-LvBTxu8bQSQkcyKOU+a1btnNFQ1dMAd0R6PyW3arXes06F6QLWLIrd681bxRPIXlrMGR3XYnW9JyML7dP3qgxg==}
         engines: {node: '>=6.9.0'}
         dependencies:
    -      '@babel/types': 7.22.10
    +      '@babel/types': 7.23.0
         dev: true
     
       /@babel/helper-builder-binary-assignment-operator-visitor@7.22.10:
         resolution: {integrity: sha512-Av0qubwDQxC56DoUReVDeLfMEjYYSN1nZrTUrWkXd7hpU73ymRANkbuDm3yni9npkn+RXy9nNbEJZEzXr7xrfQ==}
         engines: {node: '>=6.9.0'}
         dependencies:
    -      '@babel/types': 7.22.10
    +      '@babel/types': 7.23.0
         dev: true
     
       /@babel/helper-compilation-targets@7.22.10:
    @@ -1159,8 +1164,8 @@ packages:
         dependencies:
           '@babel/core': 7.22.10
           '@babel/helper-annotate-as-pure': 7.22.5
    -      '@babel/helper-environment-visitor': 7.22.5
    -      '@babel/helper-function-name': 7.22.5
    +      '@babel/helper-environment-visitor': 7.22.20
    +      '@babel/helper-function-name': 7.23.0
           '@babel/helper-member-expression-to-functions': 7.22.5
           '@babel/helper-optimise-call-expression': 7.22.5
           '@babel/helper-replace-supers': 7.22.9(@babel/core@7.22.10)
    @@ -1196,38 +1201,43 @@ packages:
           - supports-color
         dev: true
     
    +  /@babel/helper-environment-visitor@7.22.20:
    +    resolution: {integrity: sha512-zfedSIzFhat/gFhWfHtgWvlec0nqB9YEIVrpuwjruLlXfUSnA8cJB0miHKwqDnQ7d32aKo2xt88/xZptwxbfhA==}
    +    engines: {node: '>=6.9.0'}
    +    dev: true
    +
       /@babel/helper-environment-visitor@7.22.5:
         resolution: {integrity: sha512-XGmhECfVA/5sAt+H+xpSg0mfrHq6FzNr9Oxh7PSEBBRUb/mL7Kz3NICXb194rCqAEdxkhPT1a88teizAFyvk8Q==}
         engines: {node: '>=6.9.0'}
         dev: true
     
    -  /@babel/helper-function-name@7.22.5:
    -    resolution: {integrity: sha512-wtHSq6jMRE3uF2otvfuD3DIvVhOsSNshQl0Qrd7qC9oQJzHvOL4qQXlQn2916+CXGywIjpGuIkoyZRRxHPiNQQ==}
    +  /@babel/helper-function-name@7.23.0:
    +    resolution: {integrity: sha512-OErEqsrxjZTJciZ4Oo+eoZqeW9UIiOcuYKRJA4ZAgV9myA+pOXhhmpfNCKjEH/auVfEYVFJ6y1Tc4r0eIApqiw==}
         engines: {node: '>=6.9.0'}
         dependencies:
    -      '@babel/template': 7.22.5
    -      '@babel/types': 7.22.10
    +      '@babel/template': 7.22.15
    +      '@babel/types': 7.23.0
         dev: true
     
       /@babel/helper-hoist-variables@7.22.5:
         resolution: {integrity: sha512-wGjk9QZVzvknA6yKIUURb8zY3grXCcOZt+/7Wcy8O2uctxhplmUPkOdlgoNhmdVee2c92JXbf1xpMtVNbfoxRw==}
         engines: {node: '>=6.9.0'}
         dependencies:
    -      '@babel/types': 7.22.10
    +      '@babel/types': 7.23.0
         dev: true
     
       /@babel/helper-member-expression-to-functions@7.22.5:
         resolution: {integrity: sha512-aBiH1NKMG0H2cGZqspNvsaBe6wNGjbJjuLy29aU+eDZjSbbN53BaxlpB02xm9v34pLTZ1nIQPFYn2qMZoa5BQQ==}
         engines: {node: '>=6.9.0'}
         dependencies:
    -      '@babel/types': 7.22.10
    +      '@babel/types': 7.23.0
         dev: true
     
       /@babel/helper-module-imports@7.22.5:
         resolution: {integrity: sha512-8Dl6+HD/cKifutF5qGd/8ZJi84QeAKh+CEe1sBzz8UayBBGg1dAIJrdHOcOM5b2MpzWL2yuotJTtGjETq0qjXg==}
         engines: {node: '>=6.9.0'}
         dependencies:
    -      '@babel/types': 7.22.10
    +      '@babel/types': 7.23.0
         dev: true
     
       /@babel/helper-module-transforms@7.22.9(@babel/core@7.22.10):
    @@ -1241,14 +1251,14 @@ packages:
           '@babel/helper-module-imports': 7.22.5
           '@babel/helper-simple-access': 7.22.5
           '@babel/helper-split-export-declaration': 7.22.6
    -      '@babel/helper-validator-identifier': 7.22.5
    +      '@babel/helper-validator-identifier': 7.22.20
         dev: true
     
       /@babel/helper-optimise-call-expression@7.22.5:
         resolution: {integrity: sha512-HBwaojN0xFRx4yIvpwGqxiV2tUfl7401jlok564NgB9EHS1y6QT17FmKWm4ztqjeVdXLuC4fSvHc5ePpQjoTbw==}
         engines: {node: '>=6.9.0'}
         dependencies:
    -      '@babel/types': 7.22.10
    +      '@babel/types': 7.23.0
         dev: true
     
       /@babel/helper-plugin-utils@7.22.5:
    @@ -1264,7 +1274,7 @@ packages:
         dependencies:
           '@babel/core': 7.22.10
           '@babel/helper-annotate-as-pure': 7.22.5
    -      '@babel/helper-environment-visitor': 7.22.5
    +      '@babel/helper-environment-visitor': 7.22.20
           '@babel/helper-wrap-function': 7.22.10
         dev: true
     
    @@ -1275,7 +1285,7 @@ packages:
           '@babel/core': ^7.0.0
         dependencies:
           '@babel/core': 7.22.10
    -      '@babel/helper-environment-visitor': 7.22.5
    +      '@babel/helper-environment-visitor': 7.22.20
           '@babel/helper-member-expression-to-functions': 7.22.5
           '@babel/helper-optimise-call-expression': 7.22.5
         dev: true
    @@ -1284,30 +1294,35 @@ packages:
         resolution: {integrity: sha512-n0H99E/K+Bika3++WNL17POvo4rKWZ7lZEp1Q+fStVbUi8nxPQEBOlTmCOxW/0JsS56SKKQ+ojAe2pHKJHN35w==}
         engines: {node: '>=6.9.0'}
         dependencies:
    -      '@babel/types': 7.22.10
    +      '@babel/types': 7.23.0
         dev: true
     
       /@babel/helper-skip-transparent-expression-wrappers@7.22.5:
         resolution: {integrity: sha512-tK14r66JZKiC43p8Ki33yLBVJKlQDFoA8GYN67lWCDCqoL6EMMSuM9b+Iff2jHaM/RRFYl7K+iiru7hbRqNx8Q==}
         engines: {node: '>=6.9.0'}
         dependencies:
    -      '@babel/types': 7.22.10
    +      '@babel/types': 7.23.0
         dev: true
     
       /@babel/helper-split-export-declaration@7.22.6:
         resolution: {integrity: sha512-AsUnxuLhRYsisFiaJwvp1QF+I3KjD5FOxut14q/GzovUe6orHLesW2C7d754kRm53h5gqrz6sFl6sxc4BVtE/g==}
         engines: {node: '>=6.9.0'}
         dependencies:
    -      '@babel/types': 7.22.10
    +      '@babel/types': 7.23.0
         dev: true
     
       /@babel/helper-string-parser@7.22.5:
         resolution: {integrity: sha512-mM4COjgZox8U+JcXQwPijIZLElkgEpO5rsERVDJTc2qfCDfERyob6k5WegS14SX18IIjv+XD+GrqNumY5JRCDw==}
         engines: {node: '>=6.9.0'}
     
    +  /@babel/helper-validator-identifier@7.22.20:
    +    resolution: {integrity: sha512-Y4OZ+ytlatR8AI+8KZfKuL5urKp7qey08ha31L8b3BwewJAoJamTzyvxPR/5D+KkdJCGPq/+8TukHBlY10FX9A==}
    +    engines: {node: '>=6.9.0'}
    +
       /@babel/helper-validator-identifier@7.22.5:
         resolution: {integrity: sha512-aJXu+6lErq8ltp+JhkJUfk1MTGyuA4v7f3pA+BJ5HLfNC6nAQ0Cpi9uOquUj8Hehg0aUiHzWQbOVJGao6ztBAQ==}
         engines: {node: '>=6.9.0'}
    +    dev: true
     
       /@babel/helper-validator-option@7.22.5:
         resolution: {integrity: sha512-R3oB6xlIVKUnxNUxbmgq7pKjxpru24zlimpE8WK47fACIlM0II/Hm1RS8IaOI7NgCr6LNS+jl5l75m20npAziw==}
    @@ -1318,9 +1333,9 @@ packages:
         resolution: {integrity: sha512-OnMhjWjuGYtdoO3FmsEFWvBStBAe2QOgwOLsLNDjN+aaiMD8InJk1/O3HSD8lkqTjCgg5YI34Tz15KNNA3p+nQ==}
         engines: {node: '>=6.9.0'}
         dependencies:
    -      '@babel/helper-function-name': 7.22.5
    -      '@babel/template': 7.22.5
    -      '@babel/types': 7.22.10
    +      '@babel/helper-function-name': 7.23.0
    +      '@babel/template': 7.22.15
    +      '@babel/types': 7.23.0
         dev: true
     
       /@babel/helpers@7.22.10:
    @@ -1328,8 +1343,8 @@ packages:
         engines: {node: '>=6.9.0'}
         dependencies:
           '@babel/template': 7.22.5
    -      '@babel/traverse': 7.22.10
    -      '@babel/types': 7.22.10
    +      '@babel/traverse': 7.23.2
    +      '@babel/types': 7.23.0
         transitivePeerDependencies:
           - supports-color
         dev: true
    @@ -1338,7 +1353,16 @@ packages:
         resolution: {integrity: sha512-78aUtVcT7MUscr0K5mIEnkwxPE0MaxkR5RxRwuHaQ+JuU5AmTPhY+do2mdzVTnIJJpyBglql2pehuBIWHug+WQ==}
         engines: {node: '>=6.9.0'}
         dependencies:
    -      '@babel/helper-validator-identifier': 7.22.5
    +      '@babel/helper-validator-identifier': 7.22.20
    +      chalk: 2.4.2
    +      js-tokens: 4.0.0
    +    dev: true
    +
    +  /@babel/highlight@7.22.20:
    +    resolution: {integrity: sha512-dkdMCN3py0+ksCgYmGG8jKeGA/8Tk+gJwSYYlFGxG5lmhfKNoAy004YpLxpS1W2J8m/EK2Ew+yOs9pVRwO89mg==}
    +    engines: {node: '>=6.9.0'}
    +    dependencies:
    +      '@babel/helper-validator-identifier': 7.22.20
           chalk: 2.4.2
           js-tokens: 4.0.0
         dev: true
    @@ -1348,7 +1372,14 @@ packages:
         engines: {node: '>=6.0.0'}
         hasBin: true
         dependencies:
    -      '@babel/types': 7.22.10
    +      '@babel/types': 7.23.0
    +
    +  /@babel/parser@7.23.0:
    +    resolution: {integrity: sha512-vvPKKdMemU85V9WE/l5wZEmImpCtLqbnTvqDS2U1fJ96KrxoW7KrXhNsNCblQlg8Ck4b85yxdTyelsMUgFUXiw==}
    +    engines: {node: '>=6.0.0'}
    +    hasBin: true
    +    dependencies:
    +      '@babel/types': 7.23.0
     
       /@babel/plugin-bugfix-safari-id-destructuring-collision-in-function-expression@7.22.5(@babel/core@7.22.10):
         resolution: {integrity: sha512-NP1M5Rf+u2Gw9qfSO4ihjcTGW5zXTi36ITLd4/EoAcEhIZ0yjMqmftDNl3QC19CX7olhrjpyU454g/2W7X0jvQ==}
    @@ -1596,7 +1627,7 @@ packages:
           '@babel/core': ^7.0.0-0
         dependencies:
           '@babel/core': 7.22.10
    -      '@babel/helper-environment-visitor': 7.22.5
    +      '@babel/helper-environment-visitor': 7.22.20
           '@babel/helper-plugin-utils': 7.22.5
           '@babel/helper-remap-async-to-generator': 7.22.9(@babel/core@7.22.10)
           '@babel/plugin-syntax-async-generators': 7.8.4(@babel/core@7.22.10)
    @@ -1666,8 +1697,8 @@ packages:
           '@babel/core': 7.22.10
           '@babel/helper-annotate-as-pure': 7.22.5
           '@babel/helper-compilation-targets': 7.22.10
    -      '@babel/helper-environment-visitor': 7.22.5
    -      '@babel/helper-function-name': 7.22.5
    +      '@babel/helper-environment-visitor': 7.22.20
    +      '@babel/helper-function-name': 7.23.0
           '@babel/helper-optimise-call-expression': 7.22.5
           '@babel/helper-plugin-utils': 7.22.5
           '@babel/helper-replace-supers': 7.22.9(@babel/core@7.22.10)
    @@ -1683,7 +1714,7 @@ packages:
         dependencies:
           '@babel/core': 7.22.10
           '@babel/helper-plugin-utils': 7.22.5
    -      '@babel/template': 7.22.5
    +      '@babel/template': 7.22.15
         dev: true
     
       /@babel/plugin-transform-destructuring@7.22.10(@babel/core@7.22.10):
    @@ -1768,7 +1799,7 @@ packages:
         dependencies:
           '@babel/core': 7.22.10
           '@babel/helper-compilation-targets': 7.22.10
    -      '@babel/helper-function-name': 7.22.5
    +      '@babel/helper-function-name': 7.23.0
           '@babel/helper-plugin-utils': 7.22.5
         dev: true
     
    @@ -1847,7 +1878,7 @@ packages:
           '@babel/helper-hoist-variables': 7.22.5
           '@babel/helper-module-transforms': 7.22.9(@babel/core@7.22.10)
           '@babel/helper-plugin-utils': 7.22.5
    -      '@babel/helper-validator-identifier': 7.22.5
    +      '@babel/helper-validator-identifier': 7.22.20
         dev: true
     
       /@babel/plugin-transform-modules-umd@7.22.5(@babel/core@7.22.10):
    @@ -2192,11 +2223,11 @@ packages:
           '@babel/plugin-transform-unicode-regex': 7.22.5(@babel/core@7.22.10)
           '@babel/plugin-transform-unicode-sets-regex': 7.22.5(@babel/core@7.22.10)
           '@babel/preset-modules': 0.1.6-no-external-plugins(@babel/core@7.22.10)
    -      '@babel/types': 7.22.10
    +      '@babel/types': 7.23.0
           babel-plugin-polyfill-corejs2: 0.4.5(@babel/core@7.22.10)
           babel-plugin-polyfill-corejs3: 0.8.3(@babel/core@7.22.10)
           babel-plugin-polyfill-regenerator: 0.5.2(@babel/core@7.22.10)
    -      core-js-compat: 3.32.1
    +      core-js-compat: 3.32.0
           semver: 6.3.1
         transitivePeerDependencies:
           - supports-color
    @@ -2209,7 +2240,7 @@ packages:
         dependencies:
           '@babel/core': 7.22.10
           '@babel/helper-plugin-utils': 7.22.5
    -      '@babel/types': 7.22.10
    +      '@babel/types': 7.23.0
           esutils: 2.0.3
         dev: true
     
    @@ -2224,39 +2255,48 @@ packages:
           regenerator-runtime: 0.14.0
         dev: true
     
    +  /@babel/template@7.22.15:
    +    resolution: {integrity: sha512-QPErUVm4uyJa60rkI73qneDacvdvzxshT3kksGqlGWYdOTIUOwJ7RDUL8sGqslY1uXWSL6xMFKEXDS3ox2uF0w==}
    +    engines: {node: '>=6.9.0'}
    +    dependencies:
    +      '@babel/code-frame': 7.22.13
    +      '@babel/parser': 7.23.0
    +      '@babel/types': 7.23.0
    +    dev: true
    +
       /@babel/template@7.22.5:
         resolution: {integrity: sha512-X7yV7eiwAxdj9k94NEylvbVHLiVG1nvzCV2EAowhxLTwODV1jl9UzZ48leOC0sH7OnuHrIkllaBgneUykIcZaw==}
         engines: {node: '>=6.9.0'}
         dependencies:
           '@babel/code-frame': 7.22.10
    -      '@babel/parser': 7.22.10
    -      '@babel/types': 7.22.10
    +      '@babel/parser': 7.23.0
    +      '@babel/types': 7.23.0
         dev: true
     
    -  /@babel/traverse@7.22.10:
    -    resolution: {integrity: sha512-Q/urqV4pRByiNNpb/f5OSv28ZlGJiFiiTh+GAHktbIrkPhPbl90+uW6SmpoLyZqutrg9AEaEf3Q/ZBRHBXgxig==}
    +  /@babel/traverse@7.23.2:
    +    resolution: {integrity: sha512-azpe59SQ48qG6nu2CzcMLbxUudtN+dOM9kDbUqGq3HXUJRlo7i8fvPoxQUzYgLZ4cMVmuZgm8vvBpNeRhd6XSw==}
         engines: {node: '>=6.9.0'}
         dependencies:
    -      '@babel/code-frame': 7.22.10
    -      '@babel/generator': 7.22.10
    -      '@babel/helper-environment-visitor': 7.22.5
    -      '@babel/helper-function-name': 7.22.5
    +      '@babel/code-frame': 7.22.13
    +      '@babel/generator': 7.23.0
    +      '@babel/helper-environment-visitor': 7.22.20
    +      '@babel/helper-function-name': 7.23.0
           '@babel/helper-hoist-variables': 7.22.5
           '@babel/helper-split-export-declaration': 7.22.6
    -      '@babel/parser': 7.22.10
    -      '@babel/types': 7.22.10
    +      '@babel/parser': 7.23.0
    +      '@babel/types': 7.23.0
           debug: 4.3.4(supports-color@8.1.1)
           globals: 11.12.0
         transitivePeerDependencies:
           - supports-color
         dev: true
     
    -  /@babel/types@7.22.10:
    -    resolution: {integrity: sha512-obaoigiLrlDZ7TUQln/8m4mSqIW2QFeOrCQc9r+xsaHGNoplVNYlRVpsfE8Vj35GEm2ZH4ZhrNYogs/3fj85kg==}
    +  /@babel/types@7.23.0:
    +    resolution: {integrity: sha512-0oIyUfKoI3mSqMvsxBdclDwxXKXAUA8v/apZbc+iSyARYou1o8ZGDxbUYyLFoW2arqS2jDGqJuZvv1d/io1axg==}
         engines: {node: '>=6.9.0'}
         dependencies:
           '@babel/helper-string-parser': 7.22.5
    -      '@babel/helper-validator-identifier': 7.22.5
    +      '@babel/helper-validator-identifier': 7.22.20
           to-fast-properties: 2.0.0
     
       /@bcherny/json-schema-ref-parser@9.0.9:
    @@ -2305,8 +2345,8 @@ packages:
         dev: true
         optional: true
     
    -  /@commitlint/cli@17.6.1:
    -    resolution: {integrity: sha512-kCnDD9LE2ySiTnj/VPaxy4/oRayRcdv4aCuVxtoum8SxIU7OADHc0nJPQfheE8bHcs3zZdWzDMWltRosuT13bg==}
    +  /@commitlint/cli@17.7.1:
    +    resolution: {integrity: sha512-BCm/AT06SNCQtvFv921iNhudOHuY16LswT0R3OeolVGLk8oP+Rk9TfQfgjH7QPMjhvp76bNqGFEcpKojxUNW1g==}
         engines: {node: '>=v14'}
         hasBin: true
         dependencies:
    @@ -2325,11 +2365,11 @@ packages:
           - '@swc/wasm'
         dev: true
     
    -  /@commitlint/config-conventional@17.6.1:
    -    resolution: {integrity: sha512-ng/ybaSLuTCH9F+7uavSOnEQ9EFMl7lHEjfAEgRh1hwmEe8SpLKpQeMo2aT1IWvHaGMuTb+gjfbzoRf2IR23NQ==}
    +  /@commitlint/config-conventional@17.7.0:
    +    resolution: {integrity: sha512-iicqh2o6et+9kWaqsQiEYZzfLbtoWv9uZl8kbI8EGfnc0HeGafQBF7AJ0ylN9D/2kj6txltsdyQs8+2fTMwWEw==}
         engines: {node: '>=v14'}
         dependencies:
    -      conventional-changelog-conventionalcommits: 5.0.0
    +      conventional-changelog-conventionalcommits: 6.1.0
         dev: true
     
       /@commitlint/config-validator@17.6.7:
    @@ -2394,13 +2434,13 @@ packages:
           '@types/node': 20.4.7
           chalk: 4.1.2
           cosmiconfig: 8.2.0
    -      cosmiconfig-typescript-loader: 4.4.0(@types/node@20.4.7)(cosmiconfig@8.2.0)(ts-node@10.9.1)(typescript@5.1.3)
    +      cosmiconfig-typescript-loader: 4.4.0(@types/node@20.4.7)(cosmiconfig@8.2.0)(ts-node@10.9.1)(typescript@5.1.6)
           lodash.isplainobject: 4.0.6
           lodash.merge: 4.6.2
           lodash.uniq: 4.5.0
           resolve-from: 5.0.0
    -      ts-node: 10.9.1(@types/node@20.4.7)(typescript@5.1.3)
    -      typescript: 5.1.3
    +      ts-node: 10.9.1(@types/node@20.4.7)(typescript@5.1.6)
    +      typescript: 5.1.6
         transitivePeerDependencies:
           - '@swc/core'
           - '@swc/wasm'
    @@ -2525,21 +2565,88 @@ packages:
           '@cspell/dict-vue': 3.0.0
         dev: true
     
    +  /@cspell/cspell-bundled-dicts@6.31.3:
    +    resolution: {integrity: sha512-KXy3qKWYzXOGYwqOGMCXHem3fV39iEmoKLiNhoWWry/SFdvAafmeY+LIDcQTXAcOQLkMDCwP2/rY/NadcWnrjg==}
    +    engines: {node: '>=14'}
    +    dependencies:
    +      '@cspell/dict-ada': 4.0.2
    +      '@cspell/dict-aws': 3.0.0
    +      '@cspell/dict-bash': 4.1.1
    +      '@cspell/dict-companies': 3.0.19
    +      '@cspell/dict-cpp': 5.0.4
    +      '@cspell/dict-cryptocurrencies': 3.0.1
    +      '@cspell/dict-csharp': 4.0.2
    +      '@cspell/dict-css': 4.0.6
    +      '@cspell/dict-dart': 2.0.3
    +      '@cspell/dict-django': 4.1.0
    +      '@cspell/dict-docker': 1.1.7
    +      '@cspell/dict-dotnet': 5.0.0
    +      '@cspell/dict-elixir': 4.0.3
    +      '@cspell/dict-en-common-misspellings': 1.0.2
    +      '@cspell/dict-en-gb': 1.1.33
    +      '@cspell/dict-en_us': 4.3.6
    +      '@cspell/dict-filetypes': 3.0.1
    +      '@cspell/dict-fonts': 3.0.2
    +      '@cspell/dict-fullstack': 3.1.5
    +      '@cspell/dict-gaming-terms': 1.0.4
    +      '@cspell/dict-git': 2.0.0
    +      '@cspell/dict-golang': 6.0.2
    +      '@cspell/dict-haskell': 4.0.1
    +      '@cspell/dict-html': 4.0.3
    +      '@cspell/dict-html-symbol-entities': 4.0.0
    +      '@cspell/dict-java': 5.0.5
    +      '@cspell/dict-k8s': 1.0.1
    +      '@cspell/dict-latex': 4.0.0
    +      '@cspell/dict-lorem-ipsum': 3.0.0
    +      '@cspell/dict-lua': 4.0.1
    +      '@cspell/dict-node': 4.0.2
    +      '@cspell/dict-npm': 5.0.8
    +      '@cspell/dict-php': 4.0.1
    +      '@cspell/dict-powershell': 5.0.2
    +      '@cspell/dict-public-licenses': 2.0.3
    +      '@cspell/dict-python': 4.1.5
    +      '@cspell/dict-r': 2.0.1
    +      '@cspell/dict-ruby': 5.0.0
    +      '@cspell/dict-rust': 4.0.1
    +      '@cspell/dict-scala': 5.0.0
    +      '@cspell/dict-software-terms': 3.2.1
    +      '@cspell/dict-sql': 2.1.1
    +      '@cspell/dict-svelte': 1.0.2
    +      '@cspell/dict-swift': 2.0.1
    +      '@cspell/dict-typescript': 3.1.1
    +      '@cspell/dict-vue': 3.0.0
    +    dev: true
    +
       /@cspell/cspell-pipe@6.31.1:
         resolution: {integrity: sha512-zk1olZi4dr6GLm5PAjvsiZ01HURNSruUYFl1qSicGnTwYN8GaN4RhAwannAytcJ7zJPIcyXlid0YsB58nJf3wQ==}
         engines: {node: '>=14'}
         dev: true
     
    +  /@cspell/cspell-pipe@6.31.3:
    +    resolution: {integrity: sha512-Lv/y4Ya/TJyU1pf66yl1te7LneFZd3lZg1bN5oe1cPrKSmfWdiX48v7plTRecWd/OWyLGd0yN807v79A+/0W7A==}
    +    engines: {node: '>=14'}
    +    dev: true
    +
       /@cspell/cspell-service-bus@6.31.1:
         resolution: {integrity: sha512-YyBicmJyZ1uwKVxujXw7sgs9x+Eps43OkWmCtDZmZlnq489HdTSuhF1kTbVi2yeFSeaXIS87+uHo12z97KkQpg==}
         engines: {node: '>=14'}
         dev: true
     
    +  /@cspell/cspell-service-bus@6.31.3:
    +    resolution: {integrity: sha512-x5j8j3n39KN8EXOAlv75CpircdpF5WEMCC5pcO916o6GBmJBy8SrdzdsBGJhVcYGGilqy6pf8R9RCZ3yAmG8gQ==}
    +    engines: {node: '>=14'}
    +    dev: true
    +
       /@cspell/cspell-types@6.31.1:
         resolution: {integrity: sha512-1KeTQFiHMssW1eRoF2NZIEg4gPVIfXLsL2+VSD/AV6YN7lBcuf6gRRgV5KWYarhxtEfjxhDdDTmu26l/iJEUtw==}
         engines: {node: '>=14'}
         dev: true
     
    +  /@cspell/cspell-types@6.31.3:
    +    resolution: {integrity: sha512-wZ+t+lUsQJB65M31btZM4fH3K1CkRgE8pSeTiCwxYcnCL19pi4TMcEEMKdO8yFZMdocW4B7VRwzxNoQMw2ewBg==}
    +    engines: {node: '>=14'}
    +    dev: true
    +
       /@cspell/dict-ada@4.0.2:
         resolution: {integrity: sha512-0kENOWQeHjUlfyId/aCM/mKXtkEgV0Zu2RhUXCBr4hHo9F9vph+Uu8Ww2b0i5a4ZixoIkudGA+eJvyxrG1jUpA==}
         dev: true
    @@ -2737,11 +2844,11 @@ packages:
           import-meta-resolve: 2.2.2
         dev: true
     
    -  /@cspell/eslint-plugin@6.31.1:
    -    resolution: {integrity: sha512-4WJXHZVHdVqIyaxCHRgd55bqkhjSDIu/kDJAvprxhEGdyladIOHZ/c3si8AMJYnYbZY56zY2LsF2W5uNyIsRgg==}
    +  /@cspell/eslint-plugin@6.31.3:
    +    resolution: {integrity: sha512-WrgJsc4IZA/u/v1DrNOQlZt6KW+1aZlFKR7cM3a36NimedtP6Cd3gMm2hg/NtUfDjmEHBXiSkjiw4HZFJT7p0Q==}
         engines: {node: '>=14'}
         dependencies:
    -      cspell-lib: 6.31.1
    +      cspell-lib: 6.31.3
           estree-walker: 3.0.3
           synckit: 0.8.5
         transitivePeerDependencies:
    @@ -2753,21 +2860,26 @@ packages:
         engines: {node: '>=14.6'}
         dev: true
     
    +  /@cspell/strong-weak-map@6.31.3:
    +    resolution: {integrity: sha512-znwc9IlgGUPioHGshP/zyM8HsuYg1OY5S7HSiVXARh5H8RqcyBsnyn8abc0PPhqPrfDy9Fh5xHsAEPZ55dl1vQ==}
    +    engines: {node: '>=14.6'}
    +    dev: true
    +
       /@cspotcode/source-map-support@0.8.1:
         resolution: {integrity: sha512-IchNf6dN4tHoMFIn/7OE8LWZ19Y6q/67Bmf6vnGREv8RSbBVb9LPJxEcnwrcwX6ixSvaiGoomAUvu4YSxXrVgw==}
         engines: {node: '>=12'}
         dependencies:
           '@jridgewell/trace-mapping': 0.3.9
     
    -  /@cypress/code-coverage@3.10.7(@babel/core@7.22.10)(@babel/preset-env@7.22.10)(babel-loader@9.1.3)(cypress@12.10.0)(webpack@5.88.2):
    -    resolution: {integrity: sha512-kQFB8GemDAAk6JBINsR9MLEgCw2AKb3FcdHQjIJ3KV4ZER6ZF0NGdO8SRj5oTVp28oqfOab4cgoBdecRiOE3qA==}
    +  /@cypress/code-coverage@3.11.0(@babel/core@7.22.10)(@babel/preset-env@7.22.10)(babel-loader@9.1.3)(cypress@12.17.3)(webpack@5.88.2):
    +    resolution: {integrity: sha512-ihSO1s03gmLRE224oIjrbdG1ey63vw/UY+VSqQ5m/TKkAvyz6GIiniq6juk3AV/+0vQC1Eb4UWFu8ndtji4M1g==}
         peerDependencies:
           cypress: '*'
         dependencies:
           '@cypress/webpack-preprocessor': 5.17.1(@babel/core@7.22.10)(@babel/preset-env@7.22.10)(babel-loader@9.1.3)(webpack@5.88.2)
           chalk: 4.1.2
    -      cypress: 12.10.0
    -      dayjs: 1.10.7
    +      cypress: 12.17.3
    +      dayjs: 1.11.9
           debug: 4.3.4(supports-color@8.1.1)
           execa: 4.1.0
           globby: 11.0.4
    @@ -2839,15 +2951,32 @@ packages:
         engines: {node: '>=10.0.0'}
         dev: true
     
    +  /@docsearch/css@3.5.1:
    +    resolution: {integrity: sha512-2Pu9HDg/uP/IT10rbQ+4OrTQuxIWdKVUEdcw9/w7kZJv9NeHS6skJx1xuRiFyoGKwAzcHXnLp7csE99sj+O1YA==}
    +    dev: true
    +
       /@docsearch/css@3.5.2:
         resolution: {integrity: sha512-SPiDHaWKQZpwR2siD0KQUwlStvIAnEyK6tAE2h2Wuoq8ue9skzhlyVQ1ddzOxX6khULnAALDiR/isSF3bnuciA==}
         dev: true
     
    +  /@docsearch/js@3.5.1(@algolia/client-search@4.19.1)(search-insights@2.7.0):
    +    resolution: {integrity: sha512-EXi8de5njxgP6TV3N9ytnGRLG9zmBNTEZjR4VzwPcpPLbZxxTLG2gaFyJyKiFVQxHW/DPlMrDJA3qoRRGEkgZw==}
    +    dependencies:
    +      '@docsearch/react': 3.5.1(@algolia/client-search@4.19.1)(search-insights@2.7.0)
    +      preact: 10.16.0
    +    transitivePeerDependencies:
    +      - '@algolia/client-search'
    +      - '@types/react'
    +      - react
    +      - react-dom
    +      - search-insights
    +    dev: true
    +
       /@docsearch/js@3.5.2(@algolia/client-search@4.19.1)(search-insights@2.7.0):
         resolution: {integrity: sha512-p1YFTCDflk8ieHgFJYfmyHBki1D61+U9idwrLh+GQQMrBSP3DLGKpy0XUJtPjAOPltcVbqsTjiPFfH7JImjUNg==}
         dependencies:
           '@docsearch/react': 3.5.2(@algolia/client-search@4.19.1)(search-insights@2.7.0)
    -      preact: 10.17.1
    +      preact: 10.16.0
         transitivePeerDependencies:
           - '@algolia/client-search'
           - '@types/react'
    @@ -2856,6 +2985,29 @@ packages:
           - search-insights
         dev: true
     
    +  /@docsearch/react@3.5.1(@algolia/client-search@4.19.1)(search-insights@2.7.0):
    +    resolution: {integrity: sha512-t5mEODdLzZq4PTFAm/dvqcvZFdPDMdfPE5rJS5SC8OUq9mPzxEy6b+9THIqNM9P0ocCb4UC5jqBrxKclnuIbzQ==}
    +    peerDependencies:
    +      '@types/react': '>= 16.8.0 < 19.0.0'
    +      react: '>= 16.8.0 < 19.0.0'
    +      react-dom: '>= 16.8.0 < 19.0.0'
    +    peerDependenciesMeta:
    +      '@types/react':
    +        optional: true
    +      react:
    +        optional: true
    +      react-dom:
    +        optional: true
    +    dependencies:
    +      '@algolia/autocomplete-core': 1.9.3(@algolia/client-search@4.19.1)(algoliasearch@4.19.1)(search-insights@2.7.0)
    +      '@algolia/autocomplete-preset-algolia': 1.9.3(@algolia/client-search@4.19.1)(algoliasearch@4.19.1)
    +      '@docsearch/css': 3.5.1
    +      algoliasearch: 4.19.1
    +    transitivePeerDependencies:
    +      - '@algolia/client-search'
    +      - search-insights
    +    dev: true
    +
       /@docsearch/react@3.5.2(@algolia/client-search@4.19.1)(search-insights@2.7.0):
         resolution: {integrity: sha512-9Ahcrs5z2jq/DcAvYtvlqEBHImbm4YJI8M9y0x6Tqg598P40HTEkX7hsMcIuThI+hTFxRGZ9hll0Wygm2yEjng==}
         peerDependencies:
    @@ -2882,24 +3034,15 @@ packages:
           - '@algolia/client-search'
         dev: true
     
    -  /@es-joy/jsdoccomment@0.39.4:
    -    resolution: {integrity: sha512-Jvw915fjqQct445+yron7Dufix9A+m9j1fCJYlCo1FWlRvTxa3pjJelxdSTdaLWcTwRU6vbL+NYjO4YuNIS5Qg==}
    +  /@es-joy/jsdoccomment@0.40.1:
    +    resolution: {integrity: sha512-YORCdZSusAlBrFpZ77pJjc5r1bQs5caPWtAu+WWmiSo+8XaUzseapVrfAtiRFbQWnrBxxLLEwF6f6ZG/UgCQCg==}
         engines: {node: '>=16'}
         dependencies:
    -      comment-parser: 1.3.1
    +      comment-parser: 1.4.0
           esquery: 1.5.0
           jsdoc-type-pratt-parser: 4.0.0
         dev: true
     
    -  /@esbuild/android-arm64@0.17.19:
    -    resolution: {integrity: sha512-KBMWvEZooR7+kzY0BtbTQn0OAYY7CsiydT63pVEaPtVYF0hXbUaOyZog37DKxK7NF3XacBJOpYT4adIJh+avxA==}
    -    engines: {node: '>=12'}
    -    cpu: [arm64]
    -    os: [android]
    -    requiresBuild: true
    -    dev: true
    -    optional: true
    -
       /@esbuild/android-arm64@0.18.20:
         resolution: {integrity: sha512-Nz4rJcchGDtENV0eMKUNa6L12zz2zBDXuhj/Vjh18zGqB44Bi7MBMSXjgunJgjRhCmKOjnPuZp4Mb6OKqtMHLQ==}
         engines: {node: '>=12'}
    @@ -2918,15 +3061,6 @@ packages:
         dev: true
         optional: true
     
    -  /@esbuild/android-arm@0.17.19:
    -    resolution: {integrity: sha512-rIKddzqhmav7MSmoFCmDIb6e2W57geRsM94gV2l38fzhXMwq7hZoClug9USI2pFRGL06f4IOPHHpFNOkWieR8A==}
    -    engines: {node: '>=12'}
    -    cpu: [arm]
    -    os: [android]
    -    requiresBuild: true
    -    dev: true
    -    optional: true
    -
       /@esbuild/android-arm@0.18.20:
         resolution: {integrity: sha512-fyi7TDI/ijKKNZTUJAQqiG5T7YjJXgnzkURqmGj13C6dCqckZBLdl4h7bkhHt/t0WP+zO9/zwroDvANaOqO5Sw==}
         engines: {node: '>=12'}
    @@ -2945,15 +3079,6 @@ packages:
         dev: true
         optional: true
     
    -  /@esbuild/android-x64@0.17.19:
    -    resolution: {integrity: sha512-uUTTc4xGNDT7YSArp/zbtmbhO0uEEK9/ETW29Wk1thYUJBz3IVnvgEiEwEa9IeLyvnpKrWK64Utw2bgUmDveww==}
    -    engines: {node: '>=12'}
    -    cpu: [x64]
    -    os: [android]
    -    requiresBuild: true
    -    dev: true
    -    optional: true
    -
       /@esbuild/android-x64@0.18.20:
         resolution: {integrity: sha512-8GDdlePJA8D6zlZYJV/jnrRAi6rOiNaCC/JclcXpB+KIuvfBN4owLtgzY2bsxnx666XjJx2kDPUmnTtR8qKQUg==}
         engines: {node: '>=12'}
    @@ -2972,15 +3097,6 @@ packages:
         dev: true
         optional: true
     
    -  /@esbuild/darwin-arm64@0.17.19:
    -    resolution: {integrity: sha512-80wEoCfF/hFKM6WE1FyBHc9SfUblloAWx6FJkFWTWiCoht9Mc0ARGEM47e67W9rI09YoUxJL68WHfDRYEAvOhg==}
    -    engines: {node: '>=12'}
    -    cpu: [arm64]
    -    os: [darwin]
    -    requiresBuild: true
    -    dev: true
    -    optional: true
    -
       /@esbuild/darwin-arm64@0.18.20:
         resolution: {integrity: sha512-bxRHW5kHU38zS2lPTPOyuyTm+S+eobPUnTNkdJEfAddYgEcll4xkT8DB9d2008DtTbl7uJag2HuE5NZAZgnNEA==}
         engines: {node: '>=12'}
    @@ -2999,15 +3115,6 @@ packages:
         dev: true
         optional: true
     
    -  /@esbuild/darwin-x64@0.17.19:
    -    resolution: {integrity: sha512-IJM4JJsLhRYr9xdtLytPLSH9k/oxR3boaUIYiHkAawtwNOXKE8KoU8tMvryogdcT8AU+Bflmh81Xn6Q0vTZbQw==}
    -    engines: {node: '>=12'}
    -    cpu: [x64]
    -    os: [darwin]
    -    requiresBuild: true
    -    dev: true
    -    optional: true
    -
       /@esbuild/darwin-x64@0.18.20:
         resolution: {integrity: sha512-pc5gxlMDxzm513qPGbCbDukOdsGtKhfxD1zJKXjCCcU7ju50O7MeAZ8c4krSJcOIJGFR+qx21yMMVYwiQvyTyQ==}
         engines: {node: '>=12'}
    @@ -3026,15 +3133,6 @@ packages:
         dev: true
         optional: true
     
    -  /@esbuild/freebsd-arm64@0.17.19:
    -    resolution: {integrity: sha512-pBwbc7DufluUeGdjSU5Si+P3SoMF5DQ/F/UmTSb8HXO80ZEAJmrykPyzo1IfNbAoaqw48YRpv8shwd1NoI0jcQ==}
    -    engines: {node: '>=12'}
    -    cpu: [arm64]
    -    os: [freebsd]
    -    requiresBuild: true
    -    dev: true
    -    optional: true
    -
       /@esbuild/freebsd-arm64@0.18.20:
         resolution: {integrity: sha512-yqDQHy4QHevpMAaxhhIwYPMv1NECwOvIpGCZkECn8w2WFHXjEwrBn3CeNIYsibZ/iZEUemj++M26W3cNR5h+Tw==}
         engines: {node: '>=12'}
    @@ -3053,15 +3151,6 @@ packages:
         dev: true
         optional: true
     
    -  /@esbuild/freebsd-x64@0.17.19:
    -    resolution: {integrity: sha512-4lu+n8Wk0XlajEhbEffdy2xy53dpR06SlzvhGByyg36qJw6Kpfk7cp45DR/62aPH9mtJRmIyrXAS5UWBrJT6TQ==}
    -    engines: {node: '>=12'}
    -    cpu: [x64]
    -    os: [freebsd]
    -    requiresBuild: true
    -    dev: true
    -    optional: true
    -
       /@esbuild/freebsd-x64@0.18.20:
         resolution: {integrity: sha512-tgWRPPuQsd3RmBZwarGVHZQvtzfEBOreNuxEMKFcd5DaDn2PbBxfwLcj4+aenoh7ctXcbXmOQIn8HI6mCSw5MQ==}
         engines: {node: '>=12'}
    @@ -3080,15 +3169,6 @@ packages:
         dev: true
         optional: true
     
    -  /@esbuild/linux-arm64@0.17.19:
    -    resolution: {integrity: sha512-ct1Tg3WGwd3P+oZYqic+YZF4snNl2bsnMKRkb3ozHmnM0dGWuxcPTTntAF6bOP0Sp4x0PjSF+4uHQ1xvxfRKqg==}
    -    engines: {node: '>=12'}
    -    cpu: [arm64]
    -    os: [linux]
    -    requiresBuild: true
    -    dev: true
    -    optional: true
    -
       /@esbuild/linux-arm64@0.18.20:
         resolution: {integrity: sha512-2YbscF+UL7SQAVIpnWvYwM+3LskyDmPhe31pE7/aoTMFKKzIc9lLbyGUpmmb8a8AixOL61sQ/mFh3jEjHYFvdA==}
         engines: {node: '>=12'}
    @@ -3107,15 +3187,6 @@ packages:
         dev: true
         optional: true
     
    -  /@esbuild/linux-arm@0.17.19:
    -    resolution: {integrity: sha512-cdmT3KxjlOQ/gZ2cjfrQOtmhG4HJs6hhvm3mWSRDPtZ/lP5oe8FWceS10JaSJC13GBd4eH/haHnqf7hhGNLerA==}
    -    engines: {node: '>=12'}
    -    cpu: [arm]
    -    os: [linux]
    -    requiresBuild: true
    -    dev: true
    -    optional: true
    -
       /@esbuild/linux-arm@0.18.20:
         resolution: {integrity: sha512-/5bHkMWnq1EgKr1V+Ybz3s1hWXok7mDFUMQ4cG10AfW3wL02PSZi5kFpYKrptDsgb2WAJIvRcDm+qIvXf/apvg==}
         engines: {node: '>=12'}
    @@ -3134,15 +3205,6 @@ packages:
         dev: true
         optional: true
     
    -  /@esbuild/linux-ia32@0.17.19:
    -    resolution: {integrity: sha512-w4IRhSy1VbsNxHRQpeGCHEmibqdTUx61Vc38APcsRbuVgK0OPEnQ0YD39Brymn96mOx48Y2laBQGqgZ0j9w6SQ==}
    -    engines: {node: '>=12'}
    -    cpu: [ia32]
    -    os: [linux]
    -    requiresBuild: true
    -    dev: true
    -    optional: true
    -
       /@esbuild/linux-ia32@0.18.20:
         resolution: {integrity: sha512-P4etWwq6IsReT0E1KHU40bOnzMHoH73aXp96Fs8TIT6z9Hu8G6+0SHSw9i2isWrD2nbx2qo5yUqACgdfVGx7TA==}
         engines: {node: '>=12'}
    @@ -3161,15 +3223,6 @@ packages:
         dev: true
         optional: true
     
    -  /@esbuild/linux-loong64@0.17.19:
    -    resolution: {integrity: sha512-2iAngUbBPMq439a+z//gE+9WBldoMp1s5GWsUSgqHLzLJ9WoZLZhpwWuym0u0u/4XmZ3gpHmzV84PonE+9IIdQ==}
    -    engines: {node: '>=12'}
    -    cpu: [loong64]
    -    os: [linux]
    -    requiresBuild: true
    -    dev: true
    -    optional: true
    -
       /@esbuild/linux-loong64@0.18.20:
         resolution: {integrity: sha512-nXW8nqBTrOpDLPgPY9uV+/1DjxoQ7DoB2N8eocyq8I9XuqJ7BiAMDMf9n1xZM9TgW0J8zrquIb/A7s3BJv7rjg==}
         engines: {node: '>=12'}
    @@ -3188,15 +3241,6 @@ packages:
         dev: true
         optional: true
     
    -  /@esbuild/linux-mips64el@0.17.19:
    -    resolution: {integrity: sha512-LKJltc4LVdMKHsrFe4MGNPp0hqDFA1Wpt3jE1gEyM3nKUvOiO//9PheZZHfYRfYl6AwdTH4aTcXSqBerX0ml4A==}
    -    engines: {node: '>=12'}
    -    cpu: [mips64el]
    -    os: [linux]
    -    requiresBuild: true
    -    dev: true
    -    optional: true
    -
       /@esbuild/linux-mips64el@0.18.20:
         resolution: {integrity: sha512-d5NeaXZcHp8PzYy5VnXV3VSd2D328Zb+9dEq5HE6bw6+N86JVPExrA6O68OPwobntbNJ0pzCpUFZTo3w0GyetQ==}
         engines: {node: '>=12'}
    @@ -3215,15 +3259,6 @@ packages:
         dev: true
         optional: true
     
    -  /@esbuild/linux-ppc64@0.17.19:
    -    resolution: {integrity: sha512-/c/DGybs95WXNS8y3Ti/ytqETiW7EU44MEKuCAcpPto3YjQbyK3IQVKfF6nbghD7EcLUGl0NbiL5Rt5DMhn5tg==}
    -    engines: {node: '>=12'}
    -    cpu: [ppc64]
    -    os: [linux]
    -    requiresBuild: true
    -    dev: true
    -    optional: true
    -
       /@esbuild/linux-ppc64@0.18.20:
         resolution: {integrity: sha512-WHPyeScRNcmANnLQkq6AfyXRFr5D6N2sKgkFo2FqguP44Nw2eyDlbTdZwd9GYk98DZG9QItIiTlFLHJHjxP3FA==}
         engines: {node: '>=12'}
    @@ -3242,15 +3277,6 @@ packages:
         dev: true
         optional: true
     
    -  /@esbuild/linux-riscv64@0.17.19:
    -    resolution: {integrity: sha512-FC3nUAWhvFoutlhAkgHf8f5HwFWUL6bYdvLc/TTuxKlvLi3+pPzdZiFKSWz/PF30TB1K19SuCxDTI5KcqASJqA==}
    -    engines: {node: '>=12'}
    -    cpu: [riscv64]
    -    os: [linux]
    -    requiresBuild: true
    -    dev: true
    -    optional: true
    -
       /@esbuild/linux-riscv64@0.18.20:
         resolution: {integrity: sha512-WSxo6h5ecI5XH34KC7w5veNnKkju3zBRLEQNY7mv5mtBmrP/MjNBCAlsM2u5hDBlS3NGcTQpoBvRzqBcRtpq1A==}
         engines: {node: '>=12'}
    @@ -3269,15 +3295,6 @@ packages:
         dev: true
         optional: true
     
    -  /@esbuild/linux-s390x@0.17.19:
    -    resolution: {integrity: sha512-IbFsFbxMWLuKEbH+7sTkKzL6NJmG2vRyy6K7JJo55w+8xDk7RElYn6xvXtDW8HCfoKBFK69f3pgBJSUSQPr+4Q==}
    -    engines: {node: '>=12'}
    -    cpu: [s390x]
    -    os: [linux]
    -    requiresBuild: true
    -    dev: true
    -    optional: true
    -
       /@esbuild/linux-s390x@0.18.20:
         resolution: {integrity: sha512-+8231GMs3mAEth6Ja1iK0a1sQ3ohfcpzpRLH8uuc5/KVDFneH6jtAJLFGafpzpMRO6DzJ6AvXKze9LfFMrIHVQ==}
         engines: {node: '>=12'}
    @@ -3296,15 +3313,6 @@ packages:
         dev: true
         optional: true
     
    -  /@esbuild/linux-x64@0.17.19:
    -    resolution: {integrity: sha512-68ngA9lg2H6zkZcyp22tsVt38mlhWde8l3eJLWkyLrp4HwMUr3c1s/M2t7+kHIhvMjglIBrFpncX1SzMckomGw==}
    -    engines: {node: '>=12'}
    -    cpu: [x64]
    -    os: [linux]
    -    requiresBuild: true
    -    dev: true
    -    optional: true
    -
       /@esbuild/linux-x64@0.18.20:
         resolution: {integrity: sha512-UYqiqemphJcNsFEskc73jQ7B9jgwjWrSayxawS6UVFZGWrAAtkzjxSqnoclCXxWtfwLdzU+vTpcNYhpn43uP1w==}
         engines: {node: '>=12'}
    @@ -3323,15 +3331,6 @@ packages:
         dev: true
         optional: true
     
    -  /@esbuild/netbsd-x64@0.17.19:
    -    resolution: {integrity: sha512-CwFq42rXCR8TYIjIfpXCbRX0rp1jo6cPIUPSaWwzbVI4aOfX96OXY8M6KNmtPcg7QjYeDmN+DD0Wp3LaBOLf4Q==}
    -    engines: {node: '>=12'}
    -    cpu: [x64]
    -    os: [netbsd]
    -    requiresBuild: true
    -    dev: true
    -    optional: true
    -
       /@esbuild/netbsd-x64@0.18.20:
         resolution: {integrity: sha512-iO1c++VP6xUBUmltHZoMtCUdPlnPGdBom6IrO4gyKPFFVBKioIImVooR5I83nTew5UOYrk3gIJhbZh8X44y06A==}
         engines: {node: '>=12'}
    @@ -3350,15 +3349,6 @@ packages:
         dev: true
         optional: true
     
    -  /@esbuild/openbsd-x64@0.17.19:
    -    resolution: {integrity: sha512-cnq5brJYrSZ2CF6c35eCmviIN3k3RczmHz8eYaVlNasVqsNY+JKohZU5MKmaOI+KkllCdzOKKdPs762VCPC20g==}
    -    engines: {node: '>=12'}
    -    cpu: [x64]
    -    os: [openbsd]
    -    requiresBuild: true
    -    dev: true
    -    optional: true
    -
       /@esbuild/openbsd-x64@0.18.20:
         resolution: {integrity: sha512-e5e4YSsuQfX4cxcygw/UCPIEP6wbIL+se3sxPdCiMbFLBWu0eiZOJ7WoD+ptCLrmjZBK1Wk7I6D/I3NglUGOxg==}
         engines: {node: '>=12'}
    @@ -3377,15 +3367,6 @@ packages:
         dev: true
         optional: true
     
    -  /@esbuild/sunos-x64@0.17.19:
    -    resolution: {integrity: sha512-vCRT7yP3zX+bKWFeP/zdS6SqdWB8OIpaRq/mbXQxTGHnIxspRtigpkUcDMlSCOejlHowLqII7K2JKevwyRP2rg==}
    -    engines: {node: '>=12'}
    -    cpu: [x64]
    -    os: [sunos]
    -    requiresBuild: true
    -    dev: true
    -    optional: true
    -
       /@esbuild/sunos-x64@0.18.20:
         resolution: {integrity: sha512-kDbFRFp0YpTQVVrqUd5FTYmWo45zGaXe0X8E1G/LKFC0v8x0vWrhOWSLITcCn63lmZIxfOMXtCfti/RxN/0wnQ==}
         engines: {node: '>=12'}
    @@ -3404,15 +3385,6 @@ packages:
         dev: true
         optional: true
     
    -  /@esbuild/win32-arm64@0.17.19:
    -    resolution: {integrity: sha512-yYx+8jwowUstVdorcMdNlzklLYhPxjniHWFKgRqH7IFlUEa0Umu3KuYplf1HUZZ422e3NU9F4LGb+4O0Kdcaag==}
    -    engines: {node: '>=12'}
    -    cpu: [arm64]
    -    os: [win32]
    -    requiresBuild: true
    -    dev: true
    -    optional: true
    -
       /@esbuild/win32-arm64@0.18.20:
         resolution: {integrity: sha512-ddYFR6ItYgoaq4v4JmQQaAI5s7npztfV4Ag6NrhiaW0RrnOXqBkgwZLofVTlq1daVTQNhtI5oieTvkRPfZrePg==}
         engines: {node: '>=12'}
    @@ -3431,15 +3403,6 @@ packages:
         dev: true
         optional: true
     
    -  /@esbuild/win32-ia32@0.17.19:
    -    resolution: {integrity: sha512-eggDKanJszUtCdlVs0RB+h35wNlb5v4TWEkq4vZcmVt5u/HiDZrTXe2bWFQUez3RgNHwx/x4sk5++4NSSicKkw==}
    -    engines: {node: '>=12'}
    -    cpu: [ia32]
    -    os: [win32]
    -    requiresBuild: true
    -    dev: true
    -    optional: true
    -
       /@esbuild/win32-ia32@0.18.20:
         resolution: {integrity: sha512-Wv7QBi3ID/rROT08SABTS7eV4hX26sVduqDOTe1MvGMjNd3EjOz4b7zeexIR62GTIEKrfJXKL9LFxTYgkyeu7g==}
         engines: {node: '>=12'}
    @@ -3458,15 +3421,6 @@ packages:
         dev: true
         optional: true
     
    -  /@esbuild/win32-x64@0.17.19:
    -    resolution: {integrity: sha512-lAhycmKnVOuRYNtRtatQR1LPQf2oYCkRGkSFnseDAKPl8lu5SOsK/e1sXe5a0Pc5kHIHe6P2I/ilntNv2xf3cA==}
    -    engines: {node: '>=12'}
    -    cpu: [x64]
    -    os: [win32]
    -    requiresBuild: true
    -    dev: true
    -    optional: true
    -
       /@esbuild/win32-x64@0.18.20:
         resolution: {integrity: sha512-kTdfRcSiDfQca/y9QIkng02avJ+NCaQvrMejlsB3RRv5sE9rRoeBPISaZpKxHELzRxZyLvNts1P27W3wV+8geQ==}
         engines: {node: '>=12'}
    @@ -3485,16 +3439,6 @@ packages:
         dev: true
         optional: true
     
    -  /@eslint-community/eslint-utils@4.4.0(eslint@8.39.0):
    -    resolution: {integrity: sha512-1/sA4dwrzBAyeUoQ6oxahHKmrZvsnLCg4RfxW3ZFGGmQkSNQPFNLV9CUEFQP1x9EYXHTo5p6xdhZM1Ne9p/AfA==}
    -    engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0}
    -    peerDependencies:
    -      eslint: ^6.0.0 || ^7.0.0 || >=8.0.0
    -    dependencies:
    -      eslint: 8.39.0
    -      eslint-visitor-keys: 3.4.3
    -    dev: true
    -
       /@eslint-community/eslint-utils@4.4.0(eslint@8.47.0):
         resolution: {integrity: sha512-1/sA4dwrzBAyeUoQ6oxahHKmrZvsnLCg4RfxW3ZFGGmQkSNQPFNLV9CUEFQP1x9EYXHTo5p6xdhZM1Ne9p/AfA==}
         engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0}
    @@ -3527,11 +3471,6 @@ packages:
           - supports-color
         dev: true
     
    -  /@eslint/js@8.39.0:
    -    resolution: {integrity: sha512-kf9RB0Fg7NZfap83B3QOqOGg9QmD9yBudqQXzzOtn3i4y7ZUXe5ONeW34Gwi+TxhH4mvj72R1Zc300KUMa9Bng==}
    -    engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0}
    -    dev: true
    -
       /@eslint/js@8.47.0:
         resolution: {integrity: sha512-P6omY1zv5MItm93kLM8s2vr1HICJH8v0dvddDhysbIuZ+vcjOHg5Zbkf1mTkcmi2JA9oBG2anOkRnW8WJTS8Og==}
         engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0}
    @@ -3591,7 +3530,7 @@ packages:
         resolution: {integrity: sha512-M/w3PkN8zQYXi8N6qK/KhnYMfEbbb6Sk8RZVn8g+Pmmu5ybw177RpsaGwpziyHeUsu4etrexYSWq3rwnIqzYCg==}
         dependencies:
           '@antfu/install-pkg': 0.1.1
    -      '@antfu/utils': 0.7.6
    +      '@antfu/utils': 0.7.5
           '@iconify/types': 2.0.0
           debug: 4.3.4(supports-color@8.1.1)
           kolorist: 1.8.0
    @@ -3633,7 +3572,7 @@ packages:
         engines: {node: ^14.15.0 || ^16.10.0 || >=18.0.0}
         dependencies:
           '@jest/types': 29.6.1
    -      '@types/node': 18.17.6
    +      '@types/node': 18.17.5
           chalk: 4.1.2
           jest-message-util: 29.6.2
           jest-util: 29.6.2
    @@ -3654,14 +3593,14 @@ packages:
           '@jest/test-result': 29.6.2
           '@jest/transform': 29.6.2
           '@jest/types': 29.6.1
    -      '@types/node': 18.17.6
    +      '@types/node': 18.17.5
           ansi-escapes: 4.3.2
           chalk: 4.1.2
           ci-info: 3.8.0
           exit: 0.1.2
           graceful-fs: 4.2.11
           jest-changed-files: 29.5.0
    -      jest-config: 29.6.2(@types/node@18.17.6)(ts-node@10.9.1)
    +      jest-config: 29.6.2(@types/node@18.17.5)(ts-node@10.9.1)
           jest-haste-map: 29.6.2
           jest-message-util: 29.6.2
           jest-regex-util: 29.4.3
    @@ -3689,7 +3628,7 @@ packages:
         dependencies:
           '@jest/fake-timers': 29.6.2
           '@jest/types': 29.6.1
    -      '@types/node': 18.17.6
    +      '@types/node': 18.17.5
           jest-mock: 29.6.2
         dev: true
     
    @@ -3716,7 +3655,7 @@ packages:
         dependencies:
           '@jest/types': 29.6.1
           '@sinonjs/fake-timers': 10.3.0
    -      '@types/node': 18.17.6
    +      '@types/node': 18.17.5
           jest-message-util: 29.6.2
           jest-mock: 29.6.2
           jest-util: 29.6.2
    @@ -3749,7 +3688,7 @@ packages:
           '@jest/transform': 29.6.2
           '@jest/types': 29.6.1
           '@jridgewell/trace-mapping': 0.3.19
    -      '@types/node': 18.17.6
    +      '@types/node': 18.17.5
           chalk: 4.1.2
           collect-v8-coverage: 1.0.2
           exit: 0.1.2
    @@ -3837,7 +3776,7 @@ packages:
           '@jest/schemas': 29.6.0
           '@types/istanbul-lib-coverage': 2.0.4
           '@types/istanbul-reports': 3.0.1
    -      '@types/node': 18.17.6
    +      '@types/node': 18.17.5
           '@types/yargs': 17.0.24
           chalk: 4.1.2
         dev: true
    @@ -3935,7 +3874,7 @@ packages:
           is-glob: 4.0.3
           open: 9.1.0
           picocolors: 1.0.0
    -      tslib: 2.6.2
    +      tslib: 2.6.1
         dev: true
     
       /@polka/url@0.5.0:
    @@ -3988,8 +3927,8 @@ packages:
           rollup: 2.79.1
         dev: true
     
    -  /@rollup/plugin-typescript@11.1.1(typescript@5.1.3):
    -    resolution: {integrity: sha512-Ioir+x5Bejv72Lx2Zbz3/qGg7tvGbxQZALCLoJaGrkNXak/19+vKgKYJYM3i/fJxvsb23I9FuFQ8CUBEfsmBRg==}
    +  /@rollup/plugin-typescript@11.1.2(typescript@5.1.6):
    +    resolution: {integrity: sha512-0ghSOCMcA7fl1JM+0gYRf+Q/HWyg+zg7/gDSc+fRLmlJWcW5K1I+CLRzaRhXf4Y3DRyPnnDo4M2ktw+a6JcDEg==}
         engines: {node: '>=14.0.0'}
         peerDependencies:
           rollup: ^2.14.0||^3.0.0
    @@ -4003,7 +3942,7 @@ packages:
         dependencies:
           '@rollup/pluginutils': 5.0.3(rollup@2.79.1)
           resolve: 1.22.4
    -      typescript: 5.1.3
    +      typescript: 5.1.6
         dev: true
     
       /@rollup/pluginutils@3.1.0(rollup@2.79.1):
    @@ -4123,8 +4062,8 @@ packages:
       /@types/babel__core@7.20.1:
         resolution: {integrity: sha512-aACu/U/omhdk15O4Nfb+fHgH/z3QsfQzpnvRZhYhThms83ZnAOZz7zZAWO7mn2yyNQaA4xTO8GLK3uqFU4bYYw==}
         dependencies:
    -      '@babel/parser': 7.22.10
    -      '@babel/types': 7.22.10
    +      '@babel/parser': 7.23.0
    +      '@babel/types': 7.23.0
           '@types/babel__generator': 7.6.4
           '@types/babel__template': 7.4.1
           '@types/babel__traverse': 7.20.1
    @@ -4133,33 +4072,33 @@ packages:
       /@types/babel__generator@7.6.4:
         resolution: {integrity: sha512-tFkciB9j2K755yrTALxD44McOrk+gfpIpvC3sxHjRawj6PfnQxrse4Clq5y/Rq+G3mrBurMax/lG8Qn2t9mSsg==}
         dependencies:
    -      '@babel/types': 7.22.10
    +      '@babel/types': 7.23.0
         dev: true
     
       /@types/babel__template@7.4.1:
         resolution: {integrity: sha512-azBFKemX6kMg5Io+/rdGT0dkGreboUVR0Cdm3fz9QJWpaQGJRQXl7C+6hOTCZcMll7KFyEQpgbYI2lHdsS4U7g==}
         dependencies:
    -      '@babel/parser': 7.22.10
    -      '@babel/types': 7.22.10
    +      '@babel/parser': 7.23.0
    +      '@babel/types': 7.23.0
         dev: true
     
       /@types/babel__traverse@7.20.1:
         resolution: {integrity: sha512-MitHFXnhtgwsGZWtT68URpOvLN4EREih1u3QtQiN4VdAxWKRVvGCSvw/Qth0M0Qq3pJpnGOu5JaM/ydK7OGbqg==}
         dependencies:
    -      '@babel/types': 7.22.10
    +      '@babel/types': 7.23.0
         dev: true
     
       /@types/body-parser@1.19.2:
         resolution: {integrity: sha512-ALYone6pm6QmwZoAgeyNksccT9Q4AWZQ6PvfwR37GT6r6FWUPguq6sUmNGSMV2Wr761oQoBxwGGa6DR5o1DC9g==}
         dependencies:
           '@types/connect': 3.4.35
    -      '@types/node': 18.16.0
    +      '@types/node': 18.17.5
         dev: true
     
       /@types/bonjour@3.5.10:
         resolution: {integrity: sha512-p7ienRMiS41Nu2/igbJxxLDWrSZ0WxM8UQgCeO9KhoVF7cOVFkrKsiDr1EsJIla8vV3oEEjGcz11jc5yimhzZw==}
         dependencies:
    -      '@types/node': 18.17.6
    +      '@types/node': 18.17.5
         dev: true
     
       /@types/braces@3.0.2:
    @@ -4171,7 +4110,7 @@ packages:
         dependencies:
           '@types/http-cache-semantics': 4.0.1
           '@types/keyv': 3.1.4
    -      '@types/node': 18.17.6
    +      '@types/node': 18.17.5
           '@types/responselike': 1.0.0
         dev: true
     
    @@ -4189,19 +4128,19 @@ packages:
         resolution: {integrity: sha512-4x5FkPpLipqwthjPsF7ZRbOv3uoLUFkTA9G9v583qi4pACvq0uTELrB8OLUzPWUI4IJIyvM85vzkV1nyiI2Lig==}
         dependencies:
           '@types/express-serve-static-core': 4.17.35
    -      '@types/node': 18.17.6
    +      '@types/node': 18.17.5
         dev: true
     
       /@types/connect@3.4.35:
         resolution: {integrity: sha512-cdeYyv4KWoEgpBISTxWvqYsVy444DOqehiF3fM3ne10AmJ62RSyNkUnxMJXHQWRQQX2eR94m5y1IZyDwBjV9FQ==}
         dependencies:
    -      '@types/node': 18.17.6
    +      '@types/node': 18.17.5
         dev: true
     
       /@types/cors@2.8.13:
         resolution: {integrity: sha512-RG8AStHlUiV5ysZQKq97copd2UmVYw3/pRMLefISZ3S1hK104Cwm7iLQ3fTKx+lsUH2CE8FlLaYeEA2LSeqYUA==}
         dependencies:
    -      '@types/node': 18.16.0
    +      '@types/node': 18.17.5
         dev: true
     
       /@types/cytoscape@3.19.9:
    @@ -4421,13 +4360,6 @@ packages:
           '@types/estree': 1.0.1
         dev: true
     
    -  /@types/eslint@8.37.0:
    -    resolution: {integrity: sha512-Piet7dG2JBuDIfohBngQ3rCt7MgO9xCO4xIMKxBThCq5PNRB91IjlJ10eJVwfoNtvTErmxLzwBZ7rHZtbOMmFQ==}
    -    dependencies:
    -      '@types/estree': 1.0.1
    -      '@types/json-schema': 7.0.12
    -    dev: true
    -
       /@types/eslint@8.44.2:
         resolution: {integrity: sha512-sdPRb9K6iL5XZOmBubg8yiFp5yS/JdUDQsq5e6h95km91MCYMuvp7mh1fjPEYUhvHepKpZOjnEaMBR4PxjWDzg==}
         dependencies:
    @@ -4446,7 +4378,7 @@ packages:
       /@types/express-serve-static-core@4.17.35:
         resolution: {integrity: sha512-wALWQwrgiB2AWTT91CB62b6Yt0sNHpznUXeZEcnPU3DRdlDIz74x8Qg1UUYKSVFi+va5vKOLYRBI1bRKiLLKIg==}
         dependencies:
    -      '@types/node': 18.16.0
    +      '@types/node': 18.17.5
           '@types/qs': 6.9.7
           '@types/range-parser': 1.2.4
           '@types/send': 0.17.1
    @@ -4473,13 +4405,13 @@ packages:
         resolution: {integrity: sha512-ZUxbzKl0IfJILTS6t7ip5fQQM/J3TJYubDm3nMbgubNNYS62eXeUpoLUC8/7fJNiFYHTrGPQn7hspDUzIHX3UA==}
         dependencies:
           '@types/minimatch': 5.1.2
    -      '@types/node': 18.17.6
    +      '@types/node': 18.17.5
         dev: true
     
       /@types/graceful-fs@4.1.6:
         resolution: {integrity: sha512-Sig0SNORX9fdW+bQuTEovKj3uHcUL6LQKbCrrqb1X7J6/ReAbhCXRAhc+SMejhLELFj2QcyuxmUooZ4bt5ReSw==}
         dependencies:
    -      '@types/node': 18.17.6
    +      '@types/node': 18.17.5
         dev: true
     
       /@types/http-cache-semantics@4.0.1:
    @@ -4493,7 +4425,7 @@ packages:
       /@types/http-proxy@1.17.11:
         resolution: {integrity: sha512-HC8G7c1WmaF2ekqpnFq626xd3Zz0uvaqFmBJNRZCGEZCXkvSdJoNFn/8Ygbd9fKNQj8UzLdCETaI0UWPAjK7IA==}
         dependencies:
    -      '@types/node': 18.17.6
    +      '@types/node': 18.17.5
         dev: true
     
       /@types/istanbul-lib-coverage@2.0.4:
    @@ -4519,7 +4451,7 @@ packages:
       /@types/jsdom@21.1.1:
         resolution: {integrity: sha512-cZFuoVLtzKP3gmq9eNosUL1R50U+USkbLtUQ1bYVgl/lKp0FZM7Cq4aIHAL8oIvQ17uSHi7jXPtfDOdjPwBE7A==}
         dependencies:
    -      '@types/node': 18.17.6
    +      '@types/node': 18.17.5
           '@types/tough-cookie': 4.0.2
           parse5: 7.1.2
         dev: true
    @@ -4531,7 +4463,7 @@ packages:
       /@types/keyv@3.1.4:
         resolution: {integrity: sha512-BQ5aZNSCpj7D6K2ksrRCTmKRLEpnPvWDiLPfoGyhZ++8YtiK9d/3DBKPJgry359X/P1PfruyYwvnvwFjuEiEIg==}
         dependencies:
    -      '@types/node': 18.17.6
    +      '@types/node': 18.17.5
         dev: true
     
       /@types/linkify-it@3.0.2:
    @@ -4544,10 +4476,6 @@ packages:
           '@types/lodash': 4.14.197
         dev: true
     
    -  /@types/lodash@4.14.194:
    -    resolution: {integrity: sha512-r22s9tAS7imvBt2lyHC9B8AGwWnXaYb1tY09oyLkXDs4vArpYJzw09nj8MLx5VfciBPGIb+ZwG0ssYnEPJxn/g==}
    -    dev: true
    -
       /@types/lodash@4.14.197:
         resolution: {integrity: sha512-BMVOiWs0uNxHVlHBgzTIqJYmj+PgCo4euloGF+5m4okL3rEYzM2EEv78mw8zWSMM57dM7kVIgJ2QDvwHSoCI5g==}
         dev: true
    @@ -4566,12 +4494,6 @@ packages:
           '@types/mdurl': 1.0.2
         dev: true
     
    -  /@types/mdast@3.0.11:
    -    resolution: {integrity: sha512-Y/uImid8aAwrEA24/1tcRZwpxX3pIFTSilcNDKSPn+Y2iDywSEachzRuvgAYYLR3wpGXAsMbv5lvKLDZLeYPAw==}
    -    dependencies:
    -      '@types/unist': 3.0.0
    -    dev: true
    -
       /@types/mdast@3.0.12:
         resolution: {integrity: sha512-DT+iNIRNX884cx0/Q1ja7NyUPpZuv0KPyL5rGNxm1WC1OtHstl7n4Jb7nk+xacNShQMbczJjt8uFzznpp6kYBg==}
         dependencies:
    @@ -4606,16 +4528,12 @@ packages:
       /@types/ms@0.7.31:
         resolution: {integrity: sha512-iiUgKzV9AuaEkZqkOLDIvlQiL6ltuZd9tGcW3gwpnX8JbuiuhFlEGmmFXEXkN50Cvq7Os88IY2v0dkDqXYWVgA==}
     
    -  /@types/node@14.18.54:
    -    resolution: {integrity: sha512-uq7O52wvo2Lggsx1x21tKZgqkJpvwCseBBPtX/nKQfpVlEsLOb11zZ1CRsWUKvJF0+lzuA9jwvA7Pr2Wt7i3xw==}
    +  /@types/node@16.18.40:
    +    resolution: {integrity: sha512-+yno3ItTEwGxXiS/75Q/aHaa5srkpnJaH+kdkTVJ3DtJEwv92itpKbxU+FjPoh2m/5G9zmUQfrL4A4C13c+iGA==}
         dev: true
     
    -  /@types/node@18.16.0:
    -    resolution: {integrity: sha512-BsAaKhB+7X+H4GnSjGhJG9Qi8Tw+inU9nJDwmD5CgOmBLEI6ArdhikpLX7DjbjDRDTbqZzU2LSQNZg8WGPiSZQ==}
    -
    -  /@types/node@18.17.6:
    -    resolution: {integrity: sha512-fGmT/P7z7ecA6bv/ia5DlaWCH4YeZvAQMNpUhrJjtAhOhZfoxS1VLUgU2pdk63efSjQaOJWdXMuAJsws+8I6dg==}
    -    dev: true
    +  /@types/node@18.17.5:
    +    resolution: {integrity: sha512-xNbS75FxH6P4UXTPUJp/zNPq6/xsfdJKussCWNOnz4aULWIRwMgP1LgaB5RiBnMX1DPCYenuqGZfnIAx5mbFLA==}
     
       /@types/node@20.4.7:
         resolution: {integrity: sha512-bUBrPjEry2QUTsnuEjzjbS7voGWCc30W0qzgMf90GPeDGFRakvrz47ju+oqDAKCXLUCe39u57/ORMl/O/04/9g==}
    @@ -4646,13 +4564,13 @@ packages:
       /@types/resolve@1.17.1:
         resolution: {integrity: sha512-yy7HuzQhj0dhGpD8RLXSZWEkLsV9ibvxvi6EiJ3bkqLAO1RGo0WbkWQiwpRlSFymTJRz0d3k5LM3kkx8ArDbLw==}
         dependencies:
    -      '@types/node': 18.17.6
    +      '@types/node': 18.17.5
         dev: true
     
       /@types/responselike@1.0.0:
         resolution: {integrity: sha512-85Y2BjiufFzaMIlvJDvTTB8Fxl2xfLo4HgmHzVBz08w4wDePCTjYw66PdrolO0kzli3yam/YCgRufyo1DdQVTA==}
         dependencies:
    -      '@types/node': 18.17.6
    +      '@types/node': 18.17.5
         dev: true
     
       /@types/retry@0.12.0:
    @@ -4662,7 +4580,7 @@ packages:
       /@types/rollup-plugin-visualizer@4.2.1:
         resolution: {integrity: sha512-Fk4y0EgmsSbvbayYhtSI9+cGvgw1rcQ9RlbExkQt4ivXRdiEwFKuRpxNuJCr0JktXIvOPUuPR7GSmtyZu0dujQ==}
         dependencies:
    -      '@types/node': 18.16.0
    +      '@types/node': 18.17.5
           rollup: 2.79.1
         dev: true
     
    @@ -4674,7 +4592,7 @@ packages:
         resolution: {integrity: sha512-Cwo8LE/0rnvX7kIIa3QHCkcuF21c05Ayb0ZfxPiv0W8VRiZiNW/WuRupHKpqqGVGf7SUA44QSOUKaEd9lIrd/Q==}
         dependencies:
           '@types/mime': 1.3.2
    -      '@types/node': 18.17.6
    +      '@types/node': 18.17.5
         dev: true
     
       /@types/serve-index@1.9.1:
    @@ -4688,7 +4606,7 @@ packages:
         dependencies:
           '@types/http-errors': 2.0.1
           '@types/mime': 3.0.1
    -      '@types/node': 18.17.6
    +      '@types/node': 18.17.5
         dev: true
     
       /@types/sinonjs__fake-timers@8.1.1:
    @@ -4702,7 +4620,7 @@ packages:
       /@types/sockjs@0.3.33:
         resolution: {integrity: sha512-f0KEEe05NvUnat+boPTZ0dgaLZ4SfSouXUgv5noUiefG2ajgKjmETo9ZJyuqsl7dfl2aHlLJUiki6B4ZYldiiw==}
         dependencies:
    -      '@types/node': 18.17.6
    +      '@types/node': 18.17.5
         dev: true
     
       /@types/stack-utils@2.0.1:
    @@ -4724,10 +4642,6 @@ packages:
       /@types/unist@2.0.7:
         resolution: {integrity: sha512-cputDpIbFgLUaGQn6Vqg3/YsJwxUwHLO13v3i5ouxT4lat0khip9AEWxtERujXV9wxIB1EyF97BSJFt6vpdI8g==}
     
    -  /@types/unist@3.0.0:
    -    resolution: {integrity: sha512-MFETx3tbTjE7Uk6vvnWINA/1iJ7LuMdO4fcq8UfF0pRbj01aGLduVvQcRyswuACJdpnHgg8E3rQLhaRdNEJS0w==}
    -    dev: true
    -
       /@types/uuid@9.0.1:
         resolution: {integrity: sha512-rFT3ak0/2trgvp4yYZo5iKFEPsET7vKydKF+VRCxlQ9bpheehyAJH89dAkaLEq/j/RZXJIqcgsmPJKUP1Z28HA==}
         dev: true
    @@ -4740,16 +4654,14 @@ packages:
         resolution: {integrity: sha512-4p9vcSmxAayx72yn70joFoL44c9MO/0+iVEBIQXe3v2h2SiAsEIo/G5v6ObFWvNKRFjbrVadNf9LqEEZeQPzdA==}
         dev: true
     
    -  /@types/ws@8.5.4:
    -    resolution: {integrity: sha512-zdQDHKUgcX/zBc4GrwsE/7dVdAD8JR4EuiAXiiUhhfyIJXXb2+PrGshFyeXWQPMmmZ2XxgaqclgpIC7eTXc1mg==}
    -    dependencies:
    -      '@types/node': 18.17.6
    +  /@types/web-bluetooth@0.0.18:
    +    resolution: {integrity: sha512-v/ZHEj9xh82usl8LMR3GarzFY1IrbXJw5L4QfQhokjRV91q+SelFqxQWSep1ucXEZ22+dSTwLFkXeur25sPIbw==}
         dev: true
     
       /@types/ws@8.5.5:
         resolution: {integrity: sha512-lwhs8hktwxSjf9UaZ9tG5M03PGogvFaH8gUgLNbN9HKIg0dvv6q+gkSuJ8HN4/VbyxkuLzCjlN7GquQ0gUJfIg==}
         dependencies:
    -      '@types/node': 18.17.6
    +      '@types/node': 18.17.5
         dev: true
     
       /@types/yargs-parser@21.0.0:
    @@ -4766,11 +4678,11 @@ packages:
         resolution: {integrity: sha512-Cn6WYCm0tXv8p6k+A8PvbDG763EDpBoTzHdA+Q/MF6H3sapGjCm9NzoaJncJS9tUKSuCoDs9XHxYYsQDgxR6kw==}
         requiresBuild: true
         dependencies:
    -      '@types/node': 18.17.6
    +      '@types/node': 18.17.5
         dev: true
         optional: true
     
    -  /@typescript-eslint/eslint-plugin@5.59.0(@typescript-eslint/parser@5.59.0)(eslint@8.47.0)(typescript@5.1.3):
    +  /@typescript-eslint/eslint-plugin@5.59.0(@typescript-eslint/parser@5.59.0)(eslint@8.47.0)(typescript@5.0.4):
         resolution: {integrity: sha512-p0QgrEyrxAWBecR56gyn3wkG15TJdI//eetInP3zYRewDh0XS+DhB3VUAd3QqvziFsfaQIoIuZMxZRB7vXYaYw==}
         engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0}
         peerDependencies:
    @@ -4782,24 +4694,24 @@ packages:
             optional: true
         dependencies:
           '@eslint-community/regexpp': 4.6.2
    -      '@typescript-eslint/parser': 5.59.0(eslint@8.47.0)(typescript@5.1.3)
    +      '@typescript-eslint/parser': 5.59.0(eslint@8.47.0)(typescript@5.0.4)
           '@typescript-eslint/scope-manager': 5.59.0
    -      '@typescript-eslint/type-utils': 5.59.0(eslint@8.47.0)(typescript@5.1.3)
    -      '@typescript-eslint/utils': 5.59.0(eslint@8.47.0)(typescript@5.1.3)
    +      '@typescript-eslint/type-utils': 5.59.0(eslint@8.47.0)(typescript@5.0.4)
    +      '@typescript-eslint/utils': 5.59.0(eslint@8.47.0)(typescript@5.0.4)
           debug: 4.3.4(supports-color@8.1.1)
           eslint: 8.47.0
           grapheme-splitter: 1.0.4
           ignore: 5.2.4
           natural-compare-lite: 1.4.0
           semver: 7.5.4
    -      tsutils: 3.21.0(typescript@5.1.3)
    -      typescript: 5.1.3
    +      tsutils: 3.21.0(typescript@5.0.4)
    +      typescript: 5.0.4
         transitivePeerDependencies:
           - supports-color
         dev: true
     
    -  /@typescript-eslint/eslint-plugin@6.9.1(@typescript-eslint/parser@6.9.1)(eslint@8.39.0)(typescript@5.1.3):
    -    resolution: {integrity: sha512-w0tiiRc9I4S5XSXXrMHOWgHgxbrBn1Ro+PmiYhSg2ZVdxrAJtQgzU5o2m1BfP6UOn7Vxcc6152vFjQfmZR4xEg==}
    +  /@typescript-eslint/eslint-plugin@6.7.2(@typescript-eslint/parser@6.7.2)(eslint@8.47.0)(typescript@5.1.6):
    +    resolution: {integrity: sha512-ooaHxlmSgZTM6CHYAFRlifqh1OAr3PAQEwi7lhYhaegbnXrnh7CDcHmc3+ihhbQC7H0i4JF0psI5ehzkF6Yl6Q==}
         engines: {node: ^16.0.0 || >=18.0.0}
         peerDependencies:
           '@typescript-eslint/parser': ^6.0.0 || ^6.0.0-alpha
    @@ -4810,24 +4722,24 @@ packages:
             optional: true
         dependencies:
           '@eslint-community/regexpp': 4.6.2
    -      '@typescript-eslint/parser': 6.9.1(eslint@8.39.0)(typescript@5.1.3)
    -      '@typescript-eslint/scope-manager': 6.9.1
    -      '@typescript-eslint/type-utils': 6.9.1(eslint@8.39.0)(typescript@5.1.3)
    -      '@typescript-eslint/utils': 6.9.1(eslint@8.39.0)(typescript@5.1.3)
    -      '@typescript-eslint/visitor-keys': 6.9.1
    +      '@typescript-eslint/parser': 6.7.2(eslint@8.47.0)(typescript@5.1.6)
    +      '@typescript-eslint/scope-manager': 6.7.2
    +      '@typescript-eslint/type-utils': 6.7.2(eslint@8.47.0)(typescript@5.1.6)
    +      '@typescript-eslint/utils': 6.7.2(eslint@8.47.0)(typescript@5.1.6)
    +      '@typescript-eslint/visitor-keys': 6.7.2
           debug: 4.3.4(supports-color@8.1.1)
    -      eslint: 8.39.0
    +      eslint: 8.47.0
           graphemer: 1.4.0
           ignore: 5.2.4
           natural-compare: 1.4.0
           semver: 7.5.4
    -      ts-api-utils: 1.0.3(typescript@5.1.3)
    -      typescript: 5.1.3
    +      ts-api-utils: 1.0.3(typescript@5.1.6)
    +      typescript: 5.1.6
         transitivePeerDependencies:
           - supports-color
         dev: true
     
    -  /@typescript-eslint/parser@5.59.0(eslint@8.47.0)(typescript@5.1.3):
    +  /@typescript-eslint/parser@5.59.0(eslint@8.47.0)(typescript@5.0.4):
         resolution: {integrity: sha512-qK9TZ70eJtjojSUMrrEwA9ZDQ4N0e/AuoOIgXuNBorXYcBDk397D2r5MIe1B3cok/oCtdNC5j+lUUpVB+Dpb+w==}
         engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0}
         peerDependencies:
    @@ -4839,16 +4751,16 @@ packages:
         dependencies:
           '@typescript-eslint/scope-manager': 5.59.0
           '@typescript-eslint/types': 5.59.0
    -      '@typescript-eslint/typescript-estree': 5.59.0(typescript@5.1.3)
    +      '@typescript-eslint/typescript-estree': 5.59.0(typescript@5.0.4)
           debug: 4.3.4(supports-color@8.1.1)
           eslint: 8.47.0
    -      typescript: 5.1.3
    +      typescript: 5.0.4
         transitivePeerDependencies:
           - supports-color
         dev: true
     
    -  /@typescript-eslint/parser@6.9.1(eslint@8.39.0)(typescript@5.1.3):
    -    resolution: {integrity: sha512-C7AK2wn43GSaCUZ9do6Ksgi2g3mwFkMO3Cis96kzmgudoVaKyt62yNzJOktP0HDLb/iO2O0n2lBOzJgr6Q/cyg==}
    +  /@typescript-eslint/parser@6.7.2(eslint@8.47.0)(typescript@5.1.6):
    +    resolution: {integrity: sha512-KA3E4ox0ws+SPyxQf9iSI25R6b4Ne78ORhNHeVKrPQnoYsb9UhieoiRoJgrzgEeKGOXhcY1i8YtOeCHHTDa6Fw==}
         engines: {node: ^16.0.0 || >=18.0.0}
         peerDependencies:
           eslint: ^7.0.0 || ^8.0.0
    @@ -4857,13 +4769,13 @@ packages:
           typescript:
             optional: true
         dependencies:
    -      '@typescript-eslint/scope-manager': 6.9.1
    -      '@typescript-eslint/types': 6.9.1
    -      '@typescript-eslint/typescript-estree': 6.9.1(typescript@5.1.3)
    -      '@typescript-eslint/visitor-keys': 6.9.1
    +      '@typescript-eslint/scope-manager': 6.7.2
    +      '@typescript-eslint/types': 6.7.2
    +      '@typescript-eslint/typescript-estree': 6.7.2(typescript@5.1.6)
    +      '@typescript-eslint/visitor-keys': 6.7.2
           debug: 4.3.4(supports-color@8.1.1)
    -      eslint: 8.39.0
    -      typescript: 5.1.3
    +      eslint: 8.47.0
    +      typescript: 5.1.6
         transitivePeerDependencies:
           - supports-color
         dev: true
    @@ -4884,15 +4796,15 @@ packages:
           '@typescript-eslint/visitor-keys': 5.62.0
         dev: true
     
    -  /@typescript-eslint/scope-manager@6.9.1:
    -    resolution: {integrity: sha512-38IxvKB6NAne3g/+MyXMs2Cda/Sz+CEpmm+KLGEM8hx/CvnSRuw51i8ukfwB/B/sESdeTGet1NH1Wj7I0YXswg==}
    +  /@typescript-eslint/scope-manager@6.7.2:
    +    resolution: {integrity: sha512-bgi6plgyZjEqapr7u2mhxGR6E8WCzKNUFWNh6fkpVe9+yzRZeYtDTbsIBzKbcxI+r1qVWt6VIoMSNZ4r2A+6Yw==}
         engines: {node: ^16.0.0 || >=18.0.0}
         dependencies:
    -      '@typescript-eslint/types': 6.9.1
    -      '@typescript-eslint/visitor-keys': 6.9.1
    +      '@typescript-eslint/types': 6.7.2
    +      '@typescript-eslint/visitor-keys': 6.7.2
         dev: true
     
    -  /@typescript-eslint/type-utils@5.59.0(eslint@8.47.0)(typescript@5.1.3):
    +  /@typescript-eslint/type-utils@5.59.0(eslint@8.47.0)(typescript@5.0.4):
         resolution: {integrity: sha512-d/B6VSWnZwu70kcKQSCqjcXpVH+7ABKH8P1KNn4K7j5PXXuycZTPXF44Nui0TEm6rbWGi8kc78xRgOC4n7xFgA==}
         engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0}
         peerDependencies:
    @@ -4902,18 +4814,18 @@ packages:
           typescript:
             optional: true
         dependencies:
    -      '@typescript-eslint/typescript-estree': 5.59.0(typescript@5.1.3)
    -      '@typescript-eslint/utils': 5.59.0(eslint@8.47.0)(typescript@5.1.3)
    +      '@typescript-eslint/typescript-estree': 5.59.0(typescript@5.0.4)
    +      '@typescript-eslint/utils': 5.59.0(eslint@8.47.0)(typescript@5.0.4)
           debug: 4.3.4(supports-color@8.1.1)
           eslint: 8.47.0
    -      tsutils: 3.21.0(typescript@5.1.3)
    -      typescript: 5.1.3
    +      tsutils: 3.21.0(typescript@5.0.4)
    +      typescript: 5.0.4
         transitivePeerDependencies:
           - supports-color
         dev: true
     
    -  /@typescript-eslint/type-utils@6.9.1(eslint@8.39.0)(typescript@5.1.3):
    -    resolution: {integrity: sha512-eh2oHaUKCK58qIeYp19F5V5TbpM52680sB4zNSz29VBQPTWIlE/hCj5P5B1AChxECe/fmZlspAWFuRniep1Skg==}
    +  /@typescript-eslint/type-utils@6.7.2(eslint@8.47.0)(typescript@5.1.6):
    +    resolution: {integrity: sha512-36F4fOYIROYRl0qj95dYKx6kybddLtsbmPIYNK0OBeXv2j9L5nZ17j9jmfy+bIDHKQgn2EZX+cofsqi8NPATBQ==}
         engines: {node: ^16.0.0 || >=18.0.0}
         peerDependencies:
           eslint: ^7.0.0 || ^8.0.0
    @@ -4922,12 +4834,12 @@ packages:
           typescript:
             optional: true
         dependencies:
    -      '@typescript-eslint/typescript-estree': 6.9.1(typescript@5.1.3)
    -      '@typescript-eslint/utils': 6.9.1(eslint@8.39.0)(typescript@5.1.3)
    +      '@typescript-eslint/typescript-estree': 6.7.2(typescript@5.1.6)
    +      '@typescript-eslint/utils': 6.7.2(eslint@8.47.0)(typescript@5.1.6)
           debug: 4.3.4(supports-color@8.1.1)
    -      eslint: 8.39.0
    -      ts-api-utils: 1.0.3(typescript@5.1.3)
    -      typescript: 5.1.3
    +      eslint: 8.47.0
    +      ts-api-utils: 1.0.3(typescript@5.1.6)
    +      typescript: 5.1.6
         transitivePeerDependencies:
           - supports-color
         dev: true
    @@ -4942,12 +4854,12 @@ packages:
         engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0}
         dev: true
     
    -  /@typescript-eslint/types@6.9.1:
    -    resolution: {integrity: sha512-BUGslGOb14zUHOUmDB2FfT6SI1CcZEJYfF3qFwBeUrU6srJfzANonwRYHDpLBuzbq3HaoF2XL2hcr01c8f8OaQ==}
    +  /@typescript-eslint/types@6.7.2:
    +    resolution: {integrity: sha512-flJYwMYgnUNDAN9/GAI3l8+wTmvTYdv64fcH8aoJK76Y+1FCZ08RtI5zDerM/FYT5DMkAc+19E4aLmd5KqdFyg==}
         engines: {node: ^16.0.0 || >=18.0.0}
         dev: true
     
    -  /@typescript-eslint/typescript-estree@5.59.0(typescript@5.1.3):
    +  /@typescript-eslint/typescript-estree@5.59.0(typescript@5.0.4):
         resolution: {integrity: sha512-sUNnktjmI8DyGzPdZ8dRwW741zopGxltGs/SAPgGL/AAgDpiLsCFLcMNSpbfXfmnNeHmK9h3wGmCkGRGAoUZAg==}
         engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0}
         peerDependencies:
    @@ -4962,13 +4874,13 @@ packages:
           globby: 11.1.0
           is-glob: 4.0.3
           semver: 7.5.4
    -      tsutils: 3.21.0(typescript@5.1.3)
    -      typescript: 5.1.3
    +      tsutils: 3.21.0(typescript@5.0.4)
    +      typescript: 5.0.4
         transitivePeerDependencies:
           - supports-color
         dev: true
     
    -  /@typescript-eslint/typescript-estree@5.62.0(typescript@5.1.3):
    +  /@typescript-eslint/typescript-estree@5.62.0(typescript@5.1.6):
         resolution: {integrity: sha512-CmcQ6uY7b9y694lKdRB8FEel7JbU/40iSAPomu++SjLMntB+2Leay2LO6i8VnJk58MtE9/nQSFIH6jpyRWyYzA==}
         engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0}
         peerDependencies:
    @@ -4983,14 +4895,14 @@ packages:
           globby: 11.1.0
           is-glob: 4.0.3
           semver: 7.5.4
    -      tsutils: 3.21.0(typescript@5.1.3)
    -      typescript: 5.1.3
    +      tsutils: 3.21.0(typescript@5.1.6)
    +      typescript: 5.1.6
         transitivePeerDependencies:
           - supports-color
         dev: true
     
    -  /@typescript-eslint/typescript-estree@6.9.1(typescript@5.1.3):
    -    resolution: {integrity: sha512-U+mUylTHfcqeO7mLWVQ5W/tMLXqVpRv61wm9ZtfE5egz7gtnmqVIw9ryh0mgIlkKk9rZLY3UHygsBSdB9/ftyw==}
    +  /@typescript-eslint/typescript-estree@6.7.2(typescript@5.1.6):
    +    resolution: {integrity: sha512-kiJKVMLkoSciGyFU0TOY0fRxnp9qq1AzVOHNeN1+B9erKFCJ4Z8WdjAkKQPP+b1pWStGFqezMLltxO+308dJTQ==}
         engines: {node: ^16.0.0 || >=18.0.0}
         peerDependencies:
           typescript: '*'
    @@ -4998,19 +4910,19 @@ packages:
           typescript:
             optional: true
         dependencies:
    -      '@typescript-eslint/types': 6.9.1
    -      '@typescript-eslint/visitor-keys': 6.9.1
    +      '@typescript-eslint/types': 6.7.2
    +      '@typescript-eslint/visitor-keys': 6.7.2
           debug: 4.3.4(supports-color@8.1.1)
           globby: 11.1.0
           is-glob: 4.0.3
           semver: 7.5.4
    -      ts-api-utils: 1.0.3(typescript@5.1.3)
    -      typescript: 5.1.3
    +      ts-api-utils: 1.0.3(typescript@5.1.6)
    +      typescript: 5.1.6
         transitivePeerDependencies:
           - supports-color
         dev: true
     
    -  /@typescript-eslint/utils@5.59.0(eslint@8.47.0)(typescript@5.1.3):
    +  /@typescript-eslint/utils@5.59.0(eslint@8.47.0)(typescript@5.0.4):
         resolution: {integrity: sha512-GGLFd+86drlHSvPgN/el6dRQNYYGOvRSDVydsUaQluwIW3HvbXuxyuD5JETvBt/9qGYe+lOrDk6gRrWOHb/FvA==}
         engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0}
         peerDependencies:
    @@ -5021,7 +4933,7 @@ packages:
           '@types/semver': 7.5.0
           '@typescript-eslint/scope-manager': 5.59.0
           '@typescript-eslint/types': 5.59.0
    -      '@typescript-eslint/typescript-estree': 5.59.0(typescript@5.1.3)
    +      '@typescript-eslint/typescript-estree': 5.59.0(typescript@5.0.4)
           eslint: 8.47.0
           eslint-scope: 5.1.1
           semver: 7.5.4
    @@ -5030,19 +4942,19 @@ packages:
           - typescript
         dev: true
     
    -  /@typescript-eslint/utils@5.62.0(eslint@8.39.0)(typescript@5.1.3):
    +  /@typescript-eslint/utils@5.62.0(eslint@8.47.0)(typescript@5.1.6):
         resolution: {integrity: sha512-n8oxjeb5aIbPFEtmQxQYOLI0i9n5ySBEY/ZEHHZqKQSFnxio1rv6dthascc9dLuwrL0RC5mPCxB7vnAVGAYWAQ==}
         engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0}
         peerDependencies:
           eslint: ^6.0.0 || ^7.0.0 || ^8.0.0
         dependencies:
    -      '@eslint-community/eslint-utils': 4.4.0(eslint@8.39.0)
    +      '@eslint-community/eslint-utils': 4.4.0(eslint@8.47.0)
           '@types/json-schema': 7.0.12
           '@types/semver': 7.5.0
           '@typescript-eslint/scope-manager': 5.62.0
           '@typescript-eslint/types': 5.62.0
    -      '@typescript-eslint/typescript-estree': 5.62.0(typescript@5.1.3)
    -      eslint: 8.39.0
    +      '@typescript-eslint/typescript-estree': 5.62.0(typescript@5.1.6)
    +      eslint: 8.47.0
           eslint-scope: 5.1.1
           semver: 7.5.4
         transitivePeerDependencies:
    @@ -5050,19 +4962,19 @@ packages:
           - typescript
         dev: true
     
    -  /@typescript-eslint/utils@6.9.1(eslint@8.39.0)(typescript@5.1.3):
    -    resolution: {integrity: sha512-L1T0A5nFdQrMVunpZgzqPL6y2wVreSyHhKGZryS6jrEN7bD9NplVAyMryUhXsQ4TWLnZmxc2ekar/lSGIlprCA==}
    +  /@typescript-eslint/utils@6.7.2(eslint@8.47.0)(typescript@5.1.6):
    +    resolution: {integrity: sha512-ZCcBJug/TS6fXRTsoTkgnsvyWSiXwMNiPzBUani7hDidBdj1779qwM1FIAmpH4lvlOZNF3EScsxxuGifjpLSWQ==}
         engines: {node: ^16.0.0 || >=18.0.0}
         peerDependencies:
           eslint: ^7.0.0 || ^8.0.0
         dependencies:
    -      '@eslint-community/eslint-utils': 4.4.0(eslint@8.39.0)
    +      '@eslint-community/eslint-utils': 4.4.0(eslint@8.47.0)
           '@types/json-schema': 7.0.12
           '@types/semver': 7.5.0
    -      '@typescript-eslint/scope-manager': 6.9.1
    -      '@typescript-eslint/types': 6.9.1
    -      '@typescript-eslint/typescript-estree': 6.9.1(typescript@5.1.3)
    -      eslint: 8.39.0
    +      '@typescript-eslint/scope-manager': 6.7.2
    +      '@typescript-eslint/types': 6.7.2
    +      '@typescript-eslint/typescript-estree': 6.7.2(typescript@5.1.6)
    +      eslint: 8.47.0
           semver: 7.5.4
         transitivePeerDependencies:
           - supports-color
    @@ -5085,40 +4997,40 @@ packages:
           eslint-visitor-keys: 3.4.3
         dev: true
     
    -  /@typescript-eslint/visitor-keys@6.9.1:
    -    resolution: {integrity: sha512-MUaPUe/QRLEffARsmNfmpghuQkW436DvESW+h+M52w0coICHRfD6Np9/K6PdACwnrq1HmuLl+cSPZaJmeVPkSw==}
    +  /@typescript-eslint/visitor-keys@6.7.2:
    +    resolution: {integrity: sha512-uVw9VIMFBUTz8rIeaUT3fFe8xIUx8r4ywAdlQv1ifH+6acn/XF8Y6rwJ7XNmkNMDrTW+7+vxFFPIF40nJCVsMQ==}
         engines: {node: ^16.0.0 || >=18.0.0}
         dependencies:
    -      '@typescript-eslint/types': 6.9.1
    +      '@typescript-eslint/types': 6.7.2
           eslint-visitor-keys: 3.4.3
         dev: true
     
    -  /@unocss/astro@0.56.5(rollup@2.79.1)(vite@4.3.9):
    -    resolution: {integrity: sha512-nkxyGV9mA7DZ5LEr4Gap/SggM60MFNUfn56ngpxCqjQHJOMRJrAcR99hCVn+78vZ9xuZl9HxdIwgZLzn41thMw==}
    +  /@unocss/astro@0.57.1(rollup@2.79.1)(vite@4.4.9):
    +    resolution: {integrity: sha512-KNaqN/SGM/uz1QitajIkzNEw0jy9Zx9Wp8fl4GhfGYEMAN2+M4cuvBZRmlb6cLctSXmSAJQDG91ivbD1JijGnw==}
         peerDependencies:
           vite: ^2.9.0 || ^3.0.0-0 || ^4.0.0
         peerDependenciesMeta:
           vite:
             optional: true
         dependencies:
    -      '@unocss/core': 0.56.5
    -      '@unocss/reset': 0.56.5
    -      '@unocss/vite': 0.56.5(rollup@2.79.1)(vite@4.3.9)
    -      vite: 4.3.9(@types/node@18.16.0)
    +      '@unocss/core': 0.57.1
    +      '@unocss/reset': 0.57.1
    +      '@unocss/vite': 0.57.1(rollup@2.79.1)(vite@4.4.9)
    +      vite: 4.4.9(@types/node@18.17.5)
         transitivePeerDependencies:
           - rollup
         dev: true
     
    -  /@unocss/cli@0.56.5(rollup@2.79.1):
    -    resolution: {integrity: sha512-VYaqu7Dr1n9ebFFdQM+9Jyg/o9BVKRShlV8bQsBS58gkXiWsA/uAl1Uy2vzpLSrT0F6uGyDmYUF6p4DaUnUO+w==}
    +  /@unocss/cli@0.57.1(rollup@2.79.1):
    +    resolution: {integrity: sha512-wKuOaygrPNzDm5L7+2SfHsIi3knJrAQ8nH6OasVqB+bGDz6ybDlULV7wvUco6Os72ydh7YbWC2/WpqFii8U/3w==}
         engines: {node: '>=14'}
         hasBin: true
         dependencies:
           '@ampproject/remapping': 2.2.1
           '@rollup/pluginutils': 5.0.5(rollup@2.79.1)
    -      '@unocss/config': 0.56.5
    -      '@unocss/core': 0.56.5
    -      '@unocss/preset-uno': 0.56.5
    +      '@unocss/config': 0.57.1
    +      '@unocss/core': 0.57.1
    +      '@unocss/preset-uno': 0.57.1
           cac: 6.7.14
           chokidar: 3.5.3
           colorette: 2.0.20
    @@ -5131,171 +5043,172 @@ packages:
           - rollup
         dev: true
     
    -  /@unocss/config@0.56.5:
    -    resolution: {integrity: sha512-rscnFIYgUlN/0hXHdhANyjFcDjDutt3JO0ZRITdNLzoglh7GVNiDTURBJwUZejF/vGJ7IkMd3qOdNhPFuRY1Bg==}
    +  /@unocss/config@0.57.1:
    +    resolution: {integrity: sha512-mbuVO0mH1PX7rEkViMNWb3jG1ji7TUydo2DdnMHhJE+dOrGtnQzhzXGlAd4qqel1fnt/VWuOyZKwJA3QO6VCtg==}
         engines: {node: '>=14'}
         dependencies:
    -      '@unocss/core': 0.56.5
    -      unconfig: 0.3.10
    +      '@unocss/core': 0.57.1
    +      unconfig: 0.3.11
         dev: true
     
    -  /@unocss/core@0.56.5:
    -    resolution: {integrity: sha512-fx5VhOjSHn0HdV2D34pEwFMAHJcJQRTCp1xEE4GzxY1irXzaa+m2aYf5PZjmDxehiOC16IH7TO9FOWANXk1E0w==}
    +  /@unocss/core@0.57.1:
    +    resolution: {integrity: sha512-cqQW/4gCuk+bFMPg9lBanuRNQ9Lx1l4PpMN/6uKxI5WROpq7ce/Xb4uGvAxKLh3ITtFSpXs2cLfsy7QD6cVD/Q==}
         dev: true
     
    -  /@unocss/extractor-arbitrary-variants@0.56.5:
    -    resolution: {integrity: sha512-p2pyzz/ONvc5CGcaB9OZvWE8qkRSgyuhaQqFQLdBFeUhveHC0CGP0iSnXwBgAFHWM7DJo4/JpWeZ+mBt0ogVLA==}
    +  /@unocss/extractor-arbitrary-variants@0.57.1:
    +    resolution: {integrity: sha512-9s+azHhBnwjxm46TsD1RY0krDAwOR8tcw58Vtl3emd6C0VQsAOdoprt7UHE7GEXMvDVq7nMf8lAT0BM0LteW3w==}
         dependencies:
    -      '@unocss/core': 0.56.5
    +      '@unocss/core': 0.57.1
         dev: true
     
    -  /@unocss/inspector@0.56.5:
    -    resolution: {integrity: sha512-UK/X2JyqxB1uueIFlffFsBioxMptanBbJYjrmOoLPdAFoOPp9o8IIFkFs3OKWc8imvyl3w+F0opncmdJnDMllw==}
    +  /@unocss/inspector@0.57.1:
    +    resolution: {integrity: sha512-qV7ta7iHGX2EpZJ4IWY/05kgyhKFeWlvVJbrOnGsaH8gVt33T/43YAhB/8K5GIXBXIwkhwk13iB13nlg2gSheg==}
         dependencies:
    -      '@unocss/rule-utils': 0.56.5
    +      '@unocss/rule-utils': 0.57.1
           gzip-size: 6.0.0
           sirv: 2.0.3
         dev: true
     
    -  /@unocss/postcss@0.56.5(postcss@8.4.28):
    -    resolution: {integrity: sha512-oDY1vCdrCQZx/oxDkq3z77a0RoO+WVQT090oDZzLFO/cPWH3elSOHwnoJU/KzrpJLbUFpyTQKZ/k3VbBaEdGTA==}
    +  /@unocss/postcss@0.57.1(postcss@8.4.31):
    +    resolution: {integrity: sha512-DexrV+v/qkVh6t660rXigNr2Y6lON8jxD1z2KVk2bjHKhFflF6q6seps6d/MquyLJI1mXF2uANTeFAeL2q6evw==}
         engines: {node: '>=14'}
         peerDependencies:
           postcss: ^8.4.21
         dependencies:
    -      '@unocss/config': 0.56.5
    -      '@unocss/core': 0.56.5
    -      '@unocss/rule-utils': 0.56.5
    +      '@unocss/config': 0.57.1
    +      '@unocss/core': 0.57.1
    +      '@unocss/rule-utils': 0.57.1
           css-tree: 2.3.1
           fast-glob: 3.3.1
           magic-string: 0.30.5
    -      postcss: 8.4.28
    +      postcss: 8.4.31
         dev: true
     
    -  /@unocss/preset-attributify@0.56.5:
    -    resolution: {integrity: sha512-476NVv1kUmiD0ObtJceUB1ldiQvGStEUlwoHFOFl8srZbRuRlwq8Uz4sxCePf3sn2FP8UHw+By+nxwxMQuHpww==}
    +  /@unocss/preset-attributify@0.57.1:
    +    resolution: {integrity: sha512-pvGQHaqBlB0jQysWhNbcKLOGrkj8b53k0sAa9LYxQjD1fa8t/dwbuMpZv4twX+gysF0vBhxRoWBPLH1/S6zRZg==}
         dependencies:
    -      '@unocss/core': 0.56.5
    +      '@unocss/core': 0.57.1
         dev: true
     
    -  /@unocss/preset-icons@0.56.5:
    -    resolution: {integrity: sha512-Pc973z/M7+TsIPRli9xSE+rjzQnX9r0PppTOD5lrD8PqvNSGIrSWZUFDc5NqL09hKHkIdVrDLC+2ouLYqTI7iA==}
    +  /@unocss/preset-icons@0.57.1:
    +    resolution: {integrity: sha512-ve4jC6yREfS0mv97DCld9xLjMuiSCcsQPKucdtpUfCjLMqtGd1ZGGdFv02Q+92NkW7HDfgj+izEw1SKh9695Ow==}
         dependencies:
           '@iconify/utils': 2.1.11
    -      '@unocss/core': 0.56.5
    +      '@unocss/core': 0.57.1
           ofetch: 1.3.3
         transitivePeerDependencies:
           - supports-color
         dev: true
     
    -  /@unocss/preset-mini@0.56.5:
    -    resolution: {integrity: sha512-/KhlThhs1ilauM7MwRSpahLbIPZ5VGeGvaUsU8+ZlNT3sis4yoVYkPtR14tL2IT6jhOU05N/uu3aBj+1bP8GjQ==}
    +  /@unocss/preset-mini@0.57.1:
    +    resolution: {integrity: sha512-v9ZsIUGDfZNXbIrOc7zrBp+RFbFFGSQN/vKIf761js4fJ31j6lan4pPQPGcY17xHConkI1HJT/+yb/UVJaAcHw==}
         dependencies:
    -      '@unocss/core': 0.56.5
    -      '@unocss/extractor-arbitrary-variants': 0.56.5
    -      '@unocss/rule-utils': 0.56.5
    +      '@unocss/core': 0.57.1
    +      '@unocss/extractor-arbitrary-variants': 0.57.1
    +      '@unocss/rule-utils': 0.57.1
         dev: true
     
    -  /@unocss/preset-tagify@0.56.5:
    -    resolution: {integrity: sha512-ANtI7E92fuzF40MD/3V72myNqrcGNXAD3TjXHIxAPQP2F+DcYIl2TBPHVsGUt2Rc5Kid2kvvsWan6sRAQVGPng==}
    +  /@unocss/preset-tagify@0.57.1:
    +    resolution: {integrity: sha512-GV8knxnsOVH/XiG2KB+mVZeEJqr0PZvvkSTPftGPbjttoKVZ+28Y5q9/qezH7p4W6RYVAAK+3qHHy5wWZosiMw==}
         dependencies:
    -      '@unocss/core': 0.56.5
    +      '@unocss/core': 0.57.1
         dev: true
     
    -  /@unocss/preset-typography@0.56.5:
    -    resolution: {integrity: sha512-fA/q5S+s7BV3TRWRvXqG7xpa8WNfG19uaZx288FcKauMR0TX4jUM0EkkngpwrTgeXXohh5awx4OEW3E1s1Q2oA==}
    +  /@unocss/preset-typography@0.57.1:
    +    resolution: {integrity: sha512-C4cqCiGW0OSoSXsVQKgfLulYxY5C8M40f+a8VtBlAaEaN6eSlEt+catXb0chF9T2mvz/b87b0PahPvPwJdDf1Q==}
         dependencies:
    -      '@unocss/core': 0.56.5
    -      '@unocss/preset-mini': 0.56.5
    +      '@unocss/core': 0.57.1
    +      '@unocss/preset-mini': 0.57.1
         dev: true
     
    -  /@unocss/preset-uno@0.56.5:
    -    resolution: {integrity: sha512-3hzE0X1oxMbHLvWyTj/4BrJQ7OAL428BpzEJos0RsxifM04vOJX4GC4khIbmTl8KIMECMtATK3ren3JqzD2bFw==}
    +  /@unocss/preset-uno@0.57.1:
    +    resolution: {integrity: sha512-0+DKZiowYjYzq2swJzQA2dhqDvLJdm0Y437ITzc2GzZMKGUUuNi+w2v3/SzwkpkRd9zTB9/YaOIJVfdrx6ZOXQ==}
         dependencies:
    -      '@unocss/core': 0.56.5
    -      '@unocss/preset-mini': 0.56.5
    -      '@unocss/preset-wind': 0.56.5
    -      '@unocss/rule-utils': 0.56.5
    +      '@unocss/core': 0.57.1
    +      '@unocss/preset-mini': 0.57.1
    +      '@unocss/preset-wind': 0.57.1
    +      '@unocss/rule-utils': 0.57.1
         dev: true
     
    -  /@unocss/preset-web-fonts@0.56.5:
    -    resolution: {integrity: sha512-1YPbqpHK6NaXWHceNJTl2A+dNbliB6FeU5Tvkox1KOLTTwvzUF80uWHAE/l05Oc9EZyolZ8OsM37p2eJAb0wpw==}
    +  /@unocss/preset-web-fonts@0.57.1:
    +    resolution: {integrity: sha512-9DCIMlBRaGrljLmeciH4WqP+uRx2z2nLxvrvEmGbpJJpMn2H4higR5Zu5tDyKYGr9QBl9vXdWgib+43OSswkqA==}
         dependencies:
    -      '@unocss/core': 0.56.5
    +      '@unocss/core': 0.57.1
           ofetch: 1.3.3
         dev: true
     
    -  /@unocss/preset-wind@0.56.5:
    -    resolution: {integrity: sha512-iyMPvCEZkrGLHFXXlcqxDo/UcSK7KWw4x7/QUz7irrvc78cxYVuPm98QZgpCRcCwKerKVyFLjGOtwQ0kmVSVsQ==}
    +  /@unocss/preset-wind@0.57.1:
    +    resolution: {integrity: sha512-5UairNahUXNDe9AggPtTCodyPjl6NgPCsiEB22LVgN20UjBXjaqzN5wUe1OgtpLoAUaSk0KI7eLWhnWbTbST3A==}
         dependencies:
    -      '@unocss/core': 0.56.5
    -      '@unocss/preset-mini': 0.56.5
    -      '@unocss/rule-utils': 0.56.5
    +      '@unocss/core': 0.57.1
    +      '@unocss/preset-mini': 0.57.1
    +      '@unocss/rule-utils': 0.57.1
         dev: true
     
    -  /@unocss/reset@0.56.5:
    -    resolution: {integrity: sha512-//Pv2ITAdnpWB9FIr0JiQVl1rL1XgISR8mu3OikhCfi6d/4OH+o6/WyE7evF1pOmk1JjsQlznOABIZXQlcpbHQ==}
    +  /@unocss/reset@0.57.1:
    +    resolution: {integrity: sha512-f/ofoudjFN/HMtv1XV5phP58pOmNruBhr0GbVdBNylyieMQkFHowA7iSemChnC/fTbCcY6oSOAcFl4n9AefjdA==}
         dev: true
     
    -  /@unocss/rule-utils@0.56.5:
    -    resolution: {integrity: sha512-CXIGHCIC9B8WUl9KbbFMSZHcsIgfmI/+X0bjBv6xrgBVC1EQ2Acq4PYnJIbaRGBRAhl9wYjNL7Zq2UWOdowHAw==}
    +  /@unocss/rule-utils@0.57.1:
    +    resolution: {integrity: sha512-Hdicz7YORZx7SHICldzOGjPNeJwk/Xhy3cycqiPbg6nB6d639bpgZn5BsbDzHCPKpguwDomUqTZS6+C3s7tUVg==}
         engines: {node: '>=14'}
         dependencies:
    -      '@unocss/core': 0.56.5
    +      '@unocss/core': 0.57.1
    +      magic-string: 0.30.5
         dev: true
     
    -  /@unocss/scope@0.56.5:
    -    resolution: {integrity: sha512-q2eHYLuqF7RZEVGti205X2JOKSH6lBJGoQIjPtWgodNTg3S7qmUinHG0XzikI30L3EF3VljIh5TbtwGPcvbNsA==}
    +  /@unocss/scope@0.57.1:
    +    resolution: {integrity: sha512-ZAzg6lLGwKNQGCvJXEie3TvGztkAyajEFqygu0mjtHb+CmDql4iAjoygs+3dnRI5hSDwfMYFrJ2azX26+2CsoA==}
         dev: true
     
    -  /@unocss/transformer-attributify-jsx-babel@0.56.5:
    -    resolution: {integrity: sha512-QUlbmWpdfzdgEXNcOJwSvKDHB/ID/X3zlpK+fXyCKTkqiYgCwMFmLXll2HtNhIgH2E1gWo/lDlwpiW/QjQb/rw==}
    +  /@unocss/transformer-attributify-jsx-babel@0.57.1:
    +    resolution: {integrity: sha512-EOCPB8OGmhroAuFU0i0W5p6GmJpx6mAkP4KmsqVLd4QMgw+8aXkG7SKyLnxQZnekM0/dSo0TcpVGeGrZaUNgvQ==}
         dependencies:
    -      '@unocss/core': 0.56.5
    +      '@unocss/core': 0.57.1
         dev: true
     
    -  /@unocss/transformer-attributify-jsx@0.56.5:
    -    resolution: {integrity: sha512-SQW7t58s26qHYuD8v3covXtWoXlflw3FV5dlBCw/2iOfNo3OTVUWReAB5y0XQIT3828nHnPqeQq4IhmnuOENdw==}
    +  /@unocss/transformer-attributify-jsx@0.57.1:
    +    resolution: {integrity: sha512-ohgSEwm2j98ltPWl1zRPvZhRjQPpd7qZtgoROTQh6n2W7wEO1SlnYjgBBz+pGuo2dkfBN5NjuZJ93AEjS10Ysw==}
         dependencies:
    -      '@unocss/core': 0.56.5
    +      '@unocss/core': 0.57.1
         dev: true
     
    -  /@unocss/transformer-compile-class@0.56.5:
    -    resolution: {integrity: sha512-uT/+bVDN26hJ4a5FtbOT4PxAVDUTxwZCEVCesPAE5RnGd7f+KYUnk3XdDa77xyklPQsxPhr4MEjA0HGPHtSH8Q==}
    +  /@unocss/transformer-compile-class@0.57.1:
    +    resolution: {integrity: sha512-z0WZN6hbgpyBm2xqIrojqEjpQMhiyzHRbaBjWzI/6ieHWoFo5ajIwnReaFUEfJRNruLTd7/9hFDZdRXRPhttFw==}
         dependencies:
    -      '@unocss/core': 0.56.5
    +      '@unocss/core': 0.57.1
         dev: true
     
    -  /@unocss/transformer-directives@0.56.5:
    -    resolution: {integrity: sha512-ykXbvG1LbqTqlYImMtkJUeH8BtuiE+8T/txUVyooUoVgr2qSELUu2FxC6rTX2EUx+F5BZhaQd6GsdzgaDwXgIg==}
    +  /@unocss/transformer-directives@0.57.1:
    +    resolution: {integrity: sha512-rIk3XEU2NywEJUOkngBSmJfvS3IVgxkkqgMvuIqz8ZDbwWhepuMxsiI0QR3ypkipGr/eKK5DJ7eK0OVlo6FPFA==}
         dependencies:
    -      '@unocss/core': 0.56.5
    -      '@unocss/rule-utils': 0.56.5
    +      '@unocss/core': 0.57.1
    +      '@unocss/rule-utils': 0.57.1
           css-tree: 2.3.1
         dev: true
     
    -  /@unocss/transformer-variant-group@0.56.5:
    -    resolution: {integrity: sha512-e+7XtICuOtcOgGyxI06i0LK6R446KLFvzv+lw3WbwhD8OcsSFUAtCaAw0l+cyyiiZ/k2tLdUR0O4hYwQVa8f1A==}
    +  /@unocss/transformer-variant-group@0.57.1:
    +    resolution: {integrity: sha512-qwydzn2Lqz/8zW6UUXdORaUl8humsG8ll74LN/z8cjEsqtXZkVdkV0l6Brpp9Xp/XPbKwO+II+KH3/1LGwXSzQ==}
         dependencies:
    -      '@unocss/core': 0.56.5
    +      '@unocss/core': 0.57.1
         dev: true
     
    -  /@unocss/vite@0.56.5(rollup@2.79.1)(vite@4.3.9):
    -    resolution: {integrity: sha512-X4nvIukXTH//d+Oc97nJogK04sVGw4fc5LhVV1DjHQVmAOAmhTJCG6SxWGoSeqqUMx5X3gI9fVILK+5O8yl5EA==}
    +  /@unocss/vite@0.57.1(rollup@2.79.1)(vite@4.4.9):
    +    resolution: {integrity: sha512-kEBDvGgQNkX2n87S6Ao5seyFb1kuWZ5p96dGOS7VFpD7HvR5xholkJXaVhUK9/exCldjLExbo5UtVlbxFLUFYg==}
         peerDependencies:
           vite: ^2.9.0 || ^3.0.0-0 || ^4.0.0
         dependencies:
           '@ampproject/remapping': 2.2.1
           '@rollup/pluginutils': 5.0.5(rollup@2.79.1)
    -      '@unocss/config': 0.56.5
    -      '@unocss/core': 0.56.5
    -      '@unocss/inspector': 0.56.5
    -      '@unocss/scope': 0.56.5
    -      '@unocss/transformer-directives': 0.56.5
    +      '@unocss/config': 0.57.1
    +      '@unocss/core': 0.57.1
    +      '@unocss/inspector': 0.57.1
    +      '@unocss/scope': 0.57.1
    +      '@unocss/transformer-directives': 0.57.1
           chokidar: 3.5.3
           fast-glob: 3.3.1
           magic-string: 0.30.5
    -      vite: 4.3.9(@types/node@18.16.0)
    +      vite: 4.4.9(@types/node@18.17.5)
         transitivePeerDependencies:
           - rollup
         dev: true
    @@ -5305,18 +5218,40 @@ packages:
         peerDependencies:
           vite-plugin-pwa: '>=0.16.3 <1'
         dependencies:
    -      vite-plugin-pwa: 0.16.0(vite@4.3.9)(workbox-build@7.0.0)(workbox-window@7.0.0)
    +      vite-plugin-pwa: 0.16.0(vite@4.4.9)(workbox-build@7.0.0)(workbox-window@7.0.0)
         dev: true
     
    -  /@vitejs/plugin-vue@4.2.1(vite@4.3.9)(vue@3.3.0):
    +  /@vitejs/plugin-vue@4.2.1(vite@4.4.9)(vue@3.3.4):
         resolution: {integrity: sha512-ZTZjzo7bmxTRTkb8GSTwkPOYDIP7pwuyV+RV53c9PYUouwcbkIZIvWvNWlX2b1dYZqtOv7D6iUAnJLVNGcLrSw==}
         engines: {node: ^14.18.0 || >=16.0.0}
         peerDependencies:
           vite: ^4.0.0
           vue: ^3.2.25
         dependencies:
    -      vite: 4.3.9(@types/node@18.16.0)
    -      vue: 3.3.0
    +      vite: 4.4.9(@types/node@18.17.5)
    +      vue: 3.3.4
    +    dev: true
    +
    +  /@vitejs/plugin-vue@4.2.3(vite@4.4.9)(vue@3.3.4):
    +    resolution: {integrity: sha512-R6JDUfiZbJA9cMiguQ7jxALsgiprjBeHL5ikpXfJCH62pPHtI+JdJ5xWj6Ev73yXSlYl86+blXn1kZHQ7uElxw==}
    +    engines: {node: ^14.18.0 || >=16.0.0}
    +    peerDependencies:
    +      vite: ^4.0.0
    +      vue: ^3.2.25
    +    dependencies:
    +      vite: 4.4.9(@types/node@18.17.5)
    +      vue: 3.3.4
    +    dev: true
    +
    +  /@vitejs/plugin-vue@4.3.1(vite@4.5.0)(vue@3.3.7):
    +    resolution: {integrity: sha512-tUBEtWcF7wFtII7ayNiLNDTCE1X1afySEo+XNVMNkFXaThENyCowIEX095QqbJZGTgoOcSVDJGlnde2NG4jtbQ==}
    +    engines: {node: ^14.18.0 || >=16.0.0}
    +    peerDependencies:
    +      vite: ^4.0.0
    +      vue: ^3.2.25
    +    dependencies:
    +      vite: 4.5.0(@types/node@18.17.5)
    +      vue: 3.3.7(typescript@5.1.6)
         dev: true
     
       /@vitest/coverage-v8@0.34.0(vitest@0.34.0):
    @@ -5332,7 +5267,7 @@ packages:
           istanbul-reports: 3.1.6
           magic-string: 0.30.2
           picocolors: 1.0.0
    -      std-env: 3.4.0
    +      std-env: 3.3.3
           test-exclude: 6.0.0
           v8-to-istanbul: 9.1.0
           vitest: 0.34.0(@vitest/ui@0.34.0)(jsdom@22.0.0)
    @@ -5359,7 +5294,7 @@ packages:
       /@vitest/snapshot@0.34.0:
         resolution: {integrity: sha512-eGN5XBZHYOghxCOQbf8dcn6/3g7IW77GOOOC/mNFYwRXsPeoQgcgWnhj+6wgJ04pVv25wpxWL9jUkzaQ7LoFtg==}
         dependencies:
    -      magic-string: 0.30.2
    +      magic-string: 0.30.5
           pathe: 1.1.1
           pretty-format: 29.6.2
         dev: true
    @@ -5398,20 +5333,12 @@ packages:
         peerDependencies:
           vue: 3.3.4
         dependencies:
    -      '@babel/parser': 7.22.10
    +      '@babel/parser': 7.23.0
           estree-walker: 2.0.2
           source-map-js: 1.0.2
           vue: 3.3.4
         dev: false
     
    -  /@vue/compiler-core@3.3.0:
    -    resolution: {integrity: sha512-iYvUFe9/tIXNI1FyDCQYhkwJI5M9htqeCGfdZ2LiR+ZqVQE6KAH2+qUPdXixjMPUL36LdpVIBTNhxstx5RRhEw==}
    -    dependencies:
    -      '@babel/parser': 7.22.10
    -      '@vue/shared': 3.3.0
    -      estree-walker: 2.0.2
    -      source-map-js: 1.0.2
    -
       /@vue/compiler-core@3.3.4:
         resolution: {integrity: sha512-cquyDNvZ6jTbf/+x+AgM2Arrp6G4Dzbb0R64jiG804HRMfRiFXWI6kqUVqZ6ZR0bQhIoQjB4+2bhNtVwndW15g==}
         dependencies:
    @@ -5420,11 +5347,14 @@ packages:
           estree-walker: 2.0.2
           source-map-js: 1.0.2
     
    -  /@vue/compiler-dom@3.3.0:
    -    resolution: {integrity: sha512-oxWgWpY+2FOQMZxdXgVaslu7z/KSmk9pO90MrYdxfiOW3/0HkqR6nuDjukiwaz5rN/kUioNXBfAkDcNwIr1JOA==}
    +  /@vue/compiler-core@3.3.7:
    +    resolution: {integrity: sha512-pACdY6YnTNVLXsB86YD8OF9ihwpolzhhtdLVHhBL6do/ykr6kKXNYABRtNMGrsQXpEXXyAdwvWWkuTbs4MFtPQ==}
         dependencies:
    -      '@vue/compiler-core': 3.3.0
    -      '@vue/shared': 3.3.0
    +      '@babel/parser': 7.23.0
    +      '@vue/shared': 3.3.7
    +      estree-walker: 2.0.2
    +      source-map-js: 1.0.2
    +    dev: true
     
       /@vue/compiler-dom@3.3.4:
         resolution: {integrity: sha512-wyM+OjOVpuUukIq6p5+nwHYtj9cFroz9cwkfmP9O1nzH68BenTTv0u7/ndggT8cIQlnBeOo6sUT/gvHcIkLA5w==}
    @@ -5432,19 +5362,12 @@ packages:
           '@vue/compiler-core': 3.3.4
           '@vue/shared': 3.3.4
     
    -  /@vue/compiler-sfc@3.3.0:
    -    resolution: {integrity: sha512-g8j35REOBMN0oRnJ4eEO3RMLj8ebEehQk6JkH6Q9df+M1Sb8eLeX0Zb7GBBPrrjfmyKzGvp/TE3fyOLUq/H5ow==}
    +  /@vue/compiler-dom@3.3.7:
    +    resolution: {integrity: sha512-0LwkyJjnUPssXv/d1vNJ0PKfBlDoQs7n81CbO6Q0zdL7H1EzqYRrTVXDqdBVqro0aJjo/FOa1qBAPVI4PGSHBw==}
         dependencies:
    -      '@babel/parser': 7.22.10
    -      '@vue/compiler-core': 3.3.0
    -      '@vue/compiler-dom': 3.3.0
    -      '@vue/compiler-ssr': 3.3.0
    -      '@vue/reactivity-transform': 3.3.0
    -      '@vue/shared': 3.3.0
    -      estree-walker: 2.0.2
    -      magic-string: 0.30.2
    -      postcss: 8.4.28
    -      source-map-js: 1.0.2
    +      '@vue/compiler-core': 3.3.7
    +      '@vue/shared': 3.3.7
    +    dev: true
     
       /@vue/compiler-sfc@3.3.4:
         resolution: {integrity: sha512-6y/d8uw+5TkCuzBkgLS0v3lSM3hJDntFEiUORM11pQ/hKvkhSKZrXW6i69UyXlJQisJxuUEJKAWEqWbWsLeNKQ==}
    @@ -5457,14 +5380,23 @@ packages:
           '@vue/shared': 3.3.4
           estree-walker: 2.0.2
           magic-string: 0.30.2
    -      postcss: 8.4.28
    +      postcss: 8.4.27
           source-map-js: 1.0.2
     
    -  /@vue/compiler-ssr@3.3.0:
    -    resolution: {integrity: sha512-G39cqKLtSvlHM4L+P7vav9mh+ruks156VsXtzKya/FLMAWkSco6ye4SdaD6vJHMbtCypTOkMU7R6NMrCr19vpg==}
    +  /@vue/compiler-sfc@3.3.7:
    +    resolution: {integrity: sha512-7pfldWy/J75U/ZyYIXRVqvLRw3vmfxDo2YLMwVtWVNew8Sm8d6wodM+OYFq4ll/UxfqVr0XKiVwti32PCrruAw==}
         dependencies:
    -      '@vue/compiler-dom': 3.3.0
    -      '@vue/shared': 3.3.0
    +      '@babel/parser': 7.23.0
    +      '@vue/compiler-core': 3.3.7
    +      '@vue/compiler-dom': 3.3.7
    +      '@vue/compiler-ssr': 3.3.7
    +      '@vue/reactivity-transform': 3.3.7
    +      '@vue/shared': 3.3.7
    +      estree-walker: 2.0.2
    +      magic-string: 0.30.5
    +      postcss: 8.4.31
    +      source-map-js: 1.0.2
    +    dev: true
     
       /@vue/compiler-ssr@3.3.4:
         resolution: {integrity: sha512-m0v6oKpup2nMSehwA6Uuu+j+wEwcy7QmwMkVNVfrV9P2qE5KshC6RwOCq8fjGS/Eak/uNb8AaWekfiXxbBB6gQ==}
    @@ -5472,42 +5404,49 @@ packages:
           '@vue/compiler-dom': 3.3.4
           '@vue/shared': 3.3.4
     
    +  /@vue/compiler-ssr@3.3.7:
    +    resolution: {integrity: sha512-TxOfNVVeH3zgBc82kcUv+emNHo+vKnlRrkv8YvQU5+Y5LJGJwSNzcmLUoxD/dNzv0bhQ/F0s+InlgV0NrApJZg==}
    +    dependencies:
    +      '@vue/compiler-dom': 3.3.7
    +      '@vue/shared': 3.3.7
    +    dev: true
    +
       /@vue/devtools-api@6.5.0:
         resolution: {integrity: sha512-o9KfBeaBmCKl10usN4crU53fYtC1r7jJwdGKjPT24t348rHxgfpZ0xL3Xm/gLUYnc0oTp8LAmrxOeLyu6tbk2Q==}
    +    dev: true
     
    -  /@vue/reactivity-transform@3.3.0:
    -    resolution: {integrity: sha512-Pli2ClOXOEMG2AExCfUwiPQQo7U7zcRlnZLb6FI9ns/nEiQ9KLJJYD3wAuJHSx0VXLhACaINd/1VbMeKfa8GhQ==}
    -    dependencies:
    -      '@babel/parser': 7.22.10
    -      '@vue/compiler-core': 3.3.0
    -      '@vue/shared': 3.3.0
    -      estree-walker: 2.0.2
    -      magic-string: 0.30.2
    +  /@vue/devtools-api@6.5.1:
    +    resolution: {integrity: sha512-+KpckaAQyfbvshdDW5xQylLni1asvNSGme1JFs8I1+/H5pHEhqUKMEQD/qn3Nx5+/nycBq11qAEi8lk+LXI2dA==}
     
       /@vue/reactivity-transform@3.3.4:
         resolution: {integrity: sha512-MXgwjako4nu5WFLAjpBnCj/ieqcjE2aJBINUNQzkZQfzIZA4xn+0fV1tIYBJvvva3N3OvKGofRLvQIwEQPpaXw==}
         dependencies:
    -      '@babel/parser': 7.22.10
    +      '@babel/parser': 7.23.0
           '@vue/compiler-core': 3.3.4
           '@vue/shared': 3.3.4
           estree-walker: 2.0.2
    -      magic-string: 0.30.2
    +      magic-string: 0.30.5
     
    -  /@vue/reactivity@3.3.0:
    -    resolution: {integrity: sha512-CyVK/UDaGVK9ARd6HDh+RnvSY65rItjkNvxz7yTcbsGole6KaywdLIzLhWeaO5y7LnYE4MNGxzwZxmsnd+gNmQ==}
    +  /@vue/reactivity-transform@3.3.7:
    +    resolution: {integrity: sha512-APhRmLVbgE1VPGtoLQoWBJEaQk4V8JUsqrQihImVqKT+8U6Qi3t5ATcg4Y9wGAPb3kIhetpufyZ1RhwbZCIdDA==}
         dependencies:
    -      '@vue/shared': 3.3.0
    +      '@babel/parser': 7.23.0
    +      '@vue/compiler-core': 3.3.7
    +      '@vue/shared': 3.3.7
    +      estree-walker: 2.0.2
    +      magic-string: 0.30.5
    +    dev: true
     
       /@vue/reactivity@3.3.4:
         resolution: {integrity: sha512-kLTDLwd0B1jG08NBF3R5rqULtv/f8x3rOFByTDz4J53ttIQEDmALqKqXY0J+XQeN0aV2FBxY8nJDf88yvOPAqQ==}
         dependencies:
           '@vue/shared': 3.3.4
     
    -  /@vue/runtime-core@3.3.0:
    -    resolution: {integrity: sha512-PJ6EYidRqsG0p0kijogSjA9dmJk6AhGGX387UWjbk2Y1z7t9VI0vTMLwBXf7H7QkKSAufiPRMET7qmexcOae1g==}
    +  /@vue/reactivity@3.3.7:
    +    resolution: {integrity: sha512-cZNVjWiw00708WqT0zRpyAgduG79dScKEPYJXq2xj/aMtk3SKvL3FBt2QKUlh6EHBJ1m8RhBY+ikBUzwc7/khg==}
         dependencies:
    -      '@vue/reactivity': 3.3.0
    -      '@vue/shared': 3.3.0
    +      '@vue/shared': 3.3.7
    +    dev: true
     
       /@vue/runtime-core@3.3.4:
         resolution: {integrity: sha512-R+bqxMN6pWO7zGI4OMlmvePOdP2c93GsHFM/siJI7O2nxFRzj55pLwkpCedEY+bTMgp5miZ8CxfIZo3S+gFqvA==}
    @@ -5515,12 +5454,12 @@ packages:
           '@vue/reactivity': 3.3.4
           '@vue/shared': 3.3.4
     
    -  /@vue/runtime-dom@3.3.0:
    -    resolution: {integrity: sha512-e2VwfvU6xk/BdXpFvh1UXo4mcOrKCAkPrCy/vFas9GkkYzW3nx3uJ7Jm2Zl08dRoCMP7Oy9FegT9JkJ5kU8C+g==}
    +  /@vue/runtime-core@3.3.7:
    +    resolution: {integrity: sha512-LHq9du3ubLZFdK/BP0Ysy3zhHqRfBn80Uc+T5Hz3maFJBGhci1MafccnL3rpd5/3wVfRHAe6c+PnlO2PAavPTQ==}
         dependencies:
    -      '@vue/runtime-core': 3.3.0
    -      '@vue/shared': 3.3.0
    -      csstype: 3.1.2
    +      '@vue/reactivity': 3.3.7
    +      '@vue/shared': 3.3.7
    +    dev: true
     
       /@vue/runtime-dom@3.3.4:
         resolution: {integrity: sha512-Aj5bTJ3u5sFsUckRghsNjVTtxZQ1OyMWCr5dZRAPijF/0Vy4xEoRCwLyHXcj4D0UFbJ4lbx3gPTgg06K/GnPnQ==}
    @@ -5529,14 +5468,13 @@ packages:
           '@vue/shared': 3.3.4
           csstype: 3.1.2
     
    -  /@vue/server-renderer@3.3.0(vue@3.3.0):
    -    resolution: {integrity: sha512-U8coTPJMym4U6kJ2sDQuO5BmYjfIn26f66rtCk+cS1hoSxOtxFtUJuFXAOTIHvFWeelk4qeh9Ub5ZbfVRCHQBg==}
    -    peerDependencies:
    -      vue: 3.3.0
    +  /@vue/runtime-dom@3.3.7:
    +    resolution: {integrity: sha512-PFQU1oeJxikdDmrfoNQay5nD4tcPNYixUBruZzVX/l0eyZvFKElZUjW4KctCcs52nnpMGO6UDK+jF5oV4GT5Lw==}
         dependencies:
    -      '@vue/compiler-ssr': 3.3.0
    -      '@vue/shared': 3.3.0
    -      vue: 3.3.0
    +      '@vue/runtime-core': 3.3.7
    +      '@vue/shared': 3.3.7
    +      csstype: 3.1.2
    +    dev: true
     
       /@vue/server-renderer@3.3.4(vue@3.3.4):
         resolution: {integrity: sha512-Q6jDDzR23ViIb67v+vM1Dqntu+HUexQcsWKhhQa4ARVzxOY2HbC7QRW/ggkDBd5BU+uM1sV6XOAP0b216o34JQ==}
    @@ -5547,38 +5485,61 @@ packages:
           '@vue/shared': 3.3.4
           vue: 3.3.4
     
    -  /@vue/shared@3.3.0:
    -    resolution: {integrity: sha512-U4LUNs+xkcncuiWSyYlZJPl4l8zAKs67OuLM2L91QsaYZAEylj41pGHaLPHkO0ULGTpxTMETEBXkn6QFP9/X+Q==}
    +  /@vue/server-renderer@3.3.7(vue@3.3.7):
    +    resolution: {integrity: sha512-UlpKDInd1hIZiNuVVVvLgxpfnSouxKQOSE2bOfQpBuGwxRV/JqqTCyyjXUWiwtVMyeRaZhOYYqntxElk8FhBhw==}
    +    peerDependencies:
    +      vue: 3.3.7
    +    dependencies:
    +      '@vue/compiler-ssr': 3.3.7
    +      '@vue/shared': 3.3.7
    +      vue: 3.3.7(typescript@5.0.4)
    +    dev: true
     
       /@vue/shared@3.3.4:
         resolution: {integrity: sha512-7OjdcV8vQ74eiz1TZLzZP4JwqM5fA94K6yntPS5Z25r9HDuGNzaGdgvwKYq6S+MxwF0TFRwe50fIR/MYnakdkQ==}
     
    -  /@vueuse/core@10.1.0(vue@3.3.0):
    +  /@vue/shared@3.3.7:
    +    resolution: {integrity: sha512-N/tbkINRUDExgcPTBvxNkvHGu504k8lzlNQRITVnm6YjOjwa4r0nnbd4Jb01sNpur5hAllyRJzSK5PvB9PPwRg==}
    +    dev: true
    +
    +  /@vueuse/core@10.1.0(vue@3.3.4):
         resolution: {integrity: sha512-3Znoa5m5RO+z4/C9w6DRaKTR3wCVJvD5rav8HTDGsr+7rOZRHtcgFJ8NcCs0ZvIpmev2kExTa311ns5j2RbzDQ==}
         dependencies:
           '@types/web-bluetooth': 0.0.16
           '@vueuse/metadata': 10.1.0
    -      '@vueuse/shared': 10.1.0(vue@3.3.0)
    -      vue-demi: 0.14.5(vue@3.3.0)
    +      '@vueuse/shared': 10.1.0(vue@3.3.4)
    +      vue-demi: 0.14.5(vue@3.3.4)
         transitivePeerDependencies:
           - '@vue/composition-api'
           - vue
         dev: false
     
    -  /@vueuse/core@10.4.1(vue@3.3.4):
    -    resolution: {integrity: sha512-DkHIfMIoSIBjMgRRvdIvxsyboRZQmImofLyOHADqiVbQVilP8VVHDhBX2ZqoItOgu7dWa8oXiNnScOdPLhdEXg==}
    +  /@vueuse/core@10.3.0(vue@3.3.4):
    +    resolution: {integrity: sha512-BEM5yxcFKb5btFjTSAFjTu5jmwoW66fyV9uJIP4wUXXU8aR5Hl44gndaaXp7dC5HSObmgbnR2RN+Un1p68Mf5Q==}
         dependencies:
           '@types/web-bluetooth': 0.0.17
    -      '@vueuse/metadata': 10.4.1
    -      '@vueuse/shared': 10.4.1(vue@3.3.4)
    -      vue-demi: 0.14.5(vue@3.3.4)
    +      '@vueuse/metadata': 10.3.0
    +      '@vueuse/shared': 10.3.0(vue@3.3.4)
    +      vue-demi: 0.14.6(vue@3.3.4)
         transitivePeerDependencies:
           - '@vue/composition-api'
           - vue
         dev: true
     
    -  /@vueuse/integrations@10.4.1(focus-trap@7.5.2)(vue@3.3.4):
    -    resolution: {integrity: sha512-uRBPyG5Lxoh1A/J+boiioPT3ELEAPEo4t8W6Mr4yTKIQBeW/FcbsotZNPr4k9uz+3QEksMmflWloS9wCnypM7g==}
    +  /@vueuse/core@10.5.0(vue@3.3.7):
    +    resolution: {integrity: sha512-z/tI2eSvxwLRjOhDm0h/SXAjNm8N5ld6/SC/JQs6o6kpJ6Ya50LnEL8g5hoYu005i28L0zqB5L5yAl8Jl26K3A==}
    +    dependencies:
    +      '@types/web-bluetooth': 0.0.18
    +      '@vueuse/metadata': 10.5.0
    +      '@vueuse/shared': 10.5.0(vue@3.3.7)
    +      vue-demi: 0.14.6(vue@3.3.7)
    +    transitivePeerDependencies:
    +      - '@vue/composition-api'
    +      - vue
    +    dev: true
    +
    +  /@vueuse/integrations@10.5.0(focus-trap@7.5.4)(vue@3.3.7):
    +    resolution: {integrity: sha512-fm5sXLCK0Ww3rRnzqnCQRmfjDURaI4xMsx+T+cec0ngQqHx/JgUtm8G0vRjwtonIeTBsH1Q8L3SucE+7K7upJQ==}
         peerDependencies:
           async-validator: '*'
           axios: '*'
    @@ -5618,10 +5579,10 @@ packages:
           universal-cookie:
             optional: true
         dependencies:
    -      '@vueuse/core': 10.4.1(vue@3.3.4)
    -      '@vueuse/shared': 10.4.1(vue@3.3.4)
    -      focus-trap: 7.5.2
    -      vue-demi: 0.14.5(vue@3.3.4)
    +      '@vueuse/core': 10.5.0(vue@3.3.7)
    +      '@vueuse/shared': 10.5.0(vue@3.3.7)
    +      focus-trap: 7.5.4
    +      vue-demi: 0.14.6(vue@3.3.7)
         transitivePeerDependencies:
           - '@vue/composition-api'
           - vue
    @@ -5631,35 +5592,48 @@ packages:
         resolution: {integrity: sha512-cM28HjDEw5FIrPE9rgSPFZvQ0ZYnOLAOr8hl1XM6tFl80U3WAR5ROdnAqiYybniwP5gt9MKKAJAqd/ab2aHkqg==}
         dev: false
     
    -  /@vueuse/metadata@10.4.1:
    -    resolution: {integrity: sha512-2Sc8X+iVzeuMGHr6O2j4gv/zxvQGGOYETYXEc41h0iZXIRnRbJZGmY/QP8dvzqUelf8vg0p/yEA5VpCEu+WpZg==}
    +  /@vueuse/metadata@10.3.0:
    +    resolution: {integrity: sha512-Ema3YhNOa4swDsV0V7CEY5JXvK19JI/o1szFO1iWxdFg3vhdFtCtSTP26PCvbUpnUtNHBY2wx5y3WDXND5Pvnw==}
    +    dev: true
    +
    +  /@vueuse/metadata@10.5.0:
    +    resolution: {integrity: sha512-fEbElR+MaIYyCkeM0SzWkdoMtOpIwO72x8WsZHRE7IggiOlILttqttM69AS13nrDxosnDBYdyy3C5mR1LCxHsw==}
         dev: true
     
    -  /@vueuse/shared@10.1.0(vue@3.3.0):
    +  /@vueuse/shared@10.1.0(vue@3.3.4):
         resolution: {integrity: sha512-2X52ogu12i9DkKOQ01yeb/BKg9UO87RNnpm5sXkQvyORlbq8ONS5l39MYkjkeVWWjdT0teJru7a2S41dmHmqjQ==}
         dependencies:
    -      vue-demi: 0.14.5(vue@3.3.0)
    +      vue-demi: 0.14.5(vue@3.3.4)
         transitivePeerDependencies:
           - '@vue/composition-api'
           - vue
         dev: false
     
    -  /@vueuse/shared@10.4.1(vue@3.3.4):
    -    resolution: {integrity: sha512-vz5hbAM4qA0lDKmcr2y3pPdU+2EVw/yzfRsBdu+6+USGa4PxqSQRYIUC9/NcT06y+ZgaTsyURw2I9qOFaaXHAg==}
    +  /@vueuse/shared@10.3.0(vue@3.3.4):
    +    resolution: {integrity: sha512-kGqCTEuFPMK4+fNWy6dUOiYmxGcUbtznMwBZLC1PubidF4VZY05B+Oht7Jh7/6x4VOWGpvu3R37WHi81cKpiqg==}
         dependencies:
    -      vue-demi: 0.14.5(vue@3.3.4)
    +      vue-demi: 0.14.6(vue@3.3.4)
    +    transitivePeerDependencies:
    +      - '@vue/composition-api'
    +      - vue
    +    dev: true
    +
    +  /@vueuse/shared@10.5.0(vue@3.3.7):
    +    resolution: {integrity: sha512-18iyxbbHYLst9MqU1X1QNdMHIjks6wC7XTVf0KNOv5es/Ms6gjVFCAAWTVP2JStuGqydg3DT+ExpFORUEi9yhg==}
    +    dependencies:
    +      vue-demi: 0.14.6(vue@3.3.7)
         transitivePeerDependencies:
           - '@vue/composition-api'
           - vue
         dev: true
     
    -  /@wdio/config@7.30.0(typescript@5.1.3):
    +  /@wdio/config@7.30.0(typescript@5.1.6):
         resolution: {integrity: sha512-/38rol9WCfFTMtXyd/C856/aexxIZnfVvXg7Fw2WXpqZ9qadLA+R4N35S2703n/RByjK/5XAYtHoljtvh3727w==}
         engines: {node: '>=12.0.0'}
         dependencies:
           '@wdio/logger': 7.26.0
    -      '@wdio/types': 7.26.0(typescript@5.1.3)
    -      '@wdio/utils': 7.26.0(typescript@5.1.3)
    +      '@wdio/types': 7.26.0(typescript@5.1.6)
    +      '@wdio/utils': 7.26.0(typescript@5.1.6)
           deepmerge: 4.3.1
           glob: 8.1.0
         transitivePeerDependencies:
    @@ -5681,7 +5655,7 @@ packages:
         engines: {node: '>=12.0.0'}
         dev: true
     
    -  /@wdio/types@7.26.0(typescript@5.1.3):
    +  /@wdio/types@7.26.0(typescript@5.1.6):
         resolution: {integrity: sha512-mOTfWAGQ+iT58iaZhJMwlUkdEn3XEWE4jthysMLXFnSuZ2eaODVAiK31SmlS/eUqgSIaupeGqYUrtCuSNbLefg==}
         engines: {node: '>=12.0.0'}
         peerDependencies:
    @@ -5690,17 +5664,17 @@ packages:
           typescript:
             optional: true
         dependencies:
    -      '@types/node': 18.17.6
    +      '@types/node': 18.17.5
           got: 11.8.6
    -      typescript: 5.1.3
    +      typescript: 5.1.6
         dev: true
     
    -  /@wdio/utils@7.26.0(typescript@5.1.3):
    +  /@wdio/utils@7.26.0(typescript@5.1.6):
         resolution: {integrity: sha512-pVq2MPXZAYLkKGKIIHktHejnHqg4TYKoNYSi2EDv+I3GlT8VZKXHazKhci82ov0tD+GdF27+s4DWNDCfGYfBdQ==}
         engines: {node: '>=12.0.0'}
         dependencies:
           '@wdio/logger': 7.26.0
    -      '@wdio/types': 7.26.0(typescript@5.1.3)
    +      '@wdio/types': 7.26.0(typescript@5.1.6)
           p-iteration: 1.1.8
         transitivePeerDependencies:
           - typescript
    @@ -5873,7 +5847,7 @@ packages:
           lodash: 4.17.21
           marked: 4.3.0
           pino: 8.15.0
    -      postcss: 8.4.28
    +      postcss: 8.4.27
           ramda: 0.28.0
           tailwindcss: 3.3.3(ts-node@10.9.1)
           vue: 3.3.4
    @@ -6026,6 +6000,15 @@ packages:
           uri-js: 4.4.1
         dev: true
     
    +  /ajv@8.11.2:
    +    resolution: {integrity: sha512-E4bfmKAhGiSTvMfL1Myyycaub+cUEU2/IvpylXkUu7CHBkBj1f/ikdzbD7YQ6FKUbixDxeYvB/xY4fvyroDlQg==}
    +    dependencies:
    +      fast-deep-equal: 3.1.3
    +      json-schema-traverse: 1.0.0
    +      require-from-string: 2.0.2
    +      uri-js: 4.4.1
    +    dev: true
    +
       /ajv@8.12.0:
         resolution: {integrity: sha512-sRu1kpcO9yLtYxBKvqfTeh9KzZEwO3STyX1HT+4CaDzC6HpTGYhIhPIzj9XuKU7KYDwnaeh5hcOwjy1QuJzBPA==}
         dependencies:
    @@ -6364,7 +6347,7 @@ packages:
         engines: {node: ^14.15.0 || ^16.10.0 || >=18.0.0}
         dependencies:
           '@babel/template': 7.22.5
    -      '@babel/types': 7.22.10
    +      '@babel/types': 7.23.0
           '@types/babel__core': 7.20.1
           '@types/babel__traverse': 7.20.1
         dev: true
    @@ -6389,7 +6372,7 @@ packages:
         dependencies:
           '@babel/core': 7.22.10
           '@babel/helper-define-polyfill-provider': 0.4.2(@babel/core@7.22.10)
    -      core-js-compat: 3.32.1
    +      core-js-compat: 3.32.0
         transitivePeerDependencies:
           - supports-color
         dev: true
    @@ -6516,6 +6499,10 @@ packages:
           - supports-color
         dev: true
     
    +  /body-scroll-lock@4.0.0-beta.0:
    +    resolution: {integrity: sha512-a7tP5+0Mw3YlUJcGAKUqIBkYYGlYxk2fnCasq/FUph1hadxlTRjF+gAcZksxANnaMnALjxEddmSi/H3OR8ugcQ==}
    +    dev: true
    +
       /bonjour-service@1.1.1:
         resolution: {integrity: sha512-Z/5lQRMOG9k7W+FkeGTNjh7htqn/2LMnfOvBZ8pynNZCM9MwkQkI3zeI4oz09uWdcgmgHugVvBqxGg4VQJ5PCg==}
         dependencies:
    @@ -6573,8 +6560,8 @@ packages:
         engines: {node: ^6 || ^7 || ^8 || ^9 || ^10 || ^11 || ^12 || >=13.7}
         hasBin: true
         dependencies:
    -      caniuse-lite: 1.0.30001521
    -      electron-to-chromium: 1.4.496
    +      caniuse-lite: 1.0.30001520
    +      electron-to-chromium: 1.4.490
           node-releases: 2.0.13
           update-browserslist-db: 1.0.11(browserslist@4.21.10)
         dev: true
    @@ -6722,8 +6709,8 @@ packages:
         engines: {node: '>=10'}
         dev: true
     
    -  /caniuse-lite@1.0.30001521:
    -    resolution: {integrity: sha512-fnx1grfpEOvDGH+V17eccmNjucGUnCbP6KL+l5KqBIerp26WK/+RQ7CIDE37KGJjaPyqWXXlFUyKiWmvdNNKmQ==}
    +  /caniuse-lite@1.0.30001520:
    +    resolution: {integrity: sha512-tahF5O9EiiTzwTUqAeFjIZbn4Dnqxzz7ktrgGlMYNLH43Ul26IgTMH/zvL3DG0lZxBYnlT04axvInszUsZULdA==}
         dev: true
     
       /caseless@0.12.0:
    @@ -6832,8 +6819,8 @@ packages:
         engines: {node: '>= 0.8.0'}
         dev: true
     
    -  /chevrotain-allstar@0.3.0(chevrotain@11.0.3):
    -    resolution: {integrity: sha512-ZlkKZb97Jga/2619A5KET+CmvkhvoLlGVpJ7IXd5pHTZweGhr+apCuKHUvlwFeIVxk3A8o9DMjf+DXJKQxUQ/A==}
    +  /chevrotain-allstar@0.3.1(chevrotain@11.0.3):
    +    resolution: {integrity: sha512-b7g+y9A0v4mxCW1qUhf3BSVPg+/NvGErk/dOkrDaHA0nQIQGAtrOjlX//9OQtRlSCy+x9rfB5N8yC71lH1nvMw==}
         peerDependencies:
           chevrotain: ^11.0.0
         dependencies:
    @@ -7095,11 +7082,6 @@ packages:
           repeat-string: 1.6.1
         dev: true
     
    -  /comment-parser@1.3.1:
    -    resolution: {integrity: sha512-B52sN2VNghyq5ofvUsqZjmk6YkihBX5vMSChmSK9v4ShjKf3Vk5Xcmgpw4o+iIgtrnM/u5FiMpz9VKb8lpBveA==}
    -    engines: {node: '>= 12.0.0'}
    -    dev: true
    -
       /comment-parser@1.4.0:
         resolution: {integrity: sha512-QLyTNiZ2KDOibvFPlZ6ZngVsZ/0gYnE6uTXi5aoDg8ed3AkJAz4sEje3Y8a29hQ1s6A99MZXe47fLAXQ1rTqaw==}
         engines: {node: '>= 12.0.0'}
    @@ -7150,6 +7132,22 @@ packages:
       /concat-map@0.0.1:
         resolution: {integrity: sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg==}
     
    +  /concurrently@8.0.0:
    +    resolution: {integrity: sha512-1fjagjL+RgPRAx9Wi8Yv866Whtx34MRdk9qf6wwxpQoYL2mD+lUZMOe9RXYULC6eBl6e4sde6cu8bpyg9Rd9/w==}
    +    engines: {node: ^14.13.0 || >=16.0.0}
    +    hasBin: true
    +    dependencies:
    +      chalk: 4.1.2
    +      date-fns: 2.30.0
    +      lodash: 4.17.21
    +      rxjs: 7.8.1
    +      shell-quote: 1.8.1
    +      spawn-command: 0.0.2-1
    +      supports-color: 8.1.1
    +      tree-kill: 1.2.2
    +      yargs: 17.7.2
    +    dev: true
    +
       /concurrently@8.0.1:
         resolution: {integrity: sha512-Sh8bGQMEL0TAmAm2meAXMjcASHZa7V0xXQVDBLknCPa9TPtkY9yYs+0cnGGgfdkW0SV1Mlg+hVGfXcoI8d3MJA==}
         engines: {node: ^14.13.0 || >=16.0.0}
    @@ -7207,13 +7205,11 @@ packages:
           compare-func: 2.0.0
         dev: true
     
    -  /conventional-changelog-conventionalcommits@5.0.0:
    -    resolution: {integrity: sha512-lCDbA+ZqVFQGUj7h9QBKoIpLhl8iihkO0nCTyRNzuXtcd7ubODpYB04IFy31JloiJgG0Uovu8ot8oxRzn7Nwtw==}
    -    engines: {node: '>=10'}
    +  /conventional-changelog-conventionalcommits@6.1.0:
    +    resolution: {integrity: sha512-3cS3GEtR78zTfMzk0AizXKKIdN4OvSh7ibNz6/DPbhWWQu7LqE/8+/GqSodV+sywUR2gpJAdP/1JFf4XtN7Zpw==}
    +    engines: {node: '>=14'}
         dependencies:
           compare-func: 2.0.0
    -      lodash: 4.17.21
    -      q: 1.5.1
         dev: true
     
       /conventional-commits-parser@4.0.0:
    @@ -7244,8 +7240,8 @@ packages:
         engines: {node: '>= 0.6'}
         dev: true
     
    -  /core-js-compat@3.32.1:
    -    resolution: {integrity: sha512-GSvKDv4wE0bPnQtjklV101juQ85g6H3rm5PDP20mqlS5j0kXF3pP97YvAu5hl+uFHqMictp3b2VxOHljWMAtuA==}
    +  /core-js-compat@3.32.0:
    +    resolution: {integrity: sha512-7a9a3D1k4UCVKnLhrgALyFcP7YCsLOQIxPd0dKjf/6GuPcgyiGP70ewWdCGrSK7evyhymi0qO4EqCmSJofDeYw==}
         dependencies:
           browserslist: 4.21.10
         dev: true
    @@ -7278,7 +7274,7 @@ packages:
           layout-base: 2.0.1
         dev: false
     
    -  /cosmiconfig-typescript-loader@4.4.0(@types/node@20.4.7)(cosmiconfig@8.2.0)(ts-node@10.9.1)(typescript@5.1.3):
    +  /cosmiconfig-typescript-loader@4.4.0(@types/node@20.4.7)(cosmiconfig@8.2.0)(ts-node@10.9.1)(typescript@5.1.6):
         resolution: {integrity: sha512-BabizFdC3wBHhbI4kJh0VkQP9GkBfoHPydD0COMce1nJ1kJAB3F2TmJ/I7diULBKtmEWSwEbuN/KDtgnmUUVmw==}
         engines: {node: '>=v14.21.3'}
         peerDependencies:
    @@ -7289,8 +7285,8 @@ packages:
         dependencies:
           '@types/node': 20.4.7
           cosmiconfig: 8.2.0
    -      ts-node: 10.9.1(@types/node@20.4.7)(typescript@5.1.3)
    -      typescript: 5.1.3
    +      ts-node: 10.9.1(@types/node@20.4.7)(typescript@5.1.6)
    +      typescript: 5.1.6
         dev: true
     
       /cosmiconfig@8.0.0:
    @@ -7338,7 +7334,7 @@ packages:
         dependencies:
           arrify: 3.0.0
           cp-file: 9.1.0
    -      globby: 13.1.4
    +      globby: 13.2.2
           junk: 4.0.1
           micromatch: 4.0.5
           nested-error-stacks: 2.1.1
    @@ -7385,6 +7381,17 @@ packages:
           gensequence: 5.0.2
         dev: true
     
    +  /cspell-dictionary@6.31.3:
    +    resolution: {integrity: sha512-3w5P3Md/tbHLVGPKVL0ePl1ObmNwhdDiEuZ2TXfm2oAIwg4aqeIrw42A2qmhaKLcuAIywpqGZsrGg8TviNNhig==}
    +    engines: {node: '>=14'}
    +    dependencies:
    +      '@cspell/cspell-pipe': 6.31.3
    +      '@cspell/cspell-types': 6.31.3
    +      cspell-trie-lib: 6.31.3
    +      fast-equals: 4.0.3
    +      gensequence: 5.0.2
    +    dev: true
    +
       /cspell-gitignore@6.31.1:
         resolution: {integrity: sha512-PAcmjN6X89Z8qgjem6HYb+VmvVtKuc+fWs4sk21+jv2MiLk23Bkp+8slSaIDVR//58fxJkMx17PHyo2cDO/69A==}
         engines: {node: '>=14'}
    @@ -7401,6 +7408,13 @@ packages:
           micromatch: 4.0.5
         dev: true
     
    +  /cspell-glob@6.31.3:
    +    resolution: {integrity: sha512-+koUJPSCOittQwhR0T1mj4xXT3N+ZnY2qQ53W6Gz9HY3hVfEEy0NpbwE/Uy7sIvFMbc426fK0tGXjXyIj72uhQ==}
    +    engines: {node: '>=14'}
    +    dependencies:
    +      micromatch: 4.0.5
    +    dev: true
    +
       /cspell-grammar@6.31.1:
         resolution: {integrity: sha512-AsRVP0idcNFVSb9+p9XjMumFj3BUV67WIPWApaAzJl/dYyiIygQObRE+si0/QtFWGNw873b7hNhWZiKjqIdoaQ==}
         engines: {node: '>=14'}
    @@ -7410,12 +7424,31 @@ packages:
           '@cspell/cspell-types': 6.31.1
         dev: true
     
    +  /cspell-grammar@6.31.3:
    +    resolution: {integrity: sha512-TZYaOLIGAumyHlm4w7HYKKKcR1ZgEMKt7WNjCFqq7yGVW7U+qyjQqR8jqnLiUTZl7c2Tque4mca7n0CFsjVv5A==}
    +    engines: {node: '>=14'}
    +    hasBin: true
    +    dependencies:
    +      '@cspell/cspell-pipe': 6.31.3
    +      '@cspell/cspell-types': 6.31.3
    +    dev: true
    +
       /cspell-io@6.31.1:
         resolution: {integrity: sha512-deZcpvTYY/NmLfOdOtzcm+nDvJZozKmj4TY3pPpX0HquPX0A/w42bFRT/zZNmRslFl8vvrCZZUog7SOc6ha3uA==}
         engines: {node: '>=14'}
         dependencies:
           '@cspell/cspell-service-bus': 6.31.1
    -      node-fetch: 2.6.13
    +      node-fetch: 2.6.12(encoding@0.1.13)
    +    transitivePeerDependencies:
    +      - encoding
    +    dev: true
    +
    +  /cspell-io@6.31.3:
    +    resolution: {integrity: sha512-yCnnQ5bTbngUuIAaT5yNSdI1P0Kc38uvC8aynNi7tfrCYOQbDu1F9/DcTpbdhrsCv+xUn2TB1YjuCmm0STfJlA==}
    +    engines: {node: '>=14'}
    +    dependencies:
    +      '@cspell/cspell-service-bus': 6.31.3
    +      node-fetch: 2.6.12(encoding@0.1.13)
         transitivePeerDependencies:
           - encoding
         dev: true
    @@ -7449,6 +7482,35 @@ packages:
           - encoding
         dev: true
     
    +  /cspell-lib@6.31.3:
    +    resolution: {integrity: sha512-Dv55aecaMvT/5VbNryKo0Zos8dtHon7e1K0z8DR4/kGZdQVT0bOFWeotSLhuaIqoNFdEt8ypfKbrIHIdbgt1Hg==}
    +    engines: {node: '>=14.6'}
    +    dependencies:
    +      '@cspell/cspell-bundled-dicts': 6.31.3
    +      '@cspell/cspell-pipe': 6.31.3
    +      '@cspell/cspell-types': 6.31.3
    +      '@cspell/strong-weak-map': 6.31.3
    +      clear-module: 4.1.2
    +      comment-json: 4.2.3
    +      configstore: 5.0.1
    +      cosmiconfig: 8.0.0
    +      cspell-dictionary: 6.31.3
    +      cspell-glob: 6.31.3
    +      cspell-grammar: 6.31.3
    +      cspell-io: 6.31.3
    +      cspell-trie-lib: 6.31.3
    +      fast-equals: 4.0.3
    +      find-up: 5.0.0
    +      gensequence: 5.0.2
    +      import-fresh: 3.3.0
    +      resolve-from: 5.0.0
    +      resolve-global: 1.0.0
    +      vscode-languageserver-textdocument: 1.0.8
    +      vscode-uri: 3.0.7
    +    transitivePeerDependencies:
    +      - encoding
    +    dev: true
    +
       /cspell-trie-lib@6.31.1:
         resolution: {integrity: sha512-MtYh7s4Sbr1rKT31P2BK6KY+YfOy3dWsuusq9HnqCXmq6aZ1HyFgjH/9p9uvqGi/TboMqn1KOV8nifhXK3l3jg==}
         engines: {node: '>=14'}
    @@ -7458,6 +7520,15 @@ packages:
           gensequence: 5.0.2
         dev: true
     
    +  /cspell-trie-lib@6.31.3:
    +    resolution: {integrity: sha512-HNUcLWOZAvtM3E34U+7/mSSpO0F6nLd/kFlRIcvSvPb9taqKe8bnSa0Yyb3dsdMq9rMxUmuDQtF+J6arZK343g==}
    +    engines: {node: '>=14'}
    +    dependencies:
    +      '@cspell/cspell-pipe': 6.31.3
    +      '@cspell/cspell-types': 6.31.3
    +      gensequence: 5.0.2
    +    dev: true
    +
       /cspell@6.31.1:
         resolution: {integrity: sha512-gyCtpkOpwI/TGibbtIgMBFnAUUp2hnYdvW/9Ky4RcneHtLH0+V/jUEbZD8HbRKz0GVZ6mhKWbNRSEyP9p3Cejw==}
         engines: {node: '>=14'}
    @@ -7536,32 +7607,32 @@ packages:
         resolution: {integrity: sha512-d4ZVpCW31eWwCMe1YT3ur7mUDnTXbgwyzaL320DrcRT45rfjYxkt5QWLrmOJ+/UEAI2+fQgKe/fCjR8l4TpRgw==}
         dev: true
     
    -  /cypress-image-snapshot@4.0.1(cypress@12.10.0)(jest@29.5.0):
    +  /cypress-image-snapshot@4.0.1(cypress@12.17.3)(jest@29.6.2):
         resolution: {integrity: sha512-PBpnhX/XItlx3/DAk5ozsXQHUi72exybBNH5Mpqj1DVmjq+S5Jd9WE5CRa4q5q0zuMZb2V2VpXHth6MjFpgj9Q==}
         engines: {node: '>=8'}
         peerDependencies:
           cypress: ^4.5.0
         dependencies:
           chalk: 2.4.2
    -      cypress: 12.10.0
    +      cypress: 12.17.3
           fs-extra: 7.0.1
           glob: 7.2.3
    -      jest-image-snapshot: 4.2.0(jest@29.5.0)
    +      jest-image-snapshot: 4.2.0(jest@29.6.2)
           pkg-dir: 3.0.0
           term-img: 4.1.0
         transitivePeerDependencies:
           - jest
         dev: true
     
    -  /cypress@12.10.0:
    -    resolution: {integrity: sha512-Y0wPc221xKKW1/4iAFCphkrG2jNR4MjOne3iGn4mcuCaE7Y5EtXL83N8BzRsAht7GYfWVjJ/UeTqEdDKHz39HQ==}
    +  /cypress@12.17.3:
    +    resolution: {integrity: sha512-/R4+xdIDjUSLYkiQfwJd630S81KIgicmQOLXotFxVXkl+eTeVO+3bHXxdi5KBh/OgC33HWN33kHX+0tQR/ZWpg==}
         engines: {node: ^14.0.0 || ^16.0.0 || >=18.0.0}
         hasBin: true
         requiresBuild: true
         dependencies:
           '@cypress/request': 2.88.12
           '@cypress/xvfb': 1.2.4(supports-color@8.1.1)
    -      '@types/node': 14.18.54
    +      '@types/node': 16.18.40
           '@types/sinonjs__fake-timers': 8.1.1
           '@types/sizzle': 2.3.3
           arch: 2.2.0
    @@ -7871,6 +7942,42 @@ packages:
           d3-transition: 3.0.1(d3-selection@3.0.0)
         dev: false
     
    +  /d3@7.0.0:
    +    resolution: {integrity: sha512-t+jEKGO2jQiSBLJYYq6RFc500tsCeXBB4x41oQaSnZD3Som95nQrlw9XJGrFTMUOQOkwSMauWy9+8Tz1qm9UZw==}
    +    engines: {node: '>=12'}
    +    dependencies:
    +      d3-array: 3.2.4
    +      d3-axis: 3.0.0
    +      d3-brush: 3.0.0
    +      d3-chord: 3.0.1
    +      d3-color: 3.1.0
    +      d3-contour: 3.1.0
    +      d3-delaunay: 6.0.4
    +      d3-dispatch: 3.0.1
    +      d3-drag: 3.0.0
    +      d3-dsv: 3.0.1
    +      d3-ease: 3.0.1
    +      d3-fetch: 3.0.1
    +      d3-force: 3.0.0
    +      d3-format: 3.1.0
    +      d3-geo: 3.1.0
    +      d3-hierarchy: 3.1.2
    +      d3-interpolate: 3.0.1
    +      d3-path: 3.1.0
    +      d3-polygon: 3.0.1
    +      d3-quadtree: 3.0.1
    +      d3-random: 3.0.1
    +      d3-scale: 4.0.2
    +      d3-scale-chromatic: 3.0.0
    +      d3-selection: 3.0.0
    +      d3-shape: 3.2.0
    +      d3-time: 3.1.0
    +      d3-time-format: 4.1.0
    +      d3-timer: 3.0.1
    +      d3-transition: 3.0.1(d3-selection@3.0.0)
    +      d3-zoom: 3.0.0
    +    dev: false
    +
       /d3@7.4.0:
         resolution: {integrity: sha512-/xKyIYpKzd+I2DhiS2ANYJtEfHkE9lHKBFwqsplKsazPcXy2N1KIJSMTJsRk42jHbHCH0KPJGd0RnBt6NBJ1MA==}
         engines: {node: '>=12'}
    @@ -7999,10 +8106,6 @@ packages:
           '@babel/runtime': 7.22.10
         dev: true
     
    -  /dayjs@1.10.7:
    -    resolution: {integrity: sha512-P6twpd70BcPK34K26uJ1KT3wlhpuOAPoMwJzpsIWUxHZ7wpmbdZL/hQqBDfz7hGurYSa5PhzdhDHtt319hL3ig==}
    -    dev: true
    -
       /dayjs@1.11.7:
         resolution: {integrity: sha512-+Yw9U6YO5TQohxLcIkrXBeY73WP3ejHWVvx8XCk3gxvQDCTEmS48ZrSZCKciI7Bhl/uCMyxYtE9UqRILmFphkQ==}
         dev: false
    @@ -8350,8 +8453,8 @@ packages:
           jake: 10.8.7
         dev: true
     
    -  /electron-to-chromium@1.4.496:
    -    resolution: {integrity: sha512-qeXC3Zbykq44RCrBa4kr8v/dWzYJA8rAwpyh9Qd+NKWoJfjG5vvJqy9XOJ9H4P/lqulZBCgUWAYi+FeK5AuJ8g==}
    +  /electron-to-chromium@1.4.490:
    +    resolution: {integrity: sha512-6s7NVJz+sATdYnIwhdshx/N/9O6rvMxmhVoDSDFdj6iA45gHR8EQje70+RYsF4GeB+k0IeNSBnP7yG9ZXJFr7A==}
         dev: true
     
       /elkjs@0.8.2:
    @@ -8536,36 +8639,6 @@ packages:
           es6-symbol: 3.1.3
         dev: true
     
    -  /esbuild@0.17.19:
    -    resolution: {integrity: sha512-XQ0jAPFkK/u3LcVRcvVHQcTIqD6E2H1fvZMA5dQPSOWb3suUbWbfbRf94pjc0bNzRYLfIrDRQXr7X+LHIm5oHw==}
    -    engines: {node: '>=12'}
    -    hasBin: true
    -    requiresBuild: true
    -    optionalDependencies:
    -      '@esbuild/android-arm': 0.17.19
    -      '@esbuild/android-arm64': 0.17.19
    -      '@esbuild/android-x64': 0.17.19
    -      '@esbuild/darwin-arm64': 0.17.19
    -      '@esbuild/darwin-x64': 0.17.19
    -      '@esbuild/freebsd-arm64': 0.17.19
    -      '@esbuild/freebsd-x64': 0.17.19
    -      '@esbuild/linux-arm': 0.17.19
    -      '@esbuild/linux-arm64': 0.17.19
    -      '@esbuild/linux-ia32': 0.17.19
    -      '@esbuild/linux-loong64': 0.17.19
    -      '@esbuild/linux-mips64el': 0.17.19
    -      '@esbuild/linux-ppc64': 0.17.19
    -      '@esbuild/linux-riscv64': 0.17.19
    -      '@esbuild/linux-s390x': 0.17.19
    -      '@esbuild/linux-x64': 0.17.19
    -      '@esbuild/netbsd-x64': 0.17.19
    -      '@esbuild/openbsd-x64': 0.17.19
    -      '@esbuild/sunos-x64': 0.17.19
    -      '@esbuild/win32-arm64': 0.17.19
    -      '@esbuild/win32-ia32': 0.17.19
    -      '@esbuild/win32-x64': 0.17.19
    -    dev: true
    -
       /esbuild@0.18.20:
         resolution: {integrity: sha512-ceqxoedUrcayh7Y7ZX6NdbbDzGROiyVBgC4PriJThBKSVPWnnFHZAkfI1lJT8QFkOwH4qOS2SJkS4wvpGl8BpA==}
         engines: {node: '>=12'}
    @@ -8679,22 +8752,22 @@ packages:
           source-map: 0.6.1
         dev: true
     
    -  /eslint-config-prettier@8.8.0(eslint@8.39.0):
    -    resolution: {integrity: sha512-wLbQiFre3tdGgpDv67NQKnJuTlcUVYHas3k+DZCc2U2BadthoEY4B7hLPvAxaqdyOGCzuLfii2fqGph10va7oA==}
    +  /eslint-config-prettier@8.10.0(eslint@8.47.0):
    +    resolution: {integrity: sha512-SM8AMJdeQqRYT9O9zguiruQZaN7+z+E4eAP9oiLNGKMtomwaB1E9dcgUD6ZAn/eQAb52USbvezbiljfZUhbJcg==}
         hasBin: true
         peerDependencies:
           eslint: '>=7.0.0'
         dependencies:
    -      eslint: 8.39.0
    +      eslint: 8.47.0
         dev: true
     
    -  /eslint-plugin-cypress@2.13.2(eslint@8.39.0):
    -    resolution: {integrity: sha512-LlwjnBTzuKuC0A4H0RxVjs0YeAWK+CD1iM9Dp8un3lzT713ePQxfpPstCD+9HSAss8emuE3b2hCNUST+NrUwKw==}
    +  /eslint-plugin-cypress@2.14.0(eslint@8.47.0):
    +    resolution: {integrity: sha512-eW6tv7iIg7xujleAJX4Ujm649Bf5jweqa4ObPEIuueYRyLZt7qXGWhCY/n4bfeFW/j6nQZwbIBHKZt6EKcL/cg==}
         peerDependencies:
           eslint: '>= 3.2.1'
         dependencies:
    -      eslint: 8.39.0
    -      globals: 11.12.0
    +      eslint: 8.47.0
    +      globals: 13.21.0
         dev: true
     
       /eslint-plugin-html@7.1.0:
    @@ -8703,11 +8776,11 @@ packages:
           htmlparser2: 8.0.2
         dev: true
     
    -  /eslint-plugin-jest@27.2.1(@typescript-eslint/eslint-plugin@6.9.1)(eslint@8.39.0)(jest@29.5.0)(typescript@5.1.3):
    -    resolution: {integrity: sha512-l067Uxx7ZT8cO9NJuf+eJHvt6bqJyz2Z29wykyEdz/OtmcELQl2MQGQLX8J94O1cSJWAwUSEvCjwjA7KEK3Hmg==}
    +  /eslint-plugin-jest@27.2.3(@typescript-eslint/eslint-plugin@6.7.2)(eslint@8.47.0)(jest@29.6.2)(typescript@5.1.6):
    +    resolution: {integrity: sha512-sRLlSCpICzWuje66Gl9zvdF6mwD5X86I4u55hJyFBsxYOsBCmT5+kSUjf+fkFWVMMgpzNEupjW8WzUqi83hJAQ==}
         engines: {node: ^14.15.0 || ^16.10.0 || >=18.0.0}
         peerDependencies:
    -      '@typescript-eslint/eslint-plugin': ^5.0.0
    +      '@typescript-eslint/eslint-plugin': ^5.0.0 || ^6.0.0
           eslint: ^7.0.0 || ^8.0.0
           jest: '*'
         peerDependenciesMeta:
    @@ -8716,28 +8789,29 @@ packages:
           jest:
             optional: true
         dependencies:
    -      '@typescript-eslint/eslint-plugin': 6.9.1(@typescript-eslint/parser@6.9.1)(eslint@8.39.0)(typescript@5.1.3)
    -      '@typescript-eslint/utils': 5.62.0(eslint@8.39.0)(typescript@5.1.3)
    -      eslint: 8.39.0
    -      jest: 29.5.0(@types/node@18.16.0)(ts-node@10.9.1)
    +      '@typescript-eslint/eslint-plugin': 6.7.2(@typescript-eslint/parser@6.7.2)(eslint@8.47.0)(typescript@5.1.6)
    +      '@typescript-eslint/utils': 5.62.0(eslint@8.47.0)(typescript@5.1.6)
    +      eslint: 8.47.0
    +      jest: 29.6.2(@types/node@18.17.5)(ts-node@10.9.1)
         transitivePeerDependencies:
           - supports-color
           - typescript
         dev: true
     
    -  /eslint-plugin-jsdoc@46.0.0(eslint@8.39.0):
    -    resolution: {integrity: sha512-xmB5WleBcPCFYlrFfdjrcfSKOJBLwyGmKa+i+fVqlIHp8g5aAoeQpBGugUzToFtQgd4hNZYlfIcP7QSxC9NYWQ==}
    +  /eslint-plugin-jsdoc@46.4.6(eslint@8.47.0):
    +    resolution: {integrity: sha512-z4SWYnJfOqftZI+b3RM9AtWL1vF/sLWE/LlO9yOKDof9yN2+n3zOdOJTGX/pRE/xnPsooOLG2Rq6e4d+XW3lNw==}
         engines: {node: '>=16'}
         peerDependencies:
           eslint: ^7.0.0 || ^8.0.0
         dependencies:
    -      '@es-joy/jsdoccomment': 0.39.4
    +      '@es-joy/jsdoccomment': 0.40.1
           are-docs-informative: 0.0.2
    -      comment-parser: 1.3.1
    +      comment-parser: 1.4.0
           debug: 4.3.4(supports-color@8.1.1)
           escape-string-regexp: 4.0.0
    -      eslint: 8.39.0
    +      eslint: 8.47.0
           esquery: 1.5.0
    +      is-builtin-module: 3.2.1
           semver: 7.5.4
           spdx-expression-parse: 3.0.1
         transitivePeerDependencies:
    @@ -8752,23 +8826,23 @@ packages:
           vscode-json-languageservice: 4.2.1
         dev: true
     
    -  /eslint-plugin-lodash@7.4.0(eslint@8.39.0):
    +  /eslint-plugin-lodash@7.4.0(eslint@8.47.0):
         resolution: {integrity: sha512-Tl83UwVXqe1OVeBRKUeWcfg6/pCW1GTRObbdnbEJgYwjxp5Q92MEWQaH9+dmzbRt6kvYU1Mp893E79nJiCSM8A==}
         engines: {node: '>=10'}
         peerDependencies:
           eslint: '>=2'
         dependencies:
    -      eslint: 8.39.0
    +      eslint: 8.47.0
           lodash: 4.17.21
         dev: true
     
    -  /eslint-plugin-markdown@3.0.0(eslint@8.39.0):
    -    resolution: {integrity: sha512-hRs5RUJGbeHDLfS7ELanT0e29Ocyssf/7kBM+p7KluY5AwngGkDf8Oyu4658/NZSGTTq05FZeWbkxXtbVyHPwg==}
    +  /eslint-plugin-markdown@3.0.1(eslint@8.47.0):
    +    resolution: {integrity: sha512-8rqoc148DWdGdmYF6WSQFT3uQ6PO7zXYgeBpHAOAakX/zpq+NvFYbDA/H7PYzHajwtmaOzAwfxyl++x0g1/N9A==}
         engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0}
         peerDependencies:
           eslint: ^6.0.0 || ^7.0.0 || ^8.0.0
         dependencies:
    -      eslint: 8.39.0
    +      eslint: 8.47.0
           mdast-util-from-markdown: 0.8.5
         transitivePeerDependencies:
           - supports-color
    @@ -8786,99 +8860,50 @@ packages:
           '@microsoft/tsdoc-config': 0.16.2
         dev: true
     
    -  /eslint-plugin-unicorn@47.0.0(eslint@8.39.0):
    +  /eslint-plugin-unicorn@47.0.0(eslint@8.47.0):
         resolution: {integrity: sha512-ivB3bKk7fDIeWOUmmMm9o3Ax9zbMz1Bsza/R2qm46ufw4T6VBFBaJIR1uN3pCKSmSXm8/9Nri8V+iUut1NhQGA==}
         engines: {node: '>=16'}
         peerDependencies:
           eslint: '>=8.38.0'
    -    dependencies:
    -      '@babel/helper-validator-identifier': 7.22.5
    -      '@eslint-community/eslint-utils': 4.4.0(eslint@8.39.0)
    -      ci-info: 3.8.0
    -      clean-regexp: 1.0.0
    -      eslint: 8.39.0
    -      esquery: 1.5.0
    -      indent-string: 4.0.0
    -      is-builtin-module: 3.2.1
    -      jsesc: 3.0.2
    -      lodash: 4.17.21
    -      pluralize: 8.0.0
    -      read-pkg-up: 7.0.1
    -      regexp-tree: 0.1.27
    -      regjsparser: 0.10.0
    -      safe-regex: 2.1.1
    -      semver: 7.5.4
    -      strip-indent: 3.0.0
    -    dev: true
    -
    -  /eslint-scope@5.1.1:
    -    resolution: {integrity: sha512-2NxwbF/hZ0KpepYN0cNbo+FN6XoK7GaHlQhgx/hIZl6Va0bF45RQOOwhLIy8lQDbuCiadSLCBnH2CFYquit5bw==}
    -    engines: {node: '>=8.0.0'}
    -    dependencies:
    -      esrecurse: 4.3.0
    -      estraverse: 4.3.0
    -    dev: true
    -
    -  /eslint-scope@7.2.2:
    -    resolution: {integrity: sha512-dOt21O7lTMhDM+X9mB4GX+DZrZtCUJPL/wlcTqxyrx5IvO0IYtILdtrQGQp+8n5S0gwSVmOf9NQrjMOgfQZlIg==}
    -    engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0}
    -    dependencies:
    -      esrecurse: 4.3.0
    -      estraverse: 5.3.0
    -    dev: true
    -
    -  /eslint-visitor-keys@3.4.3:
    -    resolution: {integrity: sha512-wpc+LXeiyiisxPlEkUzU6svyS1frIO3Mgxj1fdy7Pm8Ygzguax2N3Fa/D/ag1WqbOprdI+uY6wMUl8/a2G+iag==}
    -    engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0}
    -    dev: true
    -
    -  /eslint@8.39.0:
    -    resolution: {integrity: sha512-mwiok6cy7KTW7rBpo05k6+p4YVZByLNjAZ/ACB9DRCu4YDRwjXI01tWHp6KAUWelsBetTxKK/2sHB0vdS8Z2Og==}
    -    engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0}
    -    hasBin: true
    -    dependencies:
    -      '@eslint-community/eslint-utils': 4.4.0(eslint@8.39.0)
    -      '@eslint-community/regexpp': 4.6.2
    -      '@eslint/eslintrc': 2.1.2
    -      '@eslint/js': 8.39.0
    -      '@humanwhocodes/config-array': 0.11.10
    -      '@humanwhocodes/module-importer': 1.0.1
    -      '@nodelib/fs.walk': 1.2.8
    -      ajv: 6.12.6
    -      chalk: 4.1.2
    -      cross-spawn: 7.0.3
    -      debug: 4.3.4(supports-color@8.1.1)
    -      doctrine: 3.0.0
    -      escape-string-regexp: 4.0.0
    -      eslint-scope: 7.2.2
    -      eslint-visitor-keys: 3.4.3
    -      espree: 9.6.1
    -      esquery: 1.5.0
    -      esutils: 2.0.3
    -      fast-deep-equal: 3.1.3
    -      file-entry-cache: 6.0.1
    -      find-up: 5.0.0
    -      glob-parent: 6.0.2
    -      globals: 13.21.0
    -      grapheme-splitter: 1.0.4
    -      ignore: 5.2.4
    -      import-fresh: 3.3.0
    -      imurmurhash: 0.1.4
    -      is-glob: 4.0.3
    -      is-path-inside: 3.0.3
    -      js-sdsl: 4.4.2
    -      js-yaml: 4.1.0
    -      json-stable-stringify-without-jsonify: 1.0.1
    -      levn: 0.4.1
    -      lodash.merge: 4.6.2
    -      minimatch: 3.1.2
    -      natural-compare: 1.4.0
    -      optionator: 0.9.3
    -      strip-ansi: 6.0.1
    -      strip-json-comments: 3.1.1
    -      text-table: 0.2.0
    -    transitivePeerDependencies:
    -      - supports-color
    +    dependencies:
    +      '@babel/helper-validator-identifier': 7.22.5
    +      '@eslint-community/eslint-utils': 4.4.0(eslint@8.47.0)
    +      ci-info: 3.8.0
    +      clean-regexp: 1.0.0
    +      eslint: 8.47.0
    +      esquery: 1.5.0
    +      indent-string: 4.0.0
    +      is-builtin-module: 3.2.1
    +      jsesc: 3.0.2
    +      lodash: 4.17.21
    +      pluralize: 8.0.0
    +      read-pkg-up: 7.0.1
    +      regexp-tree: 0.1.27
    +      regjsparser: 0.10.0
    +      safe-regex: 2.1.1
    +      semver: 7.5.4
    +      strip-indent: 3.0.0
    +    dev: true
    +
    +  /eslint-scope@5.1.1:
    +    resolution: {integrity: sha512-2NxwbF/hZ0KpepYN0cNbo+FN6XoK7GaHlQhgx/hIZl6Va0bF45RQOOwhLIy8lQDbuCiadSLCBnH2CFYquit5bw==}
    +    engines: {node: '>=8.0.0'}
    +    dependencies:
    +      esrecurse: 4.3.0
    +      estraverse: 4.3.0
    +    dev: true
    +
    +  /eslint-scope@7.2.2:
    +    resolution: {integrity: sha512-dOt21O7lTMhDM+X9mB4GX+DZrZtCUJPL/wlcTqxyrx5IvO0IYtILdtrQGQp+8n5S0gwSVmOf9NQrjMOgfQZlIg==}
    +    engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0}
    +    dependencies:
    +      esrecurse: 4.3.0
    +      estraverse: 5.3.0
    +    dev: true
    +
    +  /eslint-visitor-keys@3.4.3:
    +    resolution: {integrity: sha512-wpc+LXeiyiisxPlEkUzU6svyS1frIO3Mgxj1fdy7Pm8Ygzguax2N3Fa/D/ag1WqbOprdI+uY6wMUl8/a2G+iag==}
    +    engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0}
         dev: true
     
       /eslint@8.47.0:
    @@ -9115,7 +9140,7 @@ packages:
         engines: {node: ^14.15.0 || ^16.10.0 || >=18.0.0}
         dependencies:
           '@jest/expect-utils': 29.6.2
    -      '@types/node': 18.17.6
    +      '@types/node': 18.17.5
           jest-get-type: 29.4.3
           jest-matcher-utils: 29.6.2
           jest-message-util: 29.6.2
    @@ -9283,7 +9308,7 @@ packages:
           proxy-addr: 2.0.7
           rfdc: 1.3.0
           secure-json-parse: 2.7.0
    -      semver: 7.3.8
    +      semver: 7.5.4
           tiny-lru: 8.0.2
         transitivePeerDependencies:
           - supports-color
    @@ -9473,8 +9498,8 @@ packages:
         resolution: {integrity: sha512-XGozTsMPYkm+6b5QL3Z9wQcJjNYxp0CYn3U1gO7dwD6PAqU1SVWZxI9CCg3z+ml3YfqdPnrBehaBrnH2AGKbNA==}
         dev: true
     
    -  /focus-trap@7.5.2:
    -    resolution: {integrity: sha512-p6vGNNWLDGwJCiEjkSK6oERj/hEyI9ITsSwIUICBoKLlWiTWXJRfQibCwcoi50rTZdbi87qDtUlMCmQwsGSgPw==}
    +  /focus-trap@7.5.4:
    +    resolution: {integrity: sha512-N7kHdlgsO/v+iD/dMoJKtsSqs5Dz/dXZVebRgJw23LDk+jMi/974zyiOYDziY2JPp8xivq9BmUGwIJMiuSBi7w==}
         dependencies:
           tabbable: 6.2.0
         dev: true
    @@ -9767,7 +9792,7 @@ packages:
         hasBin: true
         dependencies:
           foreground-child: 3.1.1
    -      jackspeak: 2.3.0
    +      jackspeak: 2.2.3
           minimatch: 9.0.3
           minipass: 7.0.3
           path-scurry: 1.10.1
    @@ -9874,6 +9899,17 @@ packages:
           slash: 4.0.0
         dev: true
     
    +  /globby@13.2.2:
    +    resolution: {integrity: sha512-Y1zNGV+pzQdh7H39l9zgB4PJqjRNqydvdYCDG4HFXM4XuvSaQQlEc91IU1yALL8gUTDomgBAfz3XJdmUS+oo0w==}
    +    engines: {node: ^12.20.0 || ^14.13.1 || >=16.0.0}
    +    dependencies:
    +      dir-glob: 3.0.1
    +      fast-glob: 3.3.1
    +      ignore: 5.2.4
    +      merge2: 1.4.1
    +      slash: 4.0.0
    +    dev: true
    +
       /glur@1.1.2:
         resolution: {integrity: sha512-l+8esYHTKOx2G/Aao4lEQ0bnHWg4fWtJbVoZZT9Knxi01pB8C80BR85nONLFwkkQoFRCmXY+BUcGZN3yZ2QsRA==}
         dev: true
    @@ -10675,7 +10711,7 @@ packages:
         engines: {node: '>=8'}
         dependencies:
           '@babel/core': 7.22.10
    -      '@babel/parser': 7.22.10
    +      '@babel/parser': 7.23.0
           '@istanbuljs/schema': 0.1.3
           istanbul-lib-coverage: 3.2.0
           semver: 6.3.1
    @@ -10731,8 +10767,8 @@ packages:
           plist: 3.1.0
         dev: true
     
    -  /jackspeak@2.3.0:
    -    resolution: {integrity: sha512-uKmsITSsF4rUWQHzqaRUuyAir3fZfW3f202Ee34lz/gZCi970CPZwyQXLGNgWJvvZbvFyzeyGq0+4fcG/mBKZg==}
    +  /jackspeak@2.2.3:
    +    resolution: {integrity: sha512-pF0kfjmg8DJLxDrizHoCZGUFz4P4czQ3HyfW4BU0ffebYkzAVlBywp5zaxW/TM+r0sGbmrQdi8EQQVTJFxnGsQ==}
         engines: {node: '>=14'}
         dependencies:
           '@isaacs/cliui': 8.0.2
    @@ -10767,7 +10803,7 @@ packages:
           '@jest/expect': 29.6.2
           '@jest/test-result': 29.6.2
           '@jest/types': 29.6.1
    -      '@types/node': 18.17.6
    +      '@types/node': 18.17.5
           chalk: 4.1.2
           co: 4.6.0
           dedent: 1.5.1
    @@ -10788,7 +10824,7 @@ packages:
           - supports-color
         dev: true
     
    -  /jest-cli@29.6.2(@types/node@18.16.0)(ts-node@10.9.1):
    +  /jest-cli@29.6.2(@types/node@18.17.5)(ts-node@10.9.1):
         resolution: {integrity: sha512-TT6O247v6dCEX2UGHGyflMpxhnrL0DNqP2fRTKYm3nJJpCTfXX3GCMQPGFjXDoj0i5/Blp3jriKXFgdfmbYB6Q==}
         engines: {node: ^14.15.0 || ^16.10.0 || >=18.0.0}
         hasBin: true
    @@ -10805,7 +10841,7 @@ packages:
           exit: 0.1.2
           graceful-fs: 4.2.11
           import-local: 3.1.0
    -      jest-config: 29.6.2(@types/node@18.16.0)(ts-node@10.9.1)
    +      jest-config: 29.6.2(@types/node@18.17.5)(ts-node@10.9.1)
           jest-util: 29.6.2
           jest-validate: 29.6.2
           prompts: 2.4.2
    @@ -10817,48 +10853,7 @@ packages:
           - ts-node
         dev: true
     
    -  /jest-config@29.6.2(@types/node@18.16.0)(ts-node@10.9.1):
    -    resolution: {integrity: sha512-VxwFOC8gkiJbuodG9CPtMRjBUNZEHxwfQXmIudSTzFWxaci3Qub1ddTRbFNQlD/zUeaifLndh/eDccFX4wCMQw==}
    -    engines: {node: ^14.15.0 || ^16.10.0 || >=18.0.0}
    -    peerDependencies:
    -      '@types/node': '*'
    -      ts-node: '>=9.0.0'
    -    peerDependenciesMeta:
    -      '@types/node':
    -        optional: true
    -      ts-node:
    -        optional: true
    -    dependencies:
    -      '@babel/core': 7.22.10
    -      '@jest/test-sequencer': 29.6.2
    -      '@jest/types': 29.6.1
    -      '@types/node': 18.16.0
    -      babel-jest: 29.6.2(@babel/core@7.22.10)
    -      chalk: 4.1.2
    -      ci-info: 3.8.0
    -      deepmerge: 4.3.1
    -      glob: 7.2.3
    -      graceful-fs: 4.2.11
    -      jest-circus: 29.6.2
    -      jest-environment-node: 29.6.2
    -      jest-get-type: 29.4.3
    -      jest-regex-util: 29.4.3
    -      jest-resolve: 29.6.2
    -      jest-runner: 29.6.2
    -      jest-util: 29.6.2
    -      jest-validate: 29.6.2
    -      micromatch: 4.0.5
    -      parse-json: 5.2.0
    -      pretty-format: 29.6.2
    -      slash: 3.0.0
    -      strip-json-comments: 3.1.1
    -      ts-node: 10.9.1(@types/node@18.16.0)(typescript@5.1.3)
    -    transitivePeerDependencies:
    -      - babel-plugin-macros
    -      - supports-color
    -    dev: true
    -
    -  /jest-config@29.6.2(@types/node@18.17.6)(ts-node@10.9.1):
    +  /jest-config@29.6.2(@types/node@18.17.5)(ts-node@10.9.1):
         resolution: {integrity: sha512-VxwFOC8gkiJbuodG9CPtMRjBUNZEHxwfQXmIudSTzFWxaci3Qub1ddTRbFNQlD/zUeaifLndh/eDccFX4wCMQw==}
         engines: {node: ^14.15.0 || ^16.10.0 || >=18.0.0}
         peerDependencies:
    @@ -10873,7 +10868,7 @@ packages:
           '@babel/core': 7.22.10
           '@jest/test-sequencer': 29.6.2
           '@jest/types': 29.6.1
    -      '@types/node': 18.17.6
    +      '@types/node': 18.17.5
           babel-jest: 29.6.2(@babel/core@7.22.10)
           chalk: 4.1.2
           ci-info: 3.8.0
    @@ -10893,7 +10888,7 @@ packages:
           pretty-format: 29.6.2
           slash: 3.0.0
           strip-json-comments: 3.1.1
    -      ts-node: 10.9.1(@types/node@18.16.0)(typescript@5.1.3)
    +      ts-node: 10.9.1(@types/node@18.17.5)(typescript@5.1.6)
         transitivePeerDependencies:
           - babel-plugin-macros
           - supports-color
    @@ -10934,7 +10929,7 @@ packages:
           '@jest/environment': 29.6.2
           '@jest/fake-timers': 29.6.2
           '@jest/types': 29.6.1
    -      '@types/node': 18.17.6
    +      '@types/node': 18.17.5
           jest-mock: 29.6.2
           jest-util: 29.6.2
         dev: true
    @@ -10950,7 +10945,7 @@ packages:
         dependencies:
           '@jest/types': 29.6.1
           '@types/graceful-fs': 4.1.6
    -      '@types/node': 18.17.6
    +      '@types/node': 18.17.5
           anymatch: 3.1.3
           fb-watchman: 2.0.2
           graceful-fs: 4.2.11
    @@ -10963,7 +10958,7 @@ packages:
           fsevents: 2.3.2
         dev: true
     
    -  /jest-image-snapshot@4.2.0(jest@29.5.0):
    +  /jest-image-snapshot@4.2.0(jest@29.6.2):
         resolution: {integrity: sha512-6aAqv2wtfOgxiJeBayBCqHo1zX+A12SUNNzo7rIxiXh6W6xYVu8QyHWkada8HeRi+QUTHddp0O0Xa6kmQr+xbQ==}
         engines: {node: '>= 10.14.2'}
         peerDependencies:
    @@ -10972,7 +10967,7 @@ packages:
           chalk: 1.1.3
           get-stdin: 5.0.1
           glur: 1.1.2
    -      jest: 29.5.0(@types/node@18.16.0)(ts-node@10.9.1)
    +      jest: 29.6.2(@types/node@18.17.5)(ts-node@10.9.1)
           lodash: 4.17.21
           mkdirp: 0.5.6
           pixelmatch: 5.3.0
    @@ -11019,7 +11014,7 @@ packages:
         engines: {node: ^14.15.0 || ^16.10.0 || >=18.0.0}
         dependencies:
           '@jest/types': 29.6.1
    -      '@types/node': 18.17.6
    +      '@types/node': 18.17.5
           jest-util: 29.6.2
         dev: true
     
    @@ -11074,7 +11069,7 @@ packages:
           '@jest/test-result': 29.6.2
           '@jest/transform': 29.6.2
           '@jest/types': 29.6.1
    -      '@types/node': 18.17.6
    +      '@types/node': 18.17.5
           chalk: 4.1.2
           emittery: 0.13.1
           graceful-fs: 4.2.11
    @@ -11105,7 +11100,7 @@ packages:
           '@jest/test-result': 29.6.2
           '@jest/transform': 29.6.2
           '@jest/types': 29.6.1
    -      '@types/node': 18.17.6
    +      '@types/node': 18.17.5
           chalk: 4.1.2
           cjs-module-lexer: 1.2.3
           collect-v8-coverage: 1.0.2
    @@ -11132,7 +11127,7 @@ packages:
           '@babel/generator': 7.22.10
           '@babel/plugin-syntax-jsx': 7.22.5(@babel/core@7.22.10)
           '@babel/plugin-syntax-typescript': 7.22.5(@babel/core@7.22.10)
    -      '@babel/types': 7.22.10
    +      '@babel/types': 7.23.0
           '@jest/expect-utils': 29.6.2
           '@jest/transform': 29.6.2
           '@jest/types': 29.6.1
    @@ -11157,7 +11152,7 @@ packages:
         engines: {node: ^14.15.0 || ^16.10.0 || >=18.0.0}
         dependencies:
           '@jest/types': 29.6.1
    -      '@types/node': 18.17.6
    +      '@types/node': 18.17.5
           chalk: 4.1.2
           ci-info: 3.8.0
           graceful-fs: 4.2.11
    @@ -11182,7 +11177,7 @@ packages:
         dependencies:
           '@jest/test-result': 29.6.2
           '@jest/types': 29.6.1
    -      '@types/node': 18.17.6
    +      '@types/node': 18.17.5
           ansi-escapes: 4.3.2
           chalk: 4.1.2
           emittery: 0.13.1
    @@ -11194,7 +11189,7 @@ packages:
         resolution: {integrity: sha512-KWYVV1c4i+jbMpaBC+U++4Va0cp8OisU185o73T1vo99hqi7w8tSJfUXYswwqqrjzwxa6KpRK54WhPvwf5w6PQ==}
         engines: {node: '>= 10.13.0'}
         dependencies:
    -      '@types/node': 18.17.6
    +      '@types/node': 18.17.5
           merge-stream: 2.0.0
           supports-color: 7.2.0
         dev: true
    @@ -11203,7 +11198,7 @@ packages:
         resolution: {integrity: sha512-7vuh85V5cdDofPyxn58nrPjBktZo0u9x1g8WtjQol+jZDaE+fhN+cIvTj11GndBnMnyfrUOG1sZQxCdjKh+DKg==}
         engines: {node: '>= 10.13.0'}
         dependencies:
    -      '@types/node': 18.17.6
    +      '@types/node': 18.17.5
           merge-stream: 2.0.0
           supports-color: 8.1.1
         dev: true
    @@ -11212,14 +11207,14 @@ packages:
         resolution: {integrity: sha512-l3ccBOabTdkng8I/ORCkADz4eSMKejTYv1vB/Z83UiubqhC1oQ5Li6dWCyqOIvSifGjUBxuvxvlm6KGK2DtuAQ==}
         engines: {node: ^14.15.0 || ^16.10.0 || >=18.0.0}
         dependencies:
    -      '@types/node': 18.17.6
    +      '@types/node': 18.17.5
           jest-util: 29.6.2
           merge-stream: 2.0.0
           supports-color: 8.1.1
         dev: true
     
    -  /jest@29.5.0(@types/node@18.16.0)(ts-node@10.9.1):
    -    resolution: {integrity: sha512-juMg3he2uru1QoXX078zTa7pO85QyB9xajZc6bU+d9yEGwrKX6+vGmJQ3UdVZsvTEUARIdObzH68QItim6OSSQ==}
    +  /jest@29.6.2(@types/node@18.17.5)(ts-node@10.9.1):
    +    resolution: {integrity: sha512-8eQg2mqFbaP7CwfsTpCxQ+sHzw1WuNWL5UUvjnWP4hx2riGz9fPSzYOaU5q8/GqWn1TfgZIVTqYJygbGbWAANg==}
         engines: {node: ^14.15.0 || ^16.10.0 || >=18.0.0}
         hasBin: true
         peerDependencies:
    @@ -11231,7 +11226,7 @@ packages:
           '@jest/core': 29.6.2(ts-node@10.9.1)
           '@jest/types': 29.6.1
           import-local: 3.1.0
    -      jest-cli: 29.6.2(@types/node@18.16.0)(ts-node@10.9.1)
    +      jest-cli: 29.6.2(@types/node@18.17.5)(ts-node@10.9.1)
         transitivePeerDependencies:
           - '@types/node'
           - babel-plugin-macros
    @@ -11268,9 +11263,15 @@ packages:
         hasBin: true
         dev: false
     
    -  /jiti@1.19.3:
    -    resolution: {integrity: sha512-5eEbBDQT/jF1xg6l36P+mWGGoH9Spuy0PCdSr2dtWRDGC6ph/w9ZCL4lmESW8f8F7MwT3XKescfP0wnZWAKL9w==}
    +  /jiti@1.19.1:
    +    resolution: {integrity: sha512-oVhqoRDaBXf7sjkll95LHVS6Myyyb1zaunVwk4Z0+WPSW4gjS0pl01zYKHScTuyEhQsFxV5L4DR5r+YqSyqyyg==}
    +    hasBin: true
    +    dev: false
    +
    +  /jiti@1.21.0:
    +    resolution: {integrity: sha512-gFqAIbuKyyso/3G2qhiO2OM6shY6EPP/R0+mkDbyspxKazh8BXDC5FiFsUjlczgdNz/vfra0da2y+aHrusLG/Q==}
         hasBin: true
    +    dev: true
     
       /jju@1.4.0:
         resolution: {integrity: sha512-8wb9Yw966OSxApiCt0K3yNJL8pnNeIv+OEq2YMidz4FKP6nonSRoOXc80iXY4JaN2FC11B9qsNmDsm+ZOfMROA==}
    @@ -11294,10 +11295,6 @@ packages:
         resolution: {integrity: sha512-3MEt5DTINKqfScXKfJFrRbxkrnk2AxPWGBL/ycjz4dK8iqiSJ06UxD8jh8xuh6p10TX4t2+7FsBYVxxQbMg+qA==}
         dev: true
     
    -  /js-sdsl@4.4.2:
    -    resolution: {integrity: sha512-dwXFwByc/ajSV6m5bcKAPwe4yDDF6D614pxmIi5odytzxRlwqF6nwoiCek80Ixc7Cvma5awClxrzFtxCQvcM8w==}
    -    dev: true
    -
       /js-tokens@4.0.0:
         resolution: {integrity: sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==}
         dev: true
    @@ -11326,6 +11323,48 @@ packages:
         engines: {node: '>=12.0.0'}
         dev: true
     
    +  /jsdom@19.0.0:
    +    resolution: {integrity: sha512-RYAyjCbxy/vri/CfnjUWJQQtZ3LKlLnDqj+9XLNnJPgEGeirZs3hllKR20re8LUZ6o1b1X4Jat+Qd26zmP41+A==}
    +    engines: {node: '>=12'}
    +    peerDependencies:
    +      canvas: ^2.5.0
    +    peerDependenciesMeta:
    +      canvas:
    +        optional: true
    +    dependencies:
    +      abab: 2.0.6
    +      acorn: 8.10.0
    +      acorn-globals: 6.0.0
    +      cssom: 0.5.0
    +      cssstyle: 2.3.0
    +      data-urls: 3.0.2
    +      decimal.js: 10.4.3
    +      domexception: 4.0.0
    +      escodegen: 2.1.0
    +      form-data: 4.0.0
    +      html-encoding-sniffer: 3.0.0
    +      http-proxy-agent: 5.0.0
    +      https-proxy-agent: 5.0.1
    +      is-potential-custom-element-name: 1.0.1
    +      nwsapi: 2.2.7
    +      parse5: 6.0.1
    +      saxes: 5.0.1
    +      symbol-tree: 3.2.4
    +      tough-cookie: 4.1.3
    +      w3c-hr-time: 1.0.2
    +      w3c-xmlserializer: 3.0.0
    +      webidl-conversions: 7.0.0
    +      whatwg-encoding: 2.0.0
    +      whatwg-mimetype: 3.0.0
    +      whatwg-url: 10.0.0
    +      ws: 8.13.0
    +      xml-name-validator: 4.0.0
    +    transitivePeerDependencies:
    +      - bufferutil
    +      - supports-color
    +      - utf-8-validate
    +    dev: true
    +
       /jsdom@22.0.0:
         resolution: {integrity: sha512-p5ZTEb5h+O+iU02t0GfEjAnkdYPrQSkfuTSMkMYyIoMvUNEHsbG0bHHbfXIcfTqD2UfvjQX7mmgiFsyRwGscVw==}
         engines: {node: '>=16'}
    @@ -11551,7 +11590,7 @@ packages:
         engines: {node: '>=16.0.0'}
         dependencies:
           chevrotain: 11.0.3
    -      chevrotain-allstar: 0.3.0(chevrotain@11.0.3)
    +      chevrotain-allstar: 0.3.1(chevrotain@11.0.3)
           vscode-languageserver: 9.0.1
           vscode-languageserver-textdocument: 1.0.11
           vscode-uri: 3.0.8
    @@ -11608,8 +11647,8 @@ packages:
           uc.micro: 1.0.6
         dev: true
     
    -  /lint-staged@13.2.1:
    -    resolution: {integrity: sha512-8gfzinVXoPfga5Dz/ZOn8I2GOhf81Wvs+KwbEXQn/oWZAvCVS2PivrXfVbFJc93zD16uC0neS47RXHIjXKYZQw==}
    +  /lint-staged@13.2.3:
    +    resolution: {integrity: sha512-zVVEXLuQIhr1Y7R7YAWx4TZLdvuzk7DnmrsTNL0fax6Z3jrpFcas+vKbzxhhvp6TA55m1SQuWkpzI1qbfDZbAg==}
         engines: {node: ^14.13.1 || >=16.0.0}
         hasBin: true
         dependencies:
    @@ -11858,7 +11897,6 @@ packages:
         engines: {node: '>=12'}
         dependencies:
           '@jridgewell/sourcemap-codec': 1.4.15
    -    dev: true
     
       /make-dir@3.1.0:
         resolution: {integrity: sha512-g3FeP20LNwhALb/6Cz6Dd4F2ngze0jz7tbzrD2wAV+o9FeNHe4rL+yK2md0J/fiSf1sa1ADhXqi5+oVwOM/eGw==}
    @@ -11946,7 +11984,7 @@ packages:
       /mdast-util-from-markdown@0.8.5:
         resolution: {integrity: sha512-2hkTXtYYnr+NubD/g6KGBS/0mFmBcifAsI0yIWRiRo0PjVs6SSOSOdtzbp6kSGnShDN6G5aWZpKQ2lWRy27mWQ==}
         dependencies:
    -      '@types/mdast': 3.0.11
    +      '@types/mdast': 3.0.12
           mdast-util-to-string: 2.0.0
           micromark: 2.11.4
           parse-entities: 2.0.0
    @@ -11976,7 +12014,7 @@ packages:
       /mdast-util-from-markdown@1.3.1:
         resolution: {integrity: sha512-4xTO/M8c82qBcnQc1tgpNtubGUW/Y1tBQ1B0i5CtSoelOLKFYlElIr3bvgREYYO5iRqbMY1YuqZng0GVOI8Qww==}
         dependencies:
    -      '@types/mdast': 3.0.11
    +      '@types/mdast': 3.0.12
           '@types/unist': 2.0.7
           decode-named-character-reference: 1.0.2
           mdast-util-to-string: 3.2.0
    @@ -12535,6 +12573,15 @@ packages:
           ufo: 1.2.0
         dev: true
     
    +  /mlly@1.4.2:
    +    resolution: {integrity: sha512-i/Ykufi2t1EZ6NaPLdfnZk2AX8cs0d+mTzVKuPfqPKPatxLApaBoxJQ9x1/uckXtrS/U5oisPMDkNs0yQTaBRg==}
    +    dependencies:
    +      acorn: 8.10.0
    +      pathe: 1.1.1
    +      pkg-types: 1.0.3
    +      ufo: 1.3.1
    +    dev: true
    +
       /mri@1.2.0:
         resolution: {integrity: sha512-tzzskb3bG8LvYGFF/mDTpq3jpI6Q9wc3LEmBaghu+DdCssd1FakN7Bc0hVNmEyGq1bq3RgfkCb3cmQLpNPOroA==}
         engines: {node: '>=4'}
    @@ -12613,12 +12660,12 @@ packages:
         engines: {node: '>=10.5.0'}
         dev: true
     
    -  /node-fetch-native@1.4.1:
    -    resolution: {integrity: sha512-NsXBU0UgBxo2rQLOeWNZqS3fvflWePMECr8CoSWoSTqCqGbVVsvl9vZu1HfQicYN0g5piV9Gh8RTEvo/uP752w==}
    +  /node-fetch-native@1.4.0:
    +    resolution: {integrity: sha512-F5kfEj95kX8tkDhUCYdV8dg3/8Olx/94zB8+ZNthFs6Bz31UpUi8Xh40TN3thLwXgrwXry1pEg9lJ++tLWTcqA==}
         dev: true
     
    -  /node-fetch@2.6.13:
    -    resolution: {integrity: sha512-StxNAxh15zr77QvvkmveSQ8uCQ4+v5FkvNTj0OESmiHu+VRi/gXArXtkWMElOsOUNLtUEvI4yS+rdtOHZTwlQA==}
    +  /node-fetch@2.6.12(encoding@0.1.13):
    +    resolution: {integrity: sha512-C/fGU2E8ToujUivIO0H+tpQ6HWo4eEmchoPIoXtxCrVghxdKq+QOHqEZW7tuP3KlV3bC8FRMO5nMCC7Zm1VP6g==}
         engines: {node: 4.x || >=6.0.0}
         peerDependencies:
           encoding: ^0.1.0
    @@ -12626,10 +12673,11 @@ packages:
           encoding:
             optional: true
         dependencies:
    +      encoding: 0.1.13
           whatwg-url: 5.0.0
         dev: true
     
    -  /node-fetch@2.6.7(encoding@0.1.13):
    +  /node-fetch@2.6.7:
         resolution: {integrity: sha512-ZjMPFEfVx5j+y2yF35Kzx5sF7kDzxuDj6ziH4FFbOp87zKDZNx8yExJIb05OGF4Nlt9IHFIMBkRl41VdvcNdbQ==}
         engines: {node: 4.x || >=6.0.0}
         peerDependencies:
    @@ -12638,7 +12686,6 @@ packages:
           encoding:
             optional: true
         dependencies:
    -      encoding: 0.1.13
           whatwg-url: 5.0.0
         dev: true
     
    @@ -12804,10 +12851,14 @@ packages:
         resolution: {integrity: sha512-s1ZCMmQWXy4b5K/TW9i/DtiN8Ku+xCiHcjQ6/J/nDdssirrQNOoB165Zu8EqLMA2lln1JUth9a0aW9Ap2ctrUg==}
         dependencies:
           destr: 2.0.1
    -      node-fetch-native: 1.4.1
    +      node-fetch-native: 1.4.0
           ufo: 1.3.1
         dev: true
     
    +  /omggif@1.0.10:
    +    resolution: {integrity: sha512-LMJTtvgc/nugXj0Vcrrs68Mn2D1r0zf630VNtqtpI1FEO7e+O9FP4gqs9AcnBaSEeoHIPm28u6qgPR0oyEpGSw==}
    +    dev: true
    +
       /on-exit-leak-free@2.1.0:
         resolution: {integrity: sha512-VuCaZZAjReZ3vUwgOB8LxAosIurDiAW0s13rI1YwmaP++jvcxP77AWoQvenZebpCA2m8WC1/EosPYPMjnRAp/w==}
         dev: false
    @@ -13286,8 +13337,8 @@ packages:
         engines: {node: '>=12.13.0'}
         dev: true
     
    -  /pnpm@8.6.8:
    -    resolution: {integrity: sha512-OtrCRUE7VqRyKly9E3gfExZHOAHtFDpmulyVFKrnl3v/n2xJckdueVumWJqUA5FHxcRP/T42r8P8CDr9duPP9w==}
    +  /pnpm@8.6.12:
    +    resolution: {integrity: sha512-Eza4C5SO/Xl5IYozupbZ5NOA5leBRPYxmXmXfe7G4/4uCkRLhks84rB33aitxNZU/uMrnDGGjwrLktoKvPjqHA==}
         engines: {node: '>=16.14'}
         hasBin: true
         dev: true
    @@ -13299,29 +13350,29 @@ packages:
           trouter: 2.0.1
         dev: true
     
    -  /postcss-import@15.1.0(postcss@8.4.28):
    +  /postcss-import@15.1.0(postcss@8.4.31):
         resolution: {integrity: sha512-hpr+J05B2FVYUAXHeK1YyI267J/dDDhMU6B6civm8hSY1jYJnBXxzKDKDswzJmtLHryrjhnDjqqp/49t8FALew==}
         engines: {node: '>=14.0.0'}
         peerDependencies:
           postcss: ^8.0.0
         dependencies:
    -      postcss: 8.4.28
    +      postcss: 8.4.31
           postcss-value-parser: 4.2.0
           read-cache: 1.0.0
           resolve: 1.22.4
         dev: false
     
    -  /postcss-js@4.0.1(postcss@8.4.28):
    +  /postcss-js@4.0.1(postcss@8.4.31):
         resolution: {integrity: sha512-dDLF8pEO191hJMtlHFPRa8xsizHaM82MLfNkUHdUtVEV3tgTp5oj+8qbEqYM57SLfc74KSbw//4SeJma2LRVIw==}
         engines: {node: ^12 || ^14 || >= 16}
         peerDependencies:
           postcss: ^8.4.21
         dependencies:
           camelcase-css: 2.0.1
    -      postcss: 8.4.28
    +      postcss: 8.4.31
         dev: false
     
    -  /postcss-load-config@4.0.1(postcss@8.4.28)(ts-node@10.9.1):
    +  /postcss-load-config@4.0.1(postcss@8.4.31)(ts-node@10.9.1):
         resolution: {integrity: sha512-vEJIc8RdiBRu3oRAI0ymerOn+7rPuMvRXslTvZUKZonDHFIczxztIyJ1urxM1x9JXEikvpWWTUUqal5j/8QgvA==}
         engines: {node: '>= 14'}
         peerDependencies:
    @@ -13334,18 +13385,18 @@ packages:
             optional: true
         dependencies:
           lilconfig: 2.1.0
    -      postcss: 8.4.28
    -      ts-node: 10.9.1(@types/node@18.16.0)(typescript@5.1.3)
    +      postcss: 8.4.31
    +      ts-node: 10.9.1(@types/node@18.17.5)(typescript@5.1.6)
           yaml: 2.3.1
         dev: false
     
    -  /postcss-nested@6.0.1(postcss@8.4.28):
    +  /postcss-nested@6.0.1(postcss@8.4.31):
         resolution: {integrity: sha512-mEp4xPMi5bSWiMbsgoPfcP74lsWLHkQbZc3sY+jWYd65CUwXrUaTp0fmNpa01ZcETKlIgUdFN/MpS2xZtqL9dQ==}
         engines: {node: '>=12.0'}
         peerDependencies:
           postcss: ^8.2.14
         dependencies:
    -      postcss: 8.4.28
    +      postcss: 8.4.31
           postcss-selector-parser: 6.0.13
         dev: false
     
    @@ -13361,16 +13412,24 @@ packages:
         resolution: {integrity: sha512-1NNCs6uurfkVbeXG4S8JFT9t19m45ICnif8zWLd5oPSZ50QnwMfK+H3jv408d4jw/7Bttv5axS5IiHoLaVNHeQ==}
         dev: false
     
    -  /postcss@8.4.28:
    -    resolution: {integrity: sha512-Z7V5j0cq8oEKyejIKfpD8b4eBy9cwW2JWPk0+fB1HOAMsfHbnAXLLS+PfVWlzMSLQaWttKDt607I0XHmpE67Vw==}
    +  /postcss@8.4.27:
    +    resolution: {integrity: sha512-gY/ACJtJPSmUFPDCHtX78+01fHa64FaU4zaaWfuh1MhGJISufJAH4cun6k/8fwsHYeK4UQmENQK+tRLCFJE8JQ==}
    +    engines: {node: ^10 || ^12 || >=14}
    +    dependencies:
    +      nanoid: 3.3.6
    +      picocolors: 1.0.0
    +      source-map-js: 1.0.2
    +
    +  /postcss@8.4.31:
    +    resolution: {integrity: sha512-PS08Iboia9mts/2ygV3eLpY5ghnUcfLV/EXTOW1E2qYxJKGGBUtNjN76FYHnMs36RmARn41bC0AZmn+rR0OVpQ==}
         engines: {node: ^10 || ^12 || >=14}
         dependencies:
           nanoid: 3.3.6
           picocolors: 1.0.0
           source-map-js: 1.0.2
     
    -  /preact@10.17.1:
    -    resolution: {integrity: sha512-X9BODrvQ4Ekwv9GURm9AKAGaomqXmip7NQTZgY7gcNmr7XE83adOMJvd3N42id1tMFU7ojiynRsYnY6/BRFxLA==}
    +  /preact@10.16.0:
    +    resolution: {integrity: sha512-XTSj3dJ4roKIC93pald6rWuB2qQJO9gO2iLLyTe87MrjQN+HklueLsmskbywEWqCHlclgz3/M4YLL2iBr9UmMA==}
         dev: true
     
       /prelude-ls@1.2.1:
    @@ -13489,11 +13548,6 @@ packages:
         resolution: {integrity: sha512-6Yg0ekpKICSjPswYOuC5sku/TSWaRYlA0qsXqJgM/d/4pLPHPuTxK7Nbf7jFKzAeedUhR8C7K9Uv63FBsSo8xQ==}
         dev: true
     
    -  /q@1.5.1:
    -    resolution: {integrity: sha512-kV/CThkXo6xyFEZUugw/+pIOywXcDbFYgSct5cT3gqlbkBE1SJdwy6UQoZvodiWF/ckQLZyDE/Bu1M6gVu5lVw==}
    -    engines: {node: '>=0.6.0', teleport: '>=0.2.0'}
    -    dev: true
    -
       /qs@6.10.4:
         resolution: {integrity: sha512-OQiU+C+Ds5qiH91qh/mg0w+8nwQuLjM4F4M/PbmhDOoYehPh+Fb0bDjtR1sOvy7YKxvj28Y/M0PhP5uVX0kB+g==}
         engines: {node: '>=0.6'}
    @@ -13947,7 +14001,7 @@ packages:
         peerDependencies:
           rollup: ^2.0.0
         dependencies:
    -      '@babel/code-frame': 7.22.10
    +      '@babel/code-frame': 7.22.13
           jest-worker: 26.6.2
           rollup: 2.79.1
           serialize-javascript: 4.0.0
    @@ -14009,7 +14063,7 @@ packages:
       /rxjs@7.8.1:
         resolution: {integrity: sha512-AA3TVj+0A2iuIoQkWEK/tqFjBq2j+6PO6Y0zJcvzLAFhEFIO3HL0vls9hWLncZbAAbK0mar7oZ4V079I/qPMxg==}
         dependencies:
    -      tslib: 2.6.2
    +      tslib: 2.6.1
         dev: true
     
       /sade@1.8.1:
    @@ -14130,14 +14184,6 @@ packages:
         hasBin: true
         dev: true
     
    -  /semver@7.3.7:
    -    resolution: {integrity: sha512-QlYTucUYOews+WeEujDoEGziz4K6c47V/Bd+LjSSYcA94p+DmINdf7ncaUinThfvZyu13lN9OY1XDxt8C0Tw0g==}
    -    engines: {node: '>=10'}
    -    hasBin: true
    -    dependencies:
    -      lru-cache: 6.0.0
    -    dev: true
    -
       /semver@7.3.8:
         resolution: {integrity: sha512-NB1ctGL5rlHrPJtFDVIVzTyQylMLu9N9VICA6HSFJo8MCGVTMW6gfpicwKmmK/dAjTOrqu5l63JJOpDSrAis3A==}
         engines: {node: '>=10'}
    @@ -14274,8 +14320,8 @@ packages:
           vscode-textmate: 8.0.0
         dev: true
     
    -  /shiki@0.14.4:
    -    resolution: {integrity: sha512-IXCRip2IQzKwxArNNq1S+On4KPML3Yyn8Zzs/xRgcgOWIr8ntIK3IKzjFPfjy/7kt9ZMjc+FItfqHRBg8b6tNQ==}
    +  /shiki@0.14.5:
    +    resolution: {integrity: sha512-1gCAYOcmCFONmErGTrS1fjzJLA7MGZmKzrBNX7apqSwhyITJg2O102uFzXUeBxNnEkDA9vHIKLyeKq0V083vIw==}
         dependencies:
           ansi-sequence-parser: 1.1.1
           jsonc-parser: 3.2.0
    @@ -14572,8 +14618,8 @@ packages:
         engines: {node: '>= 0.8'}
         dev: true
     
    -  /std-env@3.4.0:
    -    resolution: {integrity: sha512-YqHeQIIQ8r1VtUZOTOyjsAXAsjr369SplZ5rlQaiJTBsvodvPSCME7vuz8pnQltbQ0Cw0lyFo5Q8uyNwYQ58Xw==}
    +  /std-env@3.3.3:
    +    resolution: {integrity: sha512-Rz6yejtVyWnVjC1RFvNmYL10kgjC49EOghxWn0RFqlCHGFpQx+Xe7yW3I4ceK1SGrWIGMjD5Kbue8W/udkbMJg==}
         dev: true
     
       /stream-combiner@0.0.4:
    @@ -14803,7 +14849,7 @@ packages:
         engines: {node: ^14.18.0 || >=16.0.0}
         dependencies:
           '@pkgr/utils': 2.4.2
    -      tslib: 2.6.2
    +      tslib: 2.6.1
         dev: true
     
       /tabbable@6.2.0:
    @@ -14823,17 +14869,17 @@ packages:
           fast-glob: 3.3.1
           glob-parent: 6.0.2
           is-glob: 4.0.3
    -      jiti: 1.19.3
    +      jiti: 1.19.1
           lilconfig: 2.1.0
           micromatch: 4.0.5
           normalize-path: 3.0.0
           object-hash: 3.0.0
           picocolors: 1.0.0
    -      postcss: 8.4.28
    -      postcss-import: 15.1.0(postcss@8.4.28)
    -      postcss-js: 4.0.1(postcss@8.4.28)
    -      postcss-load-config: 4.0.1(postcss@8.4.28)(ts-node@10.9.1)
    -      postcss-nested: 6.0.1(postcss@8.4.28)
    +      postcss: 8.4.31
    +      postcss-import: 15.1.0(postcss@8.4.31)
    +      postcss-js: 4.0.1(postcss@8.4.31)
    +      postcss-load-config: 4.0.1(postcss@8.4.31)(ts-node@10.9.1)
    +      postcss-nested: 6.0.1(postcss@8.4.31)
           postcss-selector-parser: 6.0.13
           resolve: 1.22.4
           sucrase: 3.34.0
    @@ -14952,10 +14998,6 @@ packages:
           real-require: 0.2.0
         dev: false
     
    -  /throat@6.0.1:
    -    resolution: {integrity: sha512-8hmiGIJMDlwjg7dlJ4yKGLK8EsYqKgPWbG3b4wjJddKNwc7N7Dpn08Df4szr/sZdMVeOstrdYSsqzX6BYbcB+w==}
    -    dev: true
    -
       /throat@6.0.2:
         resolution: {integrity: sha512-WKexMoJj3vEuK0yFEapj8y64V0A6xcuPuK9Gt1d0R+dzCSJc0lHqQytAbSB4cDAK0dWh4T0E2ETkoLE2WZ41OQ==}
         dev: true
    @@ -15104,13 +15146,13 @@ packages:
           matchit: 1.1.0
         dev: true
     
    -  /ts-api-utils@1.0.3(typescript@5.1.3):
    +  /ts-api-utils@1.0.3(typescript@5.1.6):
         resolution: {integrity: sha512-wNMeqtMz5NtwpT/UZGY5alT+VoKdSsOOP/kqHFcUW1P/VRhH2wJ48+DN2WwUliNbQ976ETwDL0Ifd2VVvgonvg==}
         engines: {node: '>=16.13.0'}
         peerDependencies:
           typescript: '>=4.2.0'
         dependencies:
    -      typescript: 5.1.3
    +      typescript: 5.1.6
         dev: true
     
       /ts-dedent@2.2.0:
    @@ -15122,7 +15164,7 @@ packages:
         resolution: {integrity: sha512-Y/arvbn+rrz3JCKl9C4kVNfTfSm2/mEp5FSz5EsZSANGPSlQrpRI5M4PKF+mJnE52jOO90PnPSc3Ur3bTQw0gA==}
         dev: false
     
    -  /ts-node@10.9.1(@types/node@18.16.0)(typescript@5.1.3):
    +  /ts-node@10.9.1(@types/node@18.17.5)(typescript@5.1.6):
         resolution: {integrity: sha512-NtVysVPkxxrwFGUUxGYhfux8k78pQB3JqYBXlLRZgdGUqTO5wU/UyHop5p70iEbGhB7q5KmiZiU0Y3KlJrScEw==}
         hasBin: true
         peerDependencies:
    @@ -15141,18 +15183,18 @@ packages:
           '@tsconfig/node12': 1.0.11
           '@tsconfig/node14': 1.0.3
           '@tsconfig/node16': 1.0.4
    -      '@types/node': 18.16.0
    +      '@types/node': 18.17.5
           acorn: 8.10.0
           acorn-walk: 8.2.0
           arg: 4.1.3
           create-require: 1.1.1
           diff: 4.0.2
           make-error: 1.3.6
    -      typescript: 5.1.3
    +      typescript: 5.1.6
           v8-compile-cache-lib: 3.0.1
           yn: 3.1.1
     
    -  /ts-node@10.9.1(@types/node@20.4.7)(typescript@5.1.3):
    +  /ts-node@10.9.1(@types/node@20.4.7)(typescript@5.1.6):
         resolution: {integrity: sha512-NtVysVPkxxrwFGUUxGYhfux8k78pQB3JqYBXlLRZgdGUqTO5wU/UyHop5p70iEbGhB7q5KmiZiU0Y3KlJrScEw==}
         hasBin: true
         peerDependencies:
    @@ -15178,7 +15220,7 @@ packages:
           create-require: 1.1.1
           diff: 4.0.2
           make-error: 1.3.6
    -      typescript: 5.1.3
    +      typescript: 5.1.6
           v8-compile-cache-lib: 3.0.1
           yn: 3.1.1
         dev: true
    @@ -15191,18 +15233,28 @@ packages:
         resolution: {integrity: sha512-Xni35NKzjgMrwevysHTCArtLDpPvye8zV/0E4EyYn43P7/7qvQwPh9BGkHewbMulVntbigmcT7rdX3BNo9wRJg==}
         dev: true
     
    -  /tslib@2.6.2:
    -    resolution: {integrity: sha512-AEYxH93jGFPn/a2iVAwW87VuUIkR1FVUKB77NwMF7nBTDkDrrT/Hpt/IrCJ0QXhW27jTBDcf5ZY7w6RiqTMw2Q==}
    +  /tslib@2.6.1:
    +    resolution: {integrity: sha512-t0hLfiEKfMUoqhG+U1oid7Pva4bbDPHYfJNiB7BiIjRkj1pyC++4N3huJfqY6aRH6VTB0rvtzQwjM4K6qpfOig==}
    +    dev: true
    +
    +  /tsutils@3.21.0(typescript@5.0.4):
    +    resolution: {integrity: sha512-mHKK3iUXL+3UF6xL5k0PEhKRUBKPBCv/+RkEOpjRWxxx27KKRBmmA60A9pgOUvMi8GKhRMPEmjBRPzs2W7O1OA==}
    +    engines: {node: '>= 6'}
    +    peerDependencies:
    +      typescript: '>=2.8.0 || >= 3.2.0-dev || >= 3.3.0-dev || >= 3.4.0-dev || >= 3.5.0-dev || >= 3.6.0-dev || >= 3.6.0-beta || >= 3.7.0-dev || >= 3.7.0-beta'
    +    dependencies:
    +      tslib: 1.14.1
    +      typescript: 5.0.4
         dev: true
     
    -  /tsutils@3.21.0(typescript@5.1.3):
    +  /tsutils@3.21.0(typescript@5.1.6):
         resolution: {integrity: sha512-mHKK3iUXL+3UF6xL5k0PEhKRUBKPBCv/+RkEOpjRWxxx27KKRBmmA60A9pgOUvMi8GKhRMPEmjBRPzs2W7O1OA==}
         engines: {node: '>= 6'}
         peerDependencies:
           typescript: '>=2.8.0 || >= 3.2.0-dev || >= 3.3.0-dev || >= 3.4.0-dev || >= 3.5.0-dev || >= 3.6.0-dev || >= 3.6.0-beta || >= 3.7.0-dev || >= 3.7.0-beta'
         dependencies:
           tslib: 1.14.1
    -      typescript: 5.1.3
    +      typescript: 5.1.6
         dev: true
     
       /tunnel-agent@0.6.0:
    @@ -15333,7 +15385,7 @@ packages:
           typedoc: '>=0.24.0'
         dependencies:
           handlebars: 4.7.8
    -      typedoc: 0.25.0(typescript@5.1.3)
    +      typedoc: 0.25.0(typescript@5.0.4)
           typedoc-plugin-mdn-links: 3.0.3(typedoc@0.25.0)
         dev: true
     
    @@ -15342,10 +15394,10 @@ packages:
         peerDependencies:
           typedoc: '>= 0.23.14 || 0.24.x'
         dependencies:
    -      typedoc: 0.25.0(typescript@5.1.3)
    +      typedoc: 0.25.0(typescript@5.0.4)
         dev: true
     
    -  /typedoc@0.25.0(typescript@5.1.3):
    +  /typedoc@0.25.0(typescript@5.0.4):
         resolution: {integrity: sha512-FvCYWhO1n5jACE0C32qg6b3dSfQ8f2VzExnnRboowHtqUD6ARzM2r8YJeZFYXhcm2hI4C2oCRDgNPk/yaQUN9g==}
         engines: {node: '>= 16'}
         hasBin: true
    @@ -15356,11 +15408,17 @@ packages:
           marked: 4.3.0
           minimatch: 9.0.3
           shiki: 0.14.3
    -      typescript: 5.1.3
    +      typescript: 5.0.4
    +    dev: true
    +
    +  /typescript@5.0.4:
    +    resolution: {integrity: sha512-cW9T5W9xY37cc+jfEnaUvX91foxtHkza3Nw3wkoF4sSlKn0MONdkdEndig/qPBWXNkmplh3NzayQzCiHM4/hqw==}
    +    engines: {node: '>=12.20'}
    +    hasBin: true
         dev: true
     
    -  /typescript@5.1.3:
    -    resolution: {integrity: sha512-XH627E9vkeqhlZFQuL+UsyAXEnibT0kWR2FWONlr4sTjvxyJYnyefgrkyECLzM5NenmKzRAy2rR/OlYLA1HkZw==}
    +  /typescript@5.1.6:
    +    resolution: {integrity: sha512-zaWCozRZ6DLEWAWFrVDz1H6FVXzUSfTy5FUMWsQlU8Ym5JP9eO4xkTIROFCQvhQf61z6O/G6ugw3SgAnvvm+HA==}
         engines: {node: '>=14.17'}
         hasBin: true
     
    @@ -15391,13 +15449,13 @@ packages:
           which-boxed-primitive: 1.0.2
         dev: true
     
    -  /unconfig@0.3.10:
    -    resolution: {integrity: sha512-tj317lhIq2iZF/NXrJnU1t2UaGUKKz1eL1sK2t63Oq66V9BxqvZV12m55fp/fpQJ+DDmVlLgo7cnLVOZkhlO/A==}
    +  /unconfig@0.3.11:
    +    resolution: {integrity: sha512-bV/nqePAKv71v3HdVUn6UefbsDKQWRX+bJIkiSm0+twIds6WiD2bJLWWT3i214+J/B4edufZpG2w7Y63Vbwxow==}
         dependencies:
           '@antfu/utils': 0.7.6
           defu: 6.1.2
    -      jiti: 1.19.3
    -      mlly: 1.4.0
    +      jiti: 1.21.0
    +      mlly: 1.4.2
         dev: true
     
       /underscore@1.1.7:
    @@ -15503,11 +15561,11 @@ packages:
         engines: {node: '>= 10.0.0'}
         dev: true
     
    -  /unocss@0.56.5(postcss@8.4.28)(rollup@2.79.1)(vite@4.3.9):
    -    resolution: {integrity: sha512-tO+9St4CntSjHpLXZqBo0/etS06MtvFF1NEny/qFJCL9sCopWwmDKuzW6/LIb4wfqZLdMpVFoEACMNv8nP849A==}
    +  /unocss@0.57.1(postcss@8.4.31)(rollup@2.79.1)(vite@4.4.9):
    +    resolution: {integrity: sha512-xLsyJ8+T1/Ux93yrqOvuQy268wF5rSzydlsbqZ5EVfi01PxYyydez3nycPqbyPZientkJ0Yohzd5aBqmZgku3A==}
         engines: {node: '>=14'}
         peerDependencies:
    -      '@unocss/webpack': 0.56.5
    +      '@unocss/webpack': 0.57.1
           vite: ^2.9.0 || ^3.0.0-0 || ^4.0.0
         peerDependenciesMeta:
           '@unocss/webpack':
    @@ -15515,27 +15573,27 @@ packages:
           vite:
             optional: true
         dependencies:
    -      '@unocss/astro': 0.56.5(rollup@2.79.1)(vite@4.3.9)
    -      '@unocss/cli': 0.56.5(rollup@2.79.1)
    -      '@unocss/core': 0.56.5
    -      '@unocss/extractor-arbitrary-variants': 0.56.5
    -      '@unocss/postcss': 0.56.5(postcss@8.4.28)
    -      '@unocss/preset-attributify': 0.56.5
    -      '@unocss/preset-icons': 0.56.5
    -      '@unocss/preset-mini': 0.56.5
    -      '@unocss/preset-tagify': 0.56.5
    -      '@unocss/preset-typography': 0.56.5
    -      '@unocss/preset-uno': 0.56.5
    -      '@unocss/preset-web-fonts': 0.56.5
    -      '@unocss/preset-wind': 0.56.5
    -      '@unocss/reset': 0.56.5
    -      '@unocss/transformer-attributify-jsx': 0.56.5
    -      '@unocss/transformer-attributify-jsx-babel': 0.56.5
    -      '@unocss/transformer-compile-class': 0.56.5
    -      '@unocss/transformer-directives': 0.56.5
    -      '@unocss/transformer-variant-group': 0.56.5
    -      '@unocss/vite': 0.56.5(rollup@2.79.1)(vite@4.3.9)
    -      vite: 4.3.9(@types/node@18.16.0)
    +      '@unocss/astro': 0.57.1(rollup@2.79.1)(vite@4.4.9)
    +      '@unocss/cli': 0.57.1(rollup@2.79.1)
    +      '@unocss/core': 0.57.1
    +      '@unocss/extractor-arbitrary-variants': 0.57.1
    +      '@unocss/postcss': 0.57.1(postcss@8.4.31)
    +      '@unocss/preset-attributify': 0.57.1
    +      '@unocss/preset-icons': 0.57.1
    +      '@unocss/preset-mini': 0.57.1
    +      '@unocss/preset-tagify': 0.57.1
    +      '@unocss/preset-typography': 0.57.1
    +      '@unocss/preset-uno': 0.57.1
    +      '@unocss/preset-web-fonts': 0.57.1
    +      '@unocss/preset-wind': 0.57.1
    +      '@unocss/reset': 0.57.1
    +      '@unocss/transformer-attributify-jsx': 0.57.1
    +      '@unocss/transformer-attributify-jsx-babel': 0.57.1
    +      '@unocss/transformer-compile-class': 0.57.1
    +      '@unocss/transformer-directives': 0.57.1
    +      '@unocss/transformer-variant-group': 0.57.1
    +      '@unocss/vite': 0.57.1(rollup@2.79.1)(vite@4.4.9)
    +      vite: 4.4.9(@types/node@18.17.5)
         transitivePeerDependencies:
           - postcss
           - rollup
    @@ -15547,7 +15605,7 @@ packages:
         engines: {node: '>= 0.8'}
         dev: true
     
    -  /unplugin-vue-components@0.25.0(rollup@2.79.1)(vue@3.3.0):
    +  /unplugin-vue-components@0.25.0(rollup@2.79.1)(vue@3.3.4):
         resolution: {integrity: sha512-HxrQ4GMSS1RwVww2av3a42cABo/v5AmTRN9iARv6e/xwkrfTyHhLh84kFwXxKkXK61vxDHxaryn694mQmkiVBg==}
         engines: {node: '>=14'}
         peerDependencies:
    @@ -15560,17 +15618,17 @@ packages:
           '@nuxt/kit':
             optional: true
         dependencies:
    -      '@antfu/utils': 0.7.6
    +      '@antfu/utils': 0.7.5
           '@rollup/pluginutils': 5.0.3(rollup@2.79.1)
           chokidar: 3.5.3
           debug: 4.3.4(supports-color@8.1.1)
    -      fast-glob: 3.3.1
    +      fast-glob: 3.2.12
           local-pkg: 0.4.3
           magic-string: 0.30.2
           minimatch: 9.0.3
           resolve: 1.22.4
           unplugin: 1.4.0
    -      vue: 3.3.0
    +      vue: 3.3.4
         transitivePeerDependencies:
           - rollup
           - supports-color
    @@ -15696,7 +15754,7 @@ packages:
           vfile-message: 3.1.4
         dev: true
     
    -  /vite-node@0.34.0(@types/node@18.16.0):
    +  /vite-node@0.34.0(@types/node@18.17.5):
         resolution: {integrity: sha512-rGZMvpb052rjUwJA/a17xMfOibzNF7byMdRSTcN2Lw8uxX08s5EfjWW5mBkm3MSFTPctMSVtT2yC+8ShrZbT5g==}
         engines: {node: '>=v14.18.0'}
         hasBin: true
    @@ -15706,7 +15764,7 @@ packages:
           mlly: 1.4.0
           pathe: 1.1.1
           picocolors: 1.0.0
    -      vite: 4.4.9(@types/node@18.16.0)
    +      vite: 4.5.0(@types/node@18.17.5)
         transitivePeerDependencies:
           - '@types/node'
           - less
    @@ -15718,7 +15776,7 @@ packages:
           - terser
         dev: true
     
    -  /vite-plugin-istanbul@4.1.0(vite@4.3.9):
    +  /vite-plugin-istanbul@4.1.0(vite@4.4.9):
         resolution: {integrity: sha512-d8FRxaswOUYlGqCCNv2BTbt9pyqt7J4RPgab3WmMf+T2TflLlCmC7S26zDRfL9Ve4JSHrcf5bdzt+E0n9CrPvA==}
         peerDependencies:
           vite: '>=2.9.1 <= 5'
    @@ -15727,12 +15785,12 @@ packages:
           istanbul-lib-instrument: 5.2.1
           picocolors: 1.0.0
           test-exclude: 6.0.0
    -      vite: 4.3.9(@types/node@18.16.0)
    +      vite: 4.4.9(@types/node@18.17.5)
         transitivePeerDependencies:
           - supports-color
         dev: true
     
    -  /vite-plugin-pwa@0.16.0(vite@4.3.9)(workbox-build@7.0.0)(workbox-window@7.0.0):
    +  /vite-plugin-pwa@0.16.0(vite@4.4.9)(workbox-build@7.0.0)(workbox-window@7.0.0):
         resolution: {integrity: sha512-E+AQRzHxqNU4ZhEeR8X37/foZB+ezJEhXauE/mcf1UITY6k2Pa1dtlFl+BQu57fTdiVlWim5S0Qy44Yap93Dkg==}
         engines: {node: '>=16.0.0'}
         peerDependencies:
    @@ -15741,22 +15799,23 @@ packages:
           workbox-window: ^7.0.0
         dependencies:
           debug: 4.3.4(supports-color@8.1.1)
    -      fast-glob: 3.3.1
    +      fast-glob: 3.2.12
           pretty-bytes: 6.1.1
    -      vite: 4.3.9(@types/node@18.16.0)
    +      vite: 4.4.9(@types/node@18.17.5)
           workbox-build: 7.0.0
           workbox-window: 7.0.0
         transitivePeerDependencies:
           - supports-color
         dev: true
     
    -  /vite@4.3.9(@types/node@18.16.0):
    -    resolution: {integrity: sha512-qsTNZjO9NoJNW7KnOrgYwczm0WctJ8m/yqYAMAK9Lxt4SoySUfS5S8ia9K7JHpa3KEeMfyF8LoJ3c5NeBJy6pg==}
    +  /vite@4.4.9(@types/node@18.17.5):
    +    resolution: {integrity: sha512-2mbUn2LlUmNASWwSCNSJ/EG2HuSRTnVNaydp6vMCm5VIqJsjMfbIWtbH2kDuwUVW5mMUKKZvGPX/rqeqVvv1XA==}
         engines: {node: ^14.18.0 || >=16.0.0}
         hasBin: true
         peerDependencies:
           '@types/node': '>= 14'
           less: '*'
    +      lightningcss: ^1.21.0
           sass: '*'
           stylus: '*'
           sugarss: '*'
    @@ -15766,6 +15825,8 @@ packages:
             optional: true
           less:
             optional: true
    +      lightningcss:
    +        optional: true
           sass:
             optional: true
           stylus:
    @@ -15775,16 +15836,16 @@ packages:
           terser:
             optional: true
         dependencies:
    -      '@types/node': 18.16.0
    -      esbuild: 0.17.19
    -      postcss: 8.4.28
    +      '@types/node': 18.17.5
    +      esbuild: 0.18.20
    +      postcss: 8.4.27
           rollup: 3.28.0
         optionalDependencies:
           fsevents: 2.3.2
         dev: true
     
    -  /vite@4.4.9(@types/node@18.16.0):
    -    resolution: {integrity: sha512-2mbUn2LlUmNASWwSCNSJ/EG2HuSRTnVNaydp6vMCm5VIqJsjMfbIWtbH2kDuwUVW5mMUKKZvGPX/rqeqVvv1XA==}
    +  /vite@4.5.0(@types/node@18.17.5):
    +    resolution: {integrity: sha512-ulr8rNLA6rkyFAlVWw2q5YJ91v098AFQ2R0PRFwPzREXOUJQPtFUG0t+/ZikhaOCDqFoDhN6/v8Sq0o4araFAw==}
         engines: {node: ^14.18.0 || >=16.0.0}
         hasBin: true
         peerDependencies:
    @@ -15811,15 +15872,15 @@ packages:
           terser:
             optional: true
         dependencies:
    -      '@types/node': 18.16.0
    +      '@types/node': 18.17.5
           esbuild: 0.18.20
    -      postcss: 8.4.28
    +      postcss: 8.4.31
           rollup: 3.28.0
         optionalDependencies:
           fsevents: 2.3.2
         dev: true
     
    -  /vitepress-plugin-search@1.0.4-alpha.20(flexsearch@0.7.31)(vitepress@1.0.0-rc.10)(vue@3.3.4):
    +  /vitepress-plugin-search@1.0.4-alpha.20(flexsearch@0.7.31)(vitepress@1.0.0-alpha.72)(vue@3.3.7):
         resolution: {integrity: sha512-zG+ev9pw1Mg7htABlFCNXb8XwnKN+qfTKw+vU0Ers6RIrABx+45EAAFBoaL1mEpl1FRFn1o/dQ7F4b8GP6HdGQ==}
         engines: {node: ^14.13.1 || ^16.7.0 || >=18}
         peerDependencies:
    @@ -15832,58 +15893,47 @@ packages:
           flexsearch: 0.7.31
           glob-to-regexp: 0.4.1
           markdown-it: 13.0.1
    -      vitepress: 1.0.0-rc.10(@algolia/client-search@4.19.1)(@types/node@18.16.0)(search-insights@2.7.0)
    -      vue: 3.3.4
    +      vitepress: 1.0.0-alpha.72(@algolia/client-search@4.19.1)(@types/node@18.17.5)(search-insights@2.7.0)
    +      vue: 3.3.7(typescript@5.0.4)
         dev: true
     
    -  /vitepress@1.0.0-rc.10(@algolia/client-search@4.19.1)(@types/node@18.16.0)(search-insights@2.7.0):
    -    resolution: {integrity: sha512-+MsahIWqq5WUEmj6MR4obcKYbT7im07jZPCQPdNJExkeOSbOAJ4xypSLx88x7rvtzWHhHc5aXbOhCRvGEGjFrw==}
    +  /vitepress@1.0.0-alpha.72(@algolia/client-search@4.19.1)(@types/node@18.17.5)(search-insights@2.7.0):
    +    resolution: {integrity: sha512-Ou7fNE/OVYLrKGQMHSTVG6AcNsdv7tm4ACrdhx93SPMzEDj8UgIb4RFa5CTTowaYf3jeDGi2EAJlzXVC+IE3dg==}
         hasBin: true
         dependencies:
    -      '@docsearch/css': 3.5.2
    -      '@docsearch/js': 3.5.2(@algolia/client-search@4.19.1)(search-insights@2.7.0)
    +      '@docsearch/css': 3.5.1
    +      '@docsearch/js': 3.5.1(@algolia/client-search@4.19.1)(search-insights@2.7.0)
    +      '@vitejs/plugin-vue': 4.2.3(vite@4.4.9)(vue@3.3.4)
           '@vue/devtools-api': 6.5.0
    -      '@vueuse/core': 10.4.1(vue@3.3.4)
    -      '@vueuse/integrations': 10.4.1(focus-trap@7.5.2)(vue@3.3.4)
    -      focus-trap: 7.5.2
    +      '@vueuse/core': 10.3.0(vue@3.3.4)
    +      body-scroll-lock: 4.0.0-beta.0
           mark.js: 8.11.1
           minisearch: 6.1.0
    -      shiki: 0.14.4
    -      vite: 4.4.9(@types/node@18.16.0)
    +      shiki: 0.14.3
    +      vite: 4.4.9(@types/node@18.17.5)
           vue: 3.3.4
         transitivePeerDependencies:
           - '@algolia/client-search'
           - '@types/node'
           - '@types/react'
           - '@vue/composition-api'
    -      - async-validator
    -      - axios
    -      - change-case
    -      - drauu
    -      - fuse.js
    -      - idb-keyval
    -      - jwt-decode
           - less
           - lightningcss
    -      - nprogress
    -      - qrcode
           - react
           - react-dom
           - sass
           - search-insights
    -      - sortablejs
           - stylus
           - sugarss
           - terser
    -      - universal-cookie
         dev: true
     
    -  /vitepress@1.0.0-rc.20(@algolia/client-search@4.19.1)(@types/node@18.16.0)(postcss@8.4.28)(search-insights@2.7.0):
    -    resolution: {integrity: sha512-CykMUJ8JLxLcGWek0ew3wln4RYbsOd1+0YzXITTpajggpynm2S331TNkJVOkHrMRc6GYe3y4pS40GfgcW0ZwAw==}
    +  /vitepress@1.0.0-rc.25(@algolia/client-search@4.19.1)(@types/node@18.17.5)(postcss@8.4.31)(search-insights@2.7.0)(typescript@5.1.6):
    +    resolution: {integrity: sha512-1dqWiHNThNrVZ08ixmfEDBEH+764KOgnev9oXga/x6cN++Vb9pnuu8p3K6DQP+KZrYcG+WiX7jxal0iSNpAWuQ==}
         hasBin: true
         peerDependencies:
           markdown-it-mathjax3: ^4.3.2
    -      postcss: ^8.4.30
    +      postcss: ^8.4.31
         peerDependenciesMeta:
           markdown-it-mathjax3:
             optional: true
    @@ -15893,16 +15943,17 @@ packages:
           '@docsearch/css': 3.5.2
           '@docsearch/js': 3.5.2(@algolia/client-search@4.19.1)(search-insights@2.7.0)
           '@types/markdown-it': 13.0.5
    -      '@vue/devtools-api': 6.5.0
    -      '@vueuse/core': 10.4.1(vue@3.3.4)
    -      '@vueuse/integrations': 10.4.1(focus-trap@7.5.2)(vue@3.3.4)
    -      focus-trap: 7.5.2
    +      '@vitejs/plugin-vue': 4.3.1(vite@4.5.0)(vue@3.3.7)
    +      '@vue/devtools-api': 6.5.1
    +      '@vueuse/core': 10.5.0(vue@3.3.7)
    +      '@vueuse/integrations': 10.5.0(focus-trap@7.5.4)(vue@3.3.7)
    +      focus-trap: 7.5.4
           mark.js: 8.11.1
           minisearch: 6.1.0
    -      postcss: 8.4.28
    -      shiki: 0.14.4
    -      vite: 4.4.9(@types/node@18.16.0)
    -      vue: 3.3.4
    +      postcss: 8.4.31
    +      shiki: 0.14.5
    +      vite: 4.5.0(@types/node@18.17.5)
    +      vue: 3.3.7(typescript@5.1.6)
         transitivePeerDependencies:
           - '@algolia/client-search'
           - '@types/node'
    @@ -15927,6 +15978,7 @@ packages:
           - stylus
           - sugarss
           - terser
    +      - typescript
           - universal-cookie
         dev: true
     
    @@ -15963,7 +16015,7 @@ packages:
         dependencies:
           '@types/chai': 4.3.5
           '@types/chai-subset': 1.3.3
    -      '@types/node': 18.16.0
    +      '@types/node': 18.17.5
           '@vitest/expect': 0.34.0
           '@vitest/runner': 0.34.0
           '@vitest/snapshot': 0.34.0
    @@ -15980,12 +16032,12 @@ packages:
           magic-string: 0.30.2
           pathe: 1.1.1
           picocolors: 1.0.0
    -      std-env: 3.4.0
    +      std-env: 3.3.3
           strip-literal: 1.3.0
           tinybench: 2.5.0
           tinypool: 0.7.0
    -      vite: 4.4.9(@types/node@18.16.0)
    -      vite-node: 0.34.0(@types/node@18.16.0)
    +      vite: 4.4.9(@types/node@18.17.5)
    +      vite-node: 0.34.0(@types/node@18.17.5)
           why-is-node-running: 2.2.2
         transitivePeerDependencies:
           - less
    @@ -16001,10 +16053,10 @@ packages:
         resolution: {integrity: sha512-xGmv9QIWs2H8obGbWg+sIPI/3/pFgj/5OWBhNzs00BkYQ9UaB2F6JJaGB/2/YOZJ3BvLXQTC4Q7muqU25QgAhA==}
         dependencies:
           jsonc-parser: 3.2.0
    -      vscode-languageserver-textdocument: 1.0.11
    -      vscode-languageserver-types: 3.17.5
    +      vscode-languageserver-textdocument: 1.0.8
    +      vscode-languageserver-types: 3.17.3
           vscode-nls: 5.2.0
    -      vscode-uri: 3.0.8
    +      vscode-uri: 3.0.7
         dev: true
     
       /vscode-jsonrpc@8.2.0:
    @@ -16024,6 +16076,10 @@ packages:
         resolution: {integrity: sha512-1bonkGqQs5/fxGT5UchTgjGVnfysL0O8v1AYMBjqTbWQTFn721zaPGDYFkOKtfDgFiSgXM3KwaG3FMGfW4Ed9Q==}
         dev: true
     
    +  /vscode-languageserver-types@3.17.3:
    +    resolution: {integrity: sha512-SYU4z1dL0PyIMd4Vj8YOqFvHu7Hz/enbWtpfnVbJHU4Nd1YNYx8u0ennumc6h48GQNeOLxmwySmnADouT/AuZA==}
    +    dev: true
    +
       /vscode-languageserver-types@3.17.5:
         resolution: {integrity: sha512-Ld1VelNuX9pdF39h2Hgaeb5hEZM2Z3jUrrMgWQAu82jMtZp7p3vJT3BzToKtZI7NgQssZje5o0zryOrhQvzQAg==}
     
    @@ -16052,7 +16108,7 @@ packages:
       /vscode-uri@3.0.8:
         resolution: {integrity: sha512-AyFQ0EVmsOZOlAnxoFOGOq1SQDWAB7C6aqMGS23svWAllfOaxbuFvcT8D1i8z3Gyn8fraVeZNNmN6e9bxxXkKw==}
     
    -  /vue-demi@0.14.5(vue@3.3.0):
    +  /vue-demi@0.14.5(vue@3.3.4):
         resolution: {integrity: sha512-o9NUVpl/YlsGJ7t+xuqJKx8EBGf1quRhCiT6D/J0pfwmk9zUwYkC7yrF4SZCe6fETvSM3UNL2edcbYrSyc4QHA==}
         engines: {node: '>=12'}
         hasBin: true
    @@ -16064,11 +16120,11 @@ packages:
           '@vue/composition-api':
             optional: true
         dependencies:
    -      vue: 3.3.0
    +      vue: 3.3.4
         dev: false
     
    -  /vue-demi@0.14.5(vue@3.3.4):
    -    resolution: {integrity: sha512-o9NUVpl/YlsGJ7t+xuqJKx8EBGf1quRhCiT6D/J0pfwmk9zUwYkC7yrF4SZCe6fETvSM3UNL2edcbYrSyc4QHA==}
    +  /vue-demi@0.14.6(vue@3.3.4):
    +    resolution: {integrity: sha512-8QA7wrYSHKaYgUxDA5ZC24w+eHm3sYCbp0EzcDwKqN3p6HqtTCGR/GVsPyZW92unff4UlcSh++lmqDWN3ZIq4w==}
         engines: {node: '>=12'}
         hasBin: true
         requiresBuild: true
    @@ -16082,14 +16138,20 @@ packages:
           vue: 3.3.4
         dev: true
     
    -  /vue@3.3.0:
    -    resolution: {integrity: sha512-cyyuVeFKvQy5eGIwN7VQlNKFu09DQSyTtunzpURRjPJwl6B2T7zo41oE1Nr/nacCsZVpnkE6FlWN0YfbY2SB2w==}
    +  /vue-demi@0.14.6(vue@3.3.7):
    +    resolution: {integrity: sha512-8QA7wrYSHKaYgUxDA5ZC24w+eHm3sYCbp0EzcDwKqN3p6HqtTCGR/GVsPyZW92unff4UlcSh++lmqDWN3ZIq4w==}
    +    engines: {node: '>=12'}
    +    hasBin: true
    +    requiresBuild: true
    +    peerDependencies:
    +      '@vue/composition-api': ^1.0.0-rc.1
    +      vue: ^3.0.0-0 || ^2.6.0
    +    peerDependenciesMeta:
    +      '@vue/composition-api':
    +        optional: true
         dependencies:
    -      '@vue/compiler-dom': 3.3.0
    -      '@vue/compiler-sfc': 3.3.0
    -      '@vue/runtime-dom': 3.3.0
    -      '@vue/server-renderer': 3.3.0(vue@3.3.0)
    -      '@vue/shared': 3.3.0
    +      vue: 3.3.7(typescript@5.1.6)
    +    dev: true
     
       /vue@3.3.4:
         resolution: {integrity: sha512-VTyEYn3yvIeY1Py0WaYGZsXnz3y5UnGi62GjVEqvEGPl6nxbOrCXbVOTQWBEJUqAyTUk2uJ5JLVnYJ6ZzGbrSw==}
    @@ -16100,12 +16162,44 @@ packages:
           '@vue/server-renderer': 3.3.4(vue@3.3.4)
           '@vue/shared': 3.3.4
     
    +  /vue@3.3.7(typescript@5.0.4):
    +    resolution: {integrity: sha512-YEMDia1ZTv1TeBbnu6VybatmSteGOS3A3YgfINOfraCbf85wdKHzscD6HSS/vB4GAtI7sa1XPX7HcQaJ1l24zA==}
    +    peerDependencies:
    +      typescript: '*'
    +    peerDependenciesMeta:
    +      typescript:
    +        optional: true
    +    dependencies:
    +      '@vue/compiler-dom': 3.3.7
    +      '@vue/compiler-sfc': 3.3.7
    +      '@vue/runtime-dom': 3.3.7
    +      '@vue/server-renderer': 3.3.7(vue@3.3.7)
    +      '@vue/shared': 3.3.7
    +      typescript: 5.0.4
    +    dev: true
    +
    +  /vue@3.3.7(typescript@5.1.6):
    +    resolution: {integrity: sha512-YEMDia1ZTv1TeBbnu6VybatmSteGOS3A3YgfINOfraCbf85wdKHzscD6HSS/vB4GAtI7sa1XPX7HcQaJ1l24zA==}
    +    peerDependencies:
    +      typescript: '*'
    +    peerDependenciesMeta:
    +      typescript:
    +        optional: true
    +    dependencies:
    +      '@vue/compiler-dom': 3.3.7
    +      '@vue/compiler-sfc': 3.3.7
    +      '@vue/runtime-dom': 3.3.7
    +      '@vue/server-renderer': 3.3.7(vue@3.3.7)
    +      '@vue/shared': 3.3.7
    +      typescript: 5.1.6
    +    dev: true
    +
       /vuex@4.1.0(vue@3.3.4):
         resolution: {integrity: sha512-hmV6UerDrPcgbSy9ORAtNXDr9M4wlNP4pEFKye4ujJF8oqgFFuxDCdOLS3eNoRTtq5O3hoBDh9Doj1bQMYHRbQ==}
         peerDependencies:
           vue: ^3.2.0
         dependencies:
    -      '@vue/devtools-api': 6.5.0
    +      '@vue/devtools-api': 6.5.1
           vue: 3.3.4
         dev: false
     
    @@ -16169,16 +16263,16 @@ packages:
         engines: {node: '>= 8'}
         dev: true
     
    -  /webdriver@7.30.0(typescript@5.1.3):
    +  /webdriver@7.30.0(typescript@5.1.6):
         resolution: {integrity: sha512-bQE4oVgjjg5sb3VkCD+Eb8mscEvf3TioP0mnEZK0f5OJUNI045gMCJgpX8X4J8ScGyEhzlhn1KvlAn3yzxjxog==}
         engines: {node: '>=12.0.0'}
         dependencies:
    -      '@types/node': 18.17.6
    -      '@wdio/config': 7.30.0(typescript@5.1.3)
    +      '@types/node': 18.17.5
    +      '@wdio/config': 7.30.0(typescript@5.1.6)
           '@wdio/logger': 7.26.0
           '@wdio/protocols': 7.27.0
    -      '@wdio/types': 7.26.0(typescript@5.1.3)
    -      '@wdio/utils': 7.26.0(typescript@5.1.3)
    +      '@wdio/types': 7.26.0(typescript@5.1.6)
    +      '@wdio/utils': 7.26.0(typescript@5.1.6)
           got: 11.8.6
           ky: 0.30.0
           lodash.merge: 4.6.2
    @@ -16678,19 +16772,6 @@ packages:
           signal-exit: 3.0.7
         dev: true
     
    -  /ws@8.12.0:
    -    resolution: {integrity: sha512-kU62emKIdKVeEIOIKVegvqpXMSTAMLJozpHZaJNDYqBjzlSYXQGviYwN1osDLJ9av68qHd4a2oSjd7yD4pacig==}
    -    engines: {node: '>=10.0.0'}
    -    peerDependencies:
    -      bufferutil: ^4.0.1
    -      utf-8-validate: '>=5.0.2'
    -    peerDependenciesMeta:
    -      bufferutil:
    -        optional: true
    -      utf-8-validate:
    -        optional: true
    -    dev: true
    -
       /ws@8.13.0:
         resolution: {integrity: sha512-x9vcZYTrFPC7aSIbj7sRCYo7L/Xb8Iy+pW0ng0wt2vCJv7M9HOMy0UoN3rr+IFC7hb7vXoqS+P9ktyLLLhO+LA==}
         engines: {node: '>=10.0.0'}
    diff --git a/tests/webpack/src/index.js b/tests/webpack/src/index.js
    index 51738aa62c..e667cfc5d0 100644
    --- a/tests/webpack/src/index.js
    +++ b/tests/webpack/src/index.js
    @@ -22,7 +22,7 @@ const load = async () => {
           Popularisation
             British popular psychology author Tony Buzan
         Research
    -      On effectivness
    and features + On effectiveness
    and features On Automatic creation Uses Creative techniques