From f7aacf98fddfe6fbd4fd19ffd57399dfe195c7ea Mon Sep 17 00:00:00 2001 From: Boris Mann Date: Thu, 19 Oct 2023 16:55:04 -0700 Subject: [PATCH] Quartz branch --- .github/workflows/publish.yml | 68 +- .gitignore | 12 +- .npmrc | 1 + .prettierignore | 3 + .prettierrc | 7 + CODE_OF_CONDUCT.md | 90 + Dockerfile | 11 + LICENSE.txt | 21 + README.md | 23 +- content/about.md | 5 + ...s_For_Thought_Jan_2023_1675102979342_0.edn | 0 ...s_For_Thought_Jan_2023_1675102979342_0.pdf | Bin ...ht_Jan_2023_1675102979342_0__136407078.edn | 0 ...TfTR-References-BibTeX_1675103690419_0.bib | 0 .../assets}/image_1671407165277_0.png | Bin content/index.md | 14 + .../journals/2021-06-15.md | 0 .../journals/2021-07-27.md | 0 .../journals/2021-08-16.md | 0 .../journals/2021-10-25.md | 0 .../journals/2021-10-29.md | 0 .../journals/2021-12-05.md | 0 .../journals/2022-02-23.md | 0 .../journals/2022-08-16.md | 0 .../journals/2022-11-13.md | 0 .../journals/2022-11-16.md | 0 .../journals/2022-11-17.md | 0 .../journals/2022-11-19.md | 0 .../journals/2022-11-20.md | 0 .../journals/2022-11-21.md | 0 .../journals/2022-11-23.md | 0 .../journals/2022-11-30.md | 0 .../journals/2023-01-26.md | 0 content/journals/2023-10-19.md | 3 + .../A Big List of Mastodon Resources.md | 6 +- .../notes}/Add your person page.md | 0 {pages => content/notes}/Ahmed Fasih.md | 0 {pages => content/notes}/Alan Chan.md | 0 .../notes}/Alexander Obernauer.md | 0 content/notes/Announcements.md | 1 + .../notes}/April 2022 Meeting.md | 0 .../notes}/August 2021 Meeting.md | 0 {pages => content/notes}/Awesome List.md | 8 +- {pages => content/notes}/Bill Seitz.md | 0 {pages => content/notes}/Boris Mann.md | 0 content/notes/Butter.md | 1 + content/notes/Calendar.md | 3 + content/notes/Charles Chamberlain.md | 5 + {pages => content/notes}/Christina Bowen.md | 0 content/notes/Colophon.md | 9 + content/notes/Community Infrastructure.md | 51 + {pages => content/notes}/Dalton Banks.md | 0 {pages => content/notes}/David Brooks.md | 0 .../notes/Early Tools for Thought Followup.md | 3 + .../Early Tools for Thought%2FChat Log.md | 0 .../Early Tools for Thought%2FReferences.md | 0 .../notes}/Early Tools for Thought.md | 0 .../notes}/Eastgate Systems, Inc..md | 0 content/notes/Events.md | 1 + .../Everything I know about Mastodon.md | 13 +- .../notes}/Export considered harmful.md | 0 content/notes/February 2022 Meeting.md | 16 + .../notes}/Federated Hosting by Spacebear.md | 0 {pages => content/notes}/Fedi Tips.md | 0 {pages => content/notes}/Fedifinder.md | 0 .../notes}/Fediverse Party - Mastodon.md | 0 {pages => content/notes}/Fission Discord.md | 0 ...n Open Servers English 5K+ Active Users.md | 0 {pages => content/notes}/Geoffrey Litt.md | 0 {pages => content/notes}/GithubIssue.md | 0 {pages => content/notes}/Gordon Brander.md | 0 ...Rocks Mastodon server be relied upon%3F.md | 0 .../How to Make the Fediverse Your Own.md | 0 ... add content to the TFTRocks Notes site.md | 0 {pages => content/notes}/Howard Rheingold.md | 0 {pages => content/notes}/Hunter Clarke.md | 0 {pages => content/notes}/Hyperdraft.md | 0 {pages => content/notes}/Instances.Social.md | 0 {pages => content/notes}/Isomorphic Git.md | 0 content/notes/Jacky Zhao.md | 1 + {pages => content/notes}/Jakob.md | 0 {pages => content/notes}/James Walker.md | 0 .../notes}/January 2022 Meeting.md | 0 {pages => content/notes}/Jared Pereira.md | 0 {pages => content/notes}/Jess Martin.md | 0 {pages => content/notes}/Joybox.md | 0 {pages => content/notes}/July 2021 Meeting.md | 0 .../notes}/July Meeting Chat Log.md | 0 .../notes}/July Meeting Recap.md | 0 {pages => content/notes}/Kevin Lin.md | 0 {pages => content/notes}/Kosmik.md | 0 {pages => content/notes}/Linus Lee.md | 0 {pages => content/notes}/LogSeq Publish.md | 0 .../notes/LogSeq Schr\303\266dinger.md" | 0 {pages => content/notes}/Lucerne.md | 0 content/notes/Luma.md | 3 + {pages => content/notes}/Maggie Appleton.md | 0 .../notes}/March 2022 Meeting.md | 0 {pages => content/notes}/Mark Anderson.md | 0 {pages => content/notes}/Mark Bernstein.md | 0 {pages => content/notes}/Mark Upton.md | 0 .../notes}/Mastodon Resources.md | 0 {pages => content/notes}/Mastodon.md | 0 {pages => content/notes}/May 2022 Meeting.md | 0 {pages => content/notes}/Michael Gartner.md | 0 {pages => content/notes}/Moa Party.md | 0 {pages => content/notes}/Monthly Meetings.md | 0 .../notes}/November 2021 Chat Log.md | 0 .../notes}/November 2021 Meeting.md | 0 .../notes}/October 2021 Chat Log.md | 0 content/notes/October 2021 Meeting.md | 36 + .../notes/Open Collective.md | 0 {pages => content/notes}/Paul Rony.md | 0 {pages => content/notes}/Paul Shen.md | 0 {pages => content/notes}/People.md | 0 {pages => content/notes}/Pol Baladas.md | 0 content/notes/Quartz.md | 7 + {pages => content/notes}/Rafael Nepo.md | 0 {pages => content/notes}/Robert Haisfield.md | 0 {pages => content/notes}/Rosano.md | 0 {pages => content/notes}/Ryan Atkinson.md | 0 ...Synthesis - Extending Tools for Thought.md | 0 .../Scaling Synthesis - Multiplayer TFT.md | 0 .../notes}/September 2021 Meeting.md | 0 {pages => content/notes}/Sophia Xu.md | 0 {pages => content/notes}/Spacebear.md | 0 .../notes}/Spatial Canvases - Part 1.md | 0 .../notes}/Spatial Canvases - Part 2.md | 0 .../notes}/Spatial Canvases - Part 3.md | 0 .../notes}/Spatial Canvases - Part 4.md | 0 {pages => content/notes}/Speaker Listing.md | 0 {pages => content/notes}/Sprout.md | 0 {pages => content/notes}/Taylor Hadden.md | 0 {pages => content/notes}/Templates.md | 0 {pages => content/notes}/Tinderbox.md | 0 {pages => content/notes}/Tony Confrey.md | 0 content/notes/Tools For Thought Mastodon.md | 8 + ...Tools For Thought Rocks Open Collective.md | 0 .../notes/Tools for THINKing \342\200\246.md" | 0 .../notes}/Tools for Thought App List.md | 0 .../notes}/Tools for Thought Rocks.md | 0 ...ts, Maggie Appleton, April 2022 Meeting.md | 0 {pages => content/notes}/Tools for Thought.md | 0 {pages => content/notes}/Tudor Girba.md | 0 {pages => content/notes}/Utilities.md | 0 .../Volunteer Organizer Kick Off Meeting.md | 0 {pages => content/notes}/Weiwei Hsu.md | 0 .../notes}/Wetware of Writing and Doing.md | 0 {pages => content/notes}/Write of Passage.md | 0 {pages => content/notes}/monthly-meeting.md | 0 pages/Yoyogi.md => content/notes/yoyogi.md | 0 docs/advanced/architecture.md | 52 + docs/advanced/creating components.md | 233 + docs/advanced/index.md | 3 + docs/advanced/making plugins.md | 302 + docs/advanced/paths.md | 51 + docs/authoring content.md | 48 + docs/build.md | 23 + docs/configuration.md | 82 + docs/features/Docker Support.md | 7 + docs/features/Latex.md | 63 + docs/features/Mermaid diagrams.md | 28 + docs/features/Obsidian compatibility.md | 31 + docs/features/OxHugo compatibility.md | 39 + docs/features/RSS Feed.md | 7 + docs/features/SPA Routing.md | 7 + docs/features/backlinks.md | 14 + docs/features/breadcrumbs.md | 35 + docs/features/callouts.md | 86 + docs/features/darkmode.md | 14 + docs/features/explorer.md | 245 + docs/features/folder and tag listings.md | 32 + docs/features/full-text search.md | 30 + docs/features/graph view.md | 59 + docs/features/index.md | 3 + docs/features/popover previews.md | 15 + docs/features/private pages.md | 27 + docs/features/recent notes.md | 16 + docs/features/syntax highlighting.md | 135 + docs/features/table of contents.md | 25 + docs/features/upcoming features.md | 23 + docs/features/wikilinks.md | 16 + docs/hosting.md | 168 + docs/images/dns records.png | Bin 0 -> 77842 bytes docs/images/quartz layout.png | Bin 0 -> 56729 bytes docs/images/quartz transform pipeline.png | Bin 0 -> 73574 bytes docs/index.md | 45 + docs/layout.md | 42 + docs/migrating from Quartz 3.md | 41 + docs/philosophy.md | 28 + docs/showcase.md | 24 + docs/tags/component.md | 5 + docs/upgrading.md | 19 + globals.d.ts | 12 + index.d.ts | 11 + .../2022-06-29T23_03_23.581Z.md | 6 - logseq/config.edn | 154 - logseq/metadata.edn | 1 - logseq/pages-metadata.edn | 648 -- package-lock.json | 6213 +++++++++++++++++ package.json | 109 + pages/About.md | 8 - pages/Butter.md | 1 - ...he contents within the %22bak%22 folder.md | 6 - pages/Charles Chamberlain.md | 3 - pages/Chartodon.md | 6 - pages/Colin the Mathmo.md | 5 - pages/Colophon.md | 7 - pages/Community Infrastructure.md | 24 - pages/Event Listing.md | 8 - pages/Events.md | 2 - pages/February 2022 Meeting.md | 5 - pages/Luma.md | 3 - pages/October 2021 Meeting.md | 23 - pages/Tools For Thought Mastodon.md | 7 - pages/contents.md | 11 - pages/favorites.md | 1 - quartz.config.ts | 77 + quartz.layout.ts | 48 + quartz/bootstrap-cli.mjs | 41 + quartz/bootstrap-worker.mjs | 7 + quartz/build.ts | 183 + quartz/cfg.ts | 55 + quartz/cli/args.js | 98 + quartz/cli/constants.js | 15 + quartz/cli/handlers.js | 511 ++ quartz/cli/helpers.js | 52 + quartz/components/ArticleTitle.tsx | 17 + quartz/components/Backlinks.tsx | 29 + quartz/components/Body.tsx | 13 + quartz/components/Breadcrumbs.tsx | 117 + quartz/components/ContentMeta.tsx | 30 + quartz/components/Darkmode.tsx | 51 + quartz/components/Date.tsx | 29 + quartz/components/DesktopOnly.tsx | 18 + quartz/components/Explorer.tsx | 126 + quartz/components/ExplorerNode.tsx | 224 + quartz/components/Footer.tsx | 32 + quartz/components/Graph.tsx | 100 + quartz/components/Head.tsx | 44 + quartz/components/Header.tsx | 22 + quartz/components/MobileOnly.tsx | 18 + quartz/components/PageList.tsx | 87 + quartz/components/PageTitle.tsx | 20 + quartz/components/RecentNotes.tsx | 81 + quartz/components/Search.tsx | 49 + quartz/components/Spacer.tsx | 7 + quartz/components/TableOfContents.tsx | 84 + quartz/components/TagList.tsx | 53 + quartz/components/index.ts | 45 + quartz/components/pages/404.tsx | 12 + quartz/components/pages/Content.tsx | 11 + quartz/components/pages/FolderContent.tsx | 49 + quartz/components/pages/TagContent.tsx | 94 + quartz/components/renderPage.tsx | 154 + quartz/components/scripts/callout.inline.ts | 44 + quartz/components/scripts/clipboard.inline.ts | 33 + quartz/components/scripts/darkmode.inline.ts | 32 + quartz/components/scripts/explorer.inline.ts | 164 + quartz/components/scripts/graph.inline.ts | 328 + quartz/components/scripts/plausible.inline.ts | 3 + quartz/components/scripts/popover.inline.ts | 83 + quartz/components/scripts/search.inline.ts | 353 + quartz/components/scripts/spa.inline.ts | 155 + quartz/components/scripts/toc.inline.ts | 42 + quartz/components/scripts/util.ts | 25 + quartz/components/styles/backlinks.scss | 20 + quartz/components/styles/breadcrumbs.scss | 22 + quartz/components/styles/clipboard.scss | 36 + quartz/components/styles/darkmode.scss | 48 + quartz/components/styles/explorer.scss | 146 + quartz/components/styles/footer.scss | 15 + quartz/components/styles/graph.scss | 70 + quartz/components/styles/legacyToc.scss | 27 + quartz/components/styles/listPage.scss | 45 + quartz/components/styles/popover.scss | 60 + quartz/components/styles/recentNotes.scss | 24 + quartz/components/styles/search.scss | 178 + quartz/components/styles/toc.scss | 59 + quartz/components/types.ts | 27 + quartz/plugins/emitters/404.tsx | 59 + quartz/plugins/emitters/aliases.ts | 52 + quartz/plugins/emitters/assets.ts | 33 + quartz/plugins/emitters/componentResources.ts | 202 + quartz/plugins/emitters/contentIndex.ts | 150 + quartz/plugins/emitters/contentPage.tsx | 72 + quartz/plugins/emitters/folderPage.tsx | 96 + quartz/plugins/emitters/index.ts | 9 + quartz/plugins/emitters/static.ts | 17 + quartz/plugins/emitters/tagPage.tsx | 94 + quartz/plugins/filters/draft.ts | 9 + quartz/plugins/filters/explicit.ts | 9 + quartz/plugins/filters/index.ts | 2 + quartz/plugins/index.ts | 34 + quartz/plugins/transformers/description.ts | 51 + quartz/plugins/transformers/frontmatter.ts | 75 + quartz/plugins/transformers/gfm.ts | 46 + quartz/plugins/transformers/index.ts | 11 + quartz/plugins/transformers/lastmod.ts | 87 + quartz/plugins/transformers/latex.ts | 45 + quartz/plugins/transformers/linebreaks.ts | 11 + quartz/plugins/transformers/links.ts | 120 + quartz/plugins/transformers/ofm.ts | 528 ++ quartz/plugins/transformers/oxhugofm.ts | 108 + quartz/plugins/transformers/syntax.ts | 16 + quartz/plugins/transformers/toc.ts | 70 + quartz/plugins/types.ts | 54 + quartz/plugins/vfile.ts | 12 + quartz/processors/emit.ts | 44 + quartz/processors/filter.ts | 24 + quartz/processors/parse.ts | 161 + quartz/static/icon.png | Bin 0 -> 17368 bytes quartz/static/og-image.png | Bin 0 -> 39281 bytes quartz/styles/base.scss | 479 ++ quartz/styles/callouts.scss | 111 + quartz/styles/custom.scss | 3 + quartz/styles/syntax.scss | 29 + quartz/styles/variables.scss | 6 + quartz/util/ctx.ts | 19 + quartz/util/escape.ts | 8 + quartz/util/glob.ts | 22 + quartz/util/lang.ts | 11 + quartz/util/log.ts | 28 + quartz/util/path.test.ts | 278 + quartz/util/path.ts | 234 + quartz/util/perf.ts | 19 + quartz/util/resources.tsx | 39 + quartz/util/sourcemap.ts | 18 + quartz/util/theme.ts | 63 + quartz/util/trace.ts | 47 + quartz/worker.ts | 19 + tsconfig.json | 20 + 332 files changed, 17349 insertions(+), 974 deletions(-) mode change 100755 => 100644 .gitignore create mode 100644 .npmrc create mode 100644 .prettierignore create mode 100644 .prettierrc create mode 100644 CODE_OF_CONDUCT.md create mode 100644 Dockerfile create mode 100644 LICENSE.txt mode change 100755 => 100644 README.md create mode 100755 content/about.md rename {assets => content/assets}/Mark_Bernstein_Early_Tools_For_Thought_Jan_2023_1675102979342_0.edn (100%) rename {assets => content/assets}/Mark_Bernstein_Early_Tools_For_Thought_Jan_2023_1675102979342_0.pdf (100%) rename {assets => content/assets}/Mark_Bernstein_Early_Tools_For_Thought_Jan_2023_1675102979342_0__136407078.edn (100%) rename {assets => content/assets}/TfTR-References-BibTeX_1675103690419_0.bib (100%) rename {assets => content/assets}/image_1671407165277_0.png (100%) create mode 100644 content/index.md rename journals/2021_06_15.md => content/journals/2021-06-15.md (100%) rename journals/2021_07_27.md => content/journals/2021-07-27.md (100%) rename journals/2021_08_16.md => content/journals/2021-08-16.md (100%) rename journals/2021_10_25.md => content/journals/2021-10-25.md (100%) rename journals/2021_10_29.md => content/journals/2021-10-29.md (100%) rename journals/2021_12_05.md => content/journals/2021-12-05.md (100%) rename logseq/custom.css => content/journals/2022-02-23.md (100%) mode change 100755 => 100644 rename journals/2022_08_16.md => content/journals/2022-08-16.md (100%) rename journals/2022_11_13.md => content/journals/2022-11-13.md (100%) rename journals/2022_11_16.md => content/journals/2022-11-16.md (100%) rename journals/2022_11_17.md => content/journals/2022-11-17.md (100%) rename journals/2022_11_19.md => content/journals/2022-11-19.md (100%) rename journals/2022_11_20.md => content/journals/2022-11-20.md (100%) rename journals/2022_11_21.md => content/journals/2022-11-21.md (100%) rename journals/2022_11_23.md => content/journals/2022-11-23.md (100%) rename journals/2022_11_30.md => content/journals/2022-11-30.md (100%) rename journals/2023_01_26.md => content/journals/2023-01-26.md (100%) create mode 100644 content/journals/2023-10-19.md rename {pages => content/notes}/A Big List of Mastodon Resources.md (72%) rename {pages => content/notes}/Add your person page.md (100%) rename {pages => content/notes}/Ahmed Fasih.md (100%) rename {pages => content/notes}/Alan Chan.md (100%) rename {pages => content/notes}/Alexander Obernauer.md (100%) create mode 100644 content/notes/Announcements.md rename {pages => content/notes}/April 2022 Meeting.md (100%) rename {pages => content/notes}/August 2021 Meeting.md (100%) rename {pages => content/notes}/Awesome List.md (87%) rename {pages => content/notes}/Bill Seitz.md (100%) rename {pages => content/notes}/Boris Mann.md (100%) create mode 100755 content/notes/Butter.md create mode 100644 content/notes/Calendar.md create mode 100644 content/notes/Charles Chamberlain.md rename {pages => content/notes}/Christina Bowen.md (100%) create mode 100755 content/notes/Colophon.md create mode 100755 content/notes/Community Infrastructure.md rename {pages => content/notes}/Dalton Banks.md (100%) rename {pages => content/notes}/David Brooks.md (100%) create mode 100644 content/notes/Early Tools for Thought Followup.md rename {pages => content/notes}/Early Tools for Thought%2FChat Log.md (100%) rename {pages => content/notes}/Early Tools for Thought%2FReferences.md (100%) rename {pages => content/notes}/Early Tools for Thought.md (100%) rename {pages => content/notes}/Eastgate Systems, Inc..md (100%) create mode 100755 content/notes/Events.md rename {pages => content/notes}/Everything I know about Mastodon.md (73%) rename {pages => content/notes}/Export considered harmful.md (100%) create mode 100755 content/notes/February 2022 Meeting.md rename {pages => content/notes}/Federated Hosting by Spacebear.md (100%) rename {pages => content/notes}/Fedi Tips.md (100%) rename {pages => content/notes}/Fedifinder.md (100%) rename {pages => content/notes}/Fediverse Party - Mastodon.md (100%) rename {pages => content/notes}/Fission Discord.md (100%) rename {pages => content/notes}/GSheet Mastodon Open Servers English 5K+ Active Users.md (100%) rename {pages => content/notes}/Geoffrey Litt.md (100%) rename {pages => content/notes}/GithubIssue.md (100%) rename {pages => content/notes}/Gordon Brander.md (100%) rename {pages => content/notes}/How long can the TFTRocks Mastodon server be relied upon%3F.md (100%) rename {pages => content/notes}/How to Make the Fediverse Your Own.md (100%) rename {pages => content/notes}/How to add content to the TFTRocks Notes site.md (100%) rename {pages => content/notes}/Howard Rheingold.md (100%) rename {pages => content/notes}/Hunter Clarke.md (100%) rename {pages => content/notes}/Hyperdraft.md (100%) rename {pages => content/notes}/Instances.Social.md (100%) rename {pages => content/notes}/Isomorphic Git.md (100%) create mode 100644 content/notes/Jacky Zhao.md rename {pages => content/notes}/Jakob.md (100%) rename {pages => content/notes}/James Walker.md (100%) rename {pages => content/notes}/January 2022 Meeting.md (100%) rename {pages => content/notes}/Jared Pereira.md (100%) rename {pages => content/notes}/Jess Martin.md (100%) rename {pages => content/notes}/Joybox.md (100%) rename {pages => content/notes}/July 2021 Meeting.md (100%) rename {pages => content/notes}/July Meeting Chat Log.md (100%) rename {pages => content/notes}/July Meeting Recap.md (100%) rename {pages => content/notes}/Kevin Lin.md (100%) rename {pages => content/notes}/Kosmik.md (100%) rename {pages => content/notes}/Linus Lee.md (100%) rename {pages => content/notes}/LogSeq Publish.md (100%) rename "pages/LogSeq Schr\303\266dinger.md" => "content/notes/LogSeq Schr\303\266dinger.md" (100%) rename {pages => content/notes}/Lucerne.md (100%) create mode 100755 content/notes/Luma.md rename {pages => content/notes}/Maggie Appleton.md (100%) rename {pages => content/notes}/March 2022 Meeting.md (100%) rename {pages => content/notes}/Mark Anderson.md (100%) rename {pages => content/notes}/Mark Bernstein.md (100%) rename {pages => content/notes}/Mark Upton.md (100%) rename {pages => content/notes}/Mastodon Resources.md (100%) rename {pages => content/notes}/Mastodon.md (100%) rename {pages => content/notes}/May 2022 Meeting.md (100%) rename {pages => content/notes}/Michael Gartner.md (100%) rename {pages => content/notes}/Moa Party.md (100%) rename {pages => content/notes}/Monthly Meetings.md (100%) rename {pages => content/notes}/November 2021 Chat Log.md (100%) rename {pages => content/notes}/November 2021 Meeting.md (100%) rename {pages => content/notes}/October 2021 Chat Log.md (100%) create mode 100755 content/notes/October 2021 Meeting.md rename logseq/graphs-txid.edn => content/notes/Open Collective.md (100%) mode change 100755 => 100644 rename {pages => content/notes}/Paul Rony.md (100%) rename {pages => content/notes}/Paul Shen.md (100%) rename {pages => content/notes}/People.md (100%) rename {pages => content/notes}/Pol Baladas.md (100%) create mode 100644 content/notes/Quartz.md rename {pages => content/notes}/Rafael Nepo.md (100%) rename {pages => content/notes}/Robert Haisfield.md (100%) rename {pages => content/notes}/Rosano.md (100%) rename {pages => content/notes}/Ryan Atkinson.md (100%) rename {pages => content/notes}/Scaling Synthesis - Extending Tools for Thought.md (100%) rename {pages => content/notes}/Scaling Synthesis - Multiplayer TFT.md (100%) rename {pages => content/notes}/September 2021 Meeting.md (100%) rename {pages => content/notes}/Sophia Xu.md (100%) rename {pages => content/notes}/Spacebear.md (100%) rename {pages => content/notes}/Spatial Canvases - Part 1.md (100%) rename {pages => content/notes}/Spatial Canvases - Part 2.md (100%) rename {pages => content/notes}/Spatial Canvases - Part 3.md (100%) rename {pages => content/notes}/Spatial Canvases - Part 4.md (100%) rename {pages => content/notes}/Speaker Listing.md (100%) rename {pages => content/notes}/Sprout.md (100%) rename {pages => content/notes}/Taylor Hadden.md (100%) rename {pages => content/notes}/Templates.md (100%) rename {pages => content/notes}/Tinderbox.md (100%) rename {pages => content/notes}/Tony Confrey.md (100%) create mode 100755 content/notes/Tools For Thought Mastodon.md rename {pages => content/notes}/Tools For Thought Rocks Open Collective.md (100%) rename "pages/Tools for THINKing \342\200\246.md" => "content/notes/Tools for THINKing \342\200\246.md" (100%) rename {pages => content/notes}/Tools for Thought App List.md (100%) rename {pages => content/notes}/Tools for Thought Rocks.md (100%) rename {pages => content/notes}/Tools for Thought as Cultural Practices, not Computational Objects, Maggie Appleton, April 2022 Meeting.md (100%) rename {pages => content/notes}/Tools for Thought.md (100%) rename {pages => content/notes}/Tudor Girba.md (100%) rename {pages => content/notes}/Utilities.md (100%) rename {pages => content/notes}/Volunteer Organizer Kick Off Meeting.md (100%) rename {pages => content/notes}/Weiwei Hsu.md (100%) rename {pages => content/notes}/Wetware of Writing and Doing.md (100%) rename {pages => content/notes}/Write of Passage.md (100%) rename {pages => content/notes}/monthly-meeting.md (100%) rename pages/Yoyogi.md => content/notes/yoyogi.md (100%) create mode 100644 docs/advanced/architecture.md create mode 100644 docs/advanced/creating components.md create mode 100644 docs/advanced/index.md create mode 100644 docs/advanced/making plugins.md create mode 100644 docs/advanced/paths.md create mode 100644 docs/authoring content.md create mode 100644 docs/build.md create mode 100644 docs/configuration.md create mode 100644 docs/features/Docker Support.md create mode 100644 docs/features/Latex.md create mode 100644 docs/features/Mermaid diagrams.md create mode 100644 docs/features/Obsidian compatibility.md create mode 100644 docs/features/OxHugo compatibility.md create mode 100644 docs/features/RSS Feed.md create mode 100644 docs/features/SPA Routing.md create mode 100644 docs/features/backlinks.md create mode 100644 docs/features/breadcrumbs.md create mode 100644 docs/features/callouts.md create mode 100644 docs/features/darkmode.md create mode 100644 docs/features/explorer.md create mode 100644 docs/features/folder and tag listings.md create mode 100644 docs/features/full-text search.md create mode 100644 docs/features/graph view.md create mode 100644 docs/features/index.md create mode 100644 docs/features/popover previews.md create mode 100644 docs/features/private pages.md create mode 100644 docs/features/recent notes.md create mode 100644 docs/features/syntax highlighting.md create mode 100644 docs/features/table of contents.md create mode 100644 docs/features/upcoming features.md create mode 100644 docs/features/wikilinks.md create mode 100644 docs/hosting.md create mode 100644 docs/images/dns records.png create mode 100644 docs/images/quartz layout.png create mode 100644 docs/images/quartz transform pipeline.png create mode 100644 docs/index.md create mode 100644 docs/layout.md create mode 100644 docs/migrating from Quartz 3.md create mode 100644 docs/philosophy.md create mode 100644 docs/showcase.md create mode 100644 docs/tags/component.md create mode 100644 docs/upgrading.md create mode 100644 globals.d.ts create mode 100644 index.d.ts delete mode 100755 logseq/bak/pages/July 2021 Meeting/2022-06-29T23_03_23.581Z.md delete mode 100644 logseq/config.edn delete mode 100755 logseq/metadata.edn delete mode 100755 logseq/pages-metadata.edn create mode 100644 package-lock.json create mode 100644 package.json delete mode 100755 pages/About.md delete mode 100755 pages/Butter.md delete mode 100755 pages/Can I delete the contents within the %22bak%22 folder.md delete mode 100644 pages/Charles Chamberlain.md delete mode 100755 pages/Chartodon.md delete mode 100755 pages/Colin the Mathmo.md delete mode 100755 pages/Colophon.md delete mode 100755 pages/Community Infrastructure.md delete mode 100755 pages/Event Listing.md delete mode 100755 pages/Events.md delete mode 100755 pages/February 2022 Meeting.md delete mode 100755 pages/Luma.md delete mode 100755 pages/October 2021 Meeting.md delete mode 100755 pages/Tools For Thought Mastodon.md delete mode 100755 pages/contents.md delete mode 100755 pages/favorites.md create mode 100644 quartz.config.ts create mode 100644 quartz.layout.ts create mode 100755 quartz/bootstrap-cli.mjs create mode 100644 quartz/bootstrap-worker.mjs create mode 100644 quartz/build.ts create mode 100644 quartz/cfg.ts create mode 100644 quartz/cli/args.js create mode 100644 quartz/cli/constants.js create mode 100644 quartz/cli/handlers.js create mode 100644 quartz/cli/helpers.js create mode 100644 quartz/components/ArticleTitle.tsx create mode 100644 quartz/components/Backlinks.tsx create mode 100644 quartz/components/Body.tsx create mode 100644 quartz/components/Breadcrumbs.tsx create mode 100644 quartz/components/ContentMeta.tsx create mode 100644 quartz/components/Darkmode.tsx create mode 100644 quartz/components/Date.tsx create mode 100644 quartz/components/DesktopOnly.tsx create mode 100644 quartz/components/Explorer.tsx create mode 100644 quartz/components/ExplorerNode.tsx create mode 100644 quartz/components/Footer.tsx create mode 100644 quartz/components/Graph.tsx create mode 100644 quartz/components/Head.tsx create mode 100644 quartz/components/Header.tsx create mode 100644 quartz/components/MobileOnly.tsx create mode 100644 quartz/components/PageList.tsx create mode 100644 quartz/components/PageTitle.tsx create mode 100644 quartz/components/RecentNotes.tsx create mode 100644 quartz/components/Search.tsx create mode 100644 quartz/components/Spacer.tsx create mode 100644 quartz/components/TableOfContents.tsx create mode 100644 quartz/components/TagList.tsx create mode 100644 quartz/components/index.ts create mode 100644 quartz/components/pages/404.tsx create mode 100644 quartz/components/pages/Content.tsx create mode 100644 quartz/components/pages/FolderContent.tsx create mode 100644 quartz/components/pages/TagContent.tsx create mode 100644 quartz/components/renderPage.tsx create mode 100644 quartz/components/scripts/callout.inline.ts create mode 100644 quartz/components/scripts/clipboard.inline.ts create mode 100644 quartz/components/scripts/darkmode.inline.ts create mode 100644 quartz/components/scripts/explorer.inline.ts create mode 100644 quartz/components/scripts/graph.inline.ts create mode 100644 quartz/components/scripts/plausible.inline.ts create mode 100644 quartz/components/scripts/popover.inline.ts create mode 100644 quartz/components/scripts/search.inline.ts create mode 100644 quartz/components/scripts/spa.inline.ts create mode 100644 quartz/components/scripts/toc.inline.ts create mode 100644 quartz/components/scripts/util.ts create mode 100644 quartz/components/styles/backlinks.scss create mode 100644 quartz/components/styles/breadcrumbs.scss create mode 100644 quartz/components/styles/clipboard.scss create mode 100644 quartz/components/styles/darkmode.scss create mode 100644 quartz/components/styles/explorer.scss create mode 100644 quartz/components/styles/footer.scss create mode 100644 quartz/components/styles/graph.scss create mode 100644 quartz/components/styles/legacyToc.scss create mode 100644 quartz/components/styles/listPage.scss create mode 100644 quartz/components/styles/popover.scss create mode 100644 quartz/components/styles/recentNotes.scss create mode 100644 quartz/components/styles/search.scss create mode 100644 quartz/components/styles/toc.scss create mode 100644 quartz/components/types.ts create mode 100644 quartz/plugins/emitters/404.tsx create mode 100644 quartz/plugins/emitters/aliases.ts create mode 100644 quartz/plugins/emitters/assets.ts create mode 100644 quartz/plugins/emitters/componentResources.ts create mode 100644 quartz/plugins/emitters/contentIndex.ts create mode 100644 quartz/plugins/emitters/contentPage.tsx create mode 100644 quartz/plugins/emitters/folderPage.tsx create mode 100644 quartz/plugins/emitters/index.ts create mode 100644 quartz/plugins/emitters/static.ts create mode 100644 quartz/plugins/emitters/tagPage.tsx create mode 100644 quartz/plugins/filters/draft.ts create mode 100644 quartz/plugins/filters/explicit.ts create mode 100644 quartz/plugins/filters/index.ts create mode 100644 quartz/plugins/index.ts create mode 100644 quartz/plugins/transformers/description.ts create mode 100644 quartz/plugins/transformers/frontmatter.ts create mode 100644 quartz/plugins/transformers/gfm.ts create mode 100644 quartz/plugins/transformers/index.ts create mode 100644 quartz/plugins/transformers/lastmod.ts create mode 100644 quartz/plugins/transformers/latex.ts create mode 100644 quartz/plugins/transformers/linebreaks.ts create mode 100644 quartz/plugins/transformers/links.ts create mode 100644 quartz/plugins/transformers/ofm.ts create mode 100644 quartz/plugins/transformers/oxhugofm.ts create mode 100644 quartz/plugins/transformers/syntax.ts create mode 100644 quartz/plugins/transformers/toc.ts create mode 100644 quartz/plugins/types.ts create mode 100644 quartz/plugins/vfile.ts create mode 100644 quartz/processors/emit.ts create mode 100644 quartz/processors/filter.ts create mode 100644 quartz/processors/parse.ts create mode 100644 quartz/static/icon.png create mode 100644 quartz/static/og-image.png create mode 100644 quartz/styles/base.scss create mode 100644 quartz/styles/callouts.scss create mode 100644 quartz/styles/custom.scss create mode 100644 quartz/styles/syntax.scss create mode 100644 quartz/styles/variables.scss create mode 100644 quartz/util/ctx.ts create mode 100644 quartz/util/escape.ts create mode 100644 quartz/util/glob.ts create mode 100644 quartz/util/lang.ts create mode 100644 quartz/util/log.ts create mode 100644 quartz/util/path.test.ts create mode 100644 quartz/util/path.ts create mode 100644 quartz/util/perf.ts create mode 100644 quartz/util/resources.tsx create mode 100644 quartz/util/sourcemap.ts create mode 100644 quartz/util/theme.ts create mode 100644 quartz/util/trace.ts create mode 100644 quartz/worker.ts create mode 100644 tsconfig.json diff --git a/.github/workflows/publish.yml b/.github/workflows/publish.yml index 7be5dff..0d4933e 100755 --- a/.github/workflows/publish.yml +++ b/.github/workflows/publish.yml @@ -1,35 +1,45 @@ -name: Publish LogSeq to GH Pages - -# Controls when the workflow will run +name: Deploy Quartz site to GitHub Pages + on: - # Triggers the workflow on push or pull request events but only for the "master" branch push: - branches: [ "master" ] - - # Allows you to run this workflow manually from the Actions tab - workflow_dispatch: - -# A workflow run is made up of one or more jobs that can run sequentially or in parallel + branches: + - quartz + +permissions: + contents: read + pages: write + id-token: write + +concurrency: + group: "pages" + cancel-in-progress: false + jobs: - # This workflow contains a single job called "build" build: - # The type of runner that the job will run on - runs-on: ubuntu-latest + runs-on: ubuntu-22.04 steps: - - name: 📂 setup - uses: actions/checkout@v2 - - name: Logseq Publish 🚩 - uses: pengx17/logseq-publish@0.1.7 - - name: add a nojekyll file # to make sure asset paths are correctly identified - run: touch $GITHUB_WORKSPACE/www/.nojekyll - - name: Deploy 🚀 - uses: JamesIves/github-pages-deploy-action@4.1.9 + - uses: actions/checkout@v3 + with: + fetch-depth: 0 # Fetch all history for git info + - uses: actions/setup-node@v3 + with: + node-version: 18.14 + - name: Install Dependencies + run: npm ci + - name: Build Quartz + run: npx quartz build + - name: Upload artifact + uses: actions/upload-pages-artifact@v2 with: - branch: gh-pages # The branch the action should deploy to. - folder: www # The folder the action should deploy. - clean: true -# If we choose to deploy to IPFS with Fission -# - name: 🚀 fission publish -# uses: fission-suite/publish-action@v1 -# with: -# machine_key: ${{ secrets.FISSION_KEY }} + path: public + + deploy: + needs: build + environment: + name: github-pages + url: ${{ steps.deployment.outputs.page_url }} + runs-on: ubuntu-latest + steps: + - name: Deploy to GitHub Pages + id: deployment + uses: actions/deploy-pages@v2 \ No newline at end of file diff --git a/.gitignore b/.gitignore old mode 100755 new mode 100644 index 04da6c8..25d07db --- a/.gitignore +++ b/.gitignore @@ -1,3 +1,11 @@ .DS_Store -logseq/bak -logseq/.recycle +.gitignore +node_modules +public +prof +tsconfig.tsbuildinfo +.obsidian +.quartz-cache +private/ +.replit +replit.nix diff --git a/.npmrc b/.npmrc new file mode 100644 index 0000000..b6f27f1 --- /dev/null +++ b/.npmrc @@ -0,0 +1 @@ +engine-strict=true diff --git a/.prettierignore b/.prettierignore new file mode 100644 index 0000000..3c0687a --- /dev/null +++ b/.prettierignore @@ -0,0 +1,3 @@ +public +node_modules +.quartz-cache diff --git a/.prettierrc b/.prettierrc new file mode 100644 index 0000000..5788b66 --- /dev/null +++ b/.prettierrc @@ -0,0 +1,7 @@ +{ + "printWidth": 100, + "quoteProps": "as-needed", + "trailingComma": "all", + "tabWidth": 2, + "semi": false +} diff --git a/CODE_OF_CONDUCT.md b/CODE_OF_CONDUCT.md new file mode 100644 index 0000000..887a2c4 --- /dev/null +++ b/CODE_OF_CONDUCT.md @@ -0,0 +1,90 @@ +# Citizen Code of Conduct + +## 1. Purpose + +A primary goal of the Quartz community is to be inclusive to the largest number of contributors, with the most varied and diverse backgrounds possible. As such, we are committed to providing a friendly, safe and welcoming environment for all, regardless of gender, sexual orientation, ability, ethnicity, socioeconomic status, and religion (or lack thereof). + +This code of conduct outlines our expectations for all those who participate in our community, as well as the consequences for unacceptable behavior. + +We invite all those who participate in the Quartz community to help us create safe and positive experiences for everyone. + +## 2. Open [Source/Culture/Tech] Citizenship + +A supplemental goal of this Code of Conduct is to increase open [source/culture/tech] citizenship by encouraging participants to recognize and strengthen the relationships between our actions and their effects on our community. + +Communities mirror the societies in which they exist and positive action is essential to counteract the many forms of inequality and abuses of power that exist in society. + +If you see someone who is making an extra effort to ensure our community is welcoming, friendly, and encourages all participants to contribute to the fullest extent, we want to know. + +## 3. Expected Behavior + +The following behaviors are expected and requested of all community members: + +- Participate in an authentic and active way. In doing so, you contribute to the health and longevity of this community. +- Exercise consideration and respect in your speech and actions. +- Attempt collaboration before conflict. +- Refrain from demeaning, discriminatory, or harassing behavior and speech. +- Be mindful of your surroundings and of your fellow participants. Alert community leaders if you notice a dangerous situation, someone in distress, or violations of this Code of Conduct, even if they seem inconsequential. +- Remember that community event venues may be shared with members of the public; please be respectful to all patrons of these locations. + +## 4. Unacceptable Behavior + +The following behaviors are considered harassment and are unacceptable within our community: + +- Violence, threats of violence or violent language directed against another person. +- Sexist, racist, homophobic, transphobic, ableist or otherwise discriminatory jokes and language. +- Posting or displaying sexually explicit or violent material. +- Posting or threatening to post other people's personally identifying information ("doxing"). +- Personal insults, particularly those related to gender, sexual orientation, race, religion, or disability. +- Inappropriate photography or recording. +- Inappropriate physical contact. You should have someone's consent before touching them. +- Unwelcome sexual attention. This includes, sexualized comments or jokes; inappropriate touching, groping, and unwelcomed sexual advances. +- Deliberate intimidation, stalking or following (online or in person). +- Advocating for, or encouraging, any of the above behavior. +- Sustained disruption of community events, including talks and presentations. + +## 5. Weapons Policy + +No weapons will be allowed at Quartz community events, community spaces, or in other spaces covered by the scope of this Code of Conduct. Weapons include but are not limited to guns, explosives (including fireworks), and large knives such as those used for hunting or display, as well as any other item used for the purpose of causing injury or harm to others. Anyone seen in possession of one of these items will be asked to leave immediately, and will only be allowed to return without the weapon. Community members are further expected to comply with all state and local laws on this matter. + +## 6. Consequences of Unacceptable Behavior + +Unacceptable behavior from any community member, including sponsors and those with decision-making authority, will not be tolerated. + +Anyone asked to stop unacceptable behavior is expected to comply immediately. + +If a community member engages in unacceptable behavior, the community organizers may take any action they deem appropriate, up to and including a temporary ban or permanent expulsion from the community without warning (and without refund in the case of a paid event). + +## 7. Reporting Guidelines + +If you are subject to or witness unacceptable behavior, or have any other concerns, please notify a community organizer as soon as possible. j.zhao2k19@gmail.com. + +Additionally, community organizers are available to help community members engage with local law enforcement or to otherwise help those experiencing unacceptable behavior feel safe. In the context of in-person events, organizers will also provide escorts as desired by the person experiencing distress. + +## 8. Addressing Grievances + +If you feel you have been falsely or unfairly accused of violating this Code of Conduct, you should notify @jackyzha0 with a concise description of your grievance. Your grievance will be handled in accordance with our existing governing policies. + +## 9. Scope + +We expect all community participants (contributors, paid or otherwise; sponsors; and other guests) to abide by this Code of Conduct in all community venues--online and in-person--as well as in all one-on-one communications pertaining to community business. + +This code of conduct and its related procedures also applies to unacceptable behavior occurring outside the scope of community activities when such behavior has the potential to adversely affect the safety and well-being of community members. + +## 10. Contact info + +j.zhao2k19@gmail.com + +## 11. License and attribution + +The Citizen Code of Conduct is distributed by [Stumptown Syndicate](http://stumptownsyndicate.org) under a [Creative Commons Attribution-ShareAlike license](http://creativecommons.org/licenses/by-sa/3.0/). + +Portions of text derived from the [Django Code of Conduct](https://www.djangoproject.com/conduct/) and the [Geek Feminism Anti-Harassment Policy](http://geekfeminism.wikia.com/wiki/Conference_anti-harassment/Policy). + +_Revision 2.3. Posted 6 March 2017._ + +_Revision 2.2. Posted 4 February 2016._ + +_Revision 2.1. Posted 23 June 2014._ + +_Revision 2.0, adopted by the [Stumptown Syndicate](http://stumptownsyndicate.org) board on 10 January 2013. Posted 17 March 2013._ diff --git a/Dockerfile b/Dockerfile new file mode 100644 index 0000000..1d9e591 --- /dev/null +++ b/Dockerfile @@ -0,0 +1,11 @@ +FROM node:20-slim as builder +WORKDIR /usr/src/app +COPY package.json . +COPY package-lock.json* . +RUN npm ci + +FROM node:20-slim +WORKDIR /usr/src/app +COPY --from=builder /usr/src/app/ /usr/src/app/ +COPY . . +CMD ["npx", "quartz", "build", "--serve"] diff --git a/LICENSE.txt b/LICENSE.txt new file mode 100644 index 0000000..147e2ca --- /dev/null +++ b/LICENSE.txt @@ -0,0 +1,21 @@ +MIT License + +Copyright (c) 2021 jackyzha0 + +Permission is hereby granted, free of charge, to any person obtaining a copy +of this software and associated documentation files (the "Software"), to deal +in the Software without restriction, including without limitation the rights +to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the Software is +furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in all +copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +SOFTWARE. diff --git a/README.md b/README.md old mode 100755 new mode 100644 index c2af719..4b4731c --- a/README.md +++ b/README.md @@ -1,3 +1,20 @@ -- # Tools For Thought Rocks Notes - - Notes from events, speakers, and recap are being kept here, and edited with the web-based version of #LogSeq \ No newline at end of file +# Quartz v4 + +> “[One] who works with the door open gets all kinds of interruptions, but [they] also occasionally gets clues as to what the world is and what might be important.” — Richard Hamming + +Quartz is a set of tools that helps you publish your [digital garden](https://jzhao.xyz/posts/networked-thought) and notes as a website for free. +Quartz v4 features a from-the-ground rewrite focusing on end-user extensibility and ease-of-use. + +**If you are looking for Quartz v3, you can find it on the [`hugo` branch](https://github.com/jackyzha0/quartz/tree/hugo).** + +🔗 Read the documentation and get started: https://quartz.jzhao.xyz/ + +[Join the Discord Community](https://discord.gg/cRFFHYye7t) + +## Sponsors + +

+ + + +

diff --git a/content/about.md b/content/about.md new file mode 100755 index 0000000..19eccf7 --- /dev/null +++ b/content/about.md @@ -0,0 +1,5 @@ +This is an experimental set of notes for the Tools For Thought Rocks community + +It is a place to document the events, speakers, videos, chat transcripts, and other content around the community + +It also will hold information and documentation about the [[Tools For Thought Mastodon]] \ No newline at end of file diff --git a/assets/Mark_Bernstein_Early_Tools_For_Thought_Jan_2023_1675102979342_0.edn b/content/assets/Mark_Bernstein_Early_Tools_For_Thought_Jan_2023_1675102979342_0.edn similarity index 100% rename from assets/Mark_Bernstein_Early_Tools_For_Thought_Jan_2023_1675102979342_0.edn rename to content/assets/Mark_Bernstein_Early_Tools_For_Thought_Jan_2023_1675102979342_0.edn diff --git a/assets/Mark_Bernstein_Early_Tools_For_Thought_Jan_2023_1675102979342_0.pdf b/content/assets/Mark_Bernstein_Early_Tools_For_Thought_Jan_2023_1675102979342_0.pdf similarity index 100% rename from assets/Mark_Bernstein_Early_Tools_For_Thought_Jan_2023_1675102979342_0.pdf rename to content/assets/Mark_Bernstein_Early_Tools_For_Thought_Jan_2023_1675102979342_0.pdf diff --git a/assets/Mark_Bernstein_Early_Tools_For_Thought_Jan_2023_1675102979342_0__136407078.edn b/content/assets/Mark_Bernstein_Early_Tools_For_Thought_Jan_2023_1675102979342_0__136407078.edn similarity index 100% rename from assets/Mark_Bernstein_Early_Tools_For_Thought_Jan_2023_1675102979342_0__136407078.edn rename to content/assets/Mark_Bernstein_Early_Tools_For_Thought_Jan_2023_1675102979342_0__136407078.edn diff --git a/assets/TfTR-References-BibTeX_1675103690419_0.bib b/content/assets/TfTR-References-BibTeX_1675103690419_0.bib similarity index 100% rename from assets/TfTR-References-BibTeX_1675103690419_0.bib rename to content/assets/TfTR-References-BibTeX_1675103690419_0.bib diff --git a/assets/image_1671407165277_0.png b/content/assets/image_1671407165277_0.png similarity index 100% rename from assets/image_1671407165277_0.png rename to content/assets/image_1671407165277_0.png diff --git a/content/index.md b/content/index.md new file mode 100644 index 0000000..5944ce1 --- /dev/null +++ b/content/index.md @@ -0,0 +1,14 @@ +--- +title: Welcome to Tools for Thought Rocks! +--- + +Tools for Thought Rocks! is a community interested in second brains, digital gardens, spatial canvases, and the design and programming of various tools for thought systems. More on the [[About]] page » + +## Events + + + +Subscribe to the [[Calendar]] +## Explore + +The [[Journals]] folder has daily notes with updates. \ No newline at end of file diff --git a/journals/2021_06_15.md b/content/journals/2021-06-15.md similarity index 100% rename from journals/2021_06_15.md rename to content/journals/2021-06-15.md diff --git a/journals/2021_07_27.md b/content/journals/2021-07-27.md similarity index 100% rename from journals/2021_07_27.md rename to content/journals/2021-07-27.md diff --git a/journals/2021_08_16.md b/content/journals/2021-08-16.md similarity index 100% rename from journals/2021_08_16.md rename to content/journals/2021-08-16.md diff --git a/journals/2021_10_25.md b/content/journals/2021-10-25.md similarity index 100% rename from journals/2021_10_25.md rename to content/journals/2021-10-25.md diff --git a/journals/2021_10_29.md b/content/journals/2021-10-29.md similarity index 100% rename from journals/2021_10_29.md rename to content/journals/2021-10-29.md diff --git a/journals/2021_12_05.md b/content/journals/2021-12-05.md similarity index 100% rename from journals/2021_12_05.md rename to content/journals/2021-12-05.md diff --git a/logseq/custom.css b/content/journals/2022-02-23.md old mode 100755 new mode 100644 similarity index 100% rename from logseq/custom.css rename to content/journals/2022-02-23.md diff --git a/journals/2022_08_16.md b/content/journals/2022-08-16.md similarity index 100% rename from journals/2022_08_16.md rename to content/journals/2022-08-16.md diff --git a/journals/2022_11_13.md b/content/journals/2022-11-13.md similarity index 100% rename from journals/2022_11_13.md rename to content/journals/2022-11-13.md diff --git a/journals/2022_11_16.md b/content/journals/2022-11-16.md similarity index 100% rename from journals/2022_11_16.md rename to content/journals/2022-11-16.md diff --git a/journals/2022_11_17.md b/content/journals/2022-11-17.md similarity index 100% rename from journals/2022_11_17.md rename to content/journals/2022-11-17.md diff --git a/journals/2022_11_19.md b/content/journals/2022-11-19.md similarity index 100% rename from journals/2022_11_19.md rename to content/journals/2022-11-19.md diff --git a/journals/2022_11_20.md b/content/journals/2022-11-20.md similarity index 100% rename from journals/2022_11_20.md rename to content/journals/2022-11-20.md diff --git a/journals/2022_11_21.md b/content/journals/2022-11-21.md similarity index 100% rename from journals/2022_11_21.md rename to content/journals/2022-11-21.md diff --git a/journals/2022_11_23.md b/content/journals/2022-11-23.md similarity index 100% rename from journals/2022_11_23.md rename to content/journals/2022-11-23.md diff --git a/journals/2022_11_30.md b/content/journals/2022-11-30.md similarity index 100% rename from journals/2022_11_30.md rename to content/journals/2022-11-30.md diff --git a/journals/2023_01_26.md b/content/journals/2023-01-26.md similarity index 100% rename from journals/2023_01_26.md rename to content/journals/2023-01-26.md diff --git a/content/journals/2023-10-19.md b/content/journals/2023-10-19.md new file mode 100644 index 0000000..2981a21 --- /dev/null +++ b/content/journals/2023-10-19.md @@ -0,0 +1,3 @@ +Set up [[Quartz]] to publish content, to be managed with [[Obsidian]]. + +#changelog #quartz \ No newline at end of file diff --git a/pages/A Big List of Mastodon Resources.md b/content/notes/A Big List of Mastodon Resources.md similarity index 72% rename from pages/A Big List of Mastodon Resources.md rename to content/notes/A Big List of Mastodon Resources.md index 8f6fd28..b4b4246 100755 --- a/pages/A Big List of Mastodon Resources.md +++ b/content/notes/A Big List of Mastodon Resources.md @@ -1,7 +1,7 @@ -link:: https://researchbuzz.me/2022/11/05/a-big-list-of-mastodon-resources/ +https://researchbuzz.me/2022/11/05/a-big-list-of-mastodon-resources/ + +Author [[Tara Calishain]] -- author:: [[Tara Calishain]] -- - Primers - [[Everything I know about Mastodon]] - [[Fedi Tips]] diff --git a/pages/Add your person page.md b/content/notes/Add your person page.md similarity index 100% rename from pages/Add your person page.md rename to content/notes/Add your person page.md diff --git a/pages/Ahmed Fasih.md b/content/notes/Ahmed Fasih.md similarity index 100% rename from pages/Ahmed Fasih.md rename to content/notes/Ahmed Fasih.md diff --git a/pages/Alan Chan.md b/content/notes/Alan Chan.md similarity index 100% rename from pages/Alan Chan.md rename to content/notes/Alan Chan.md diff --git a/pages/Alexander Obernauer.md b/content/notes/Alexander Obernauer.md similarity index 100% rename from pages/Alexander Obernauer.md rename to content/notes/Alexander Obernauer.md diff --git a/content/notes/Announcements.md b/content/notes/Announcements.md new file mode 100644 index 0000000..7b57edc --- /dev/null +++ b/content/notes/Announcements.md @@ -0,0 +1 @@ +- [[Early Tools for Thought Followup]] \ No newline at end of file diff --git a/pages/April 2022 Meeting.md b/content/notes/April 2022 Meeting.md similarity index 100% rename from pages/April 2022 Meeting.md rename to content/notes/April 2022 Meeting.md diff --git a/pages/August 2021 Meeting.md b/content/notes/August 2021 Meeting.md similarity index 100% rename from pages/August 2021 Meeting.md rename to content/notes/August 2021 Meeting.md diff --git a/pages/Awesome List.md b/content/notes/Awesome List.md similarity index 87% rename from pages/Awesome List.md rename to content/notes/Awesome List.md index f02bca5..ad150a4 100755 --- a/pages/Awesome List.md +++ b/content/notes/Awesome List.md @@ -1,6 +1,8 @@ -- Created by [[Sindre Sorhus]] -- The meta list that aggregates other awesome lists is here https://github.com/sindresorhus/awesome -- The [Awesome manifesto](https://github.com/sindresorhus/awesome/blob/main/awesome.md) describes how / why to run a list +Created by [[Sindre Sorhus]] + +The meta list that aggregates other awesome lists is here https://github.com/sindresorhus/awesome + +The [Awesome manifesto](https://github.com/sindresorhus/awesome/blob/main/awesome.md) describes how / why to run a list - **Comment on why something is awesome** - Apart from suggesting a particular item on your list, you should also inform your readers why it's on the list and how they will benefit from it. - **Make it clear what the list is about** diff --git a/pages/Bill Seitz.md b/content/notes/Bill Seitz.md similarity index 100% rename from pages/Bill Seitz.md rename to content/notes/Bill Seitz.md diff --git a/pages/Boris Mann.md b/content/notes/Boris Mann.md similarity index 100% rename from pages/Boris Mann.md rename to content/notes/Boris Mann.md diff --git a/content/notes/Butter.md b/content/notes/Butter.md new file mode 100755 index 0000000..7f52a5a --- /dev/null +++ b/content/notes/Butter.md @@ -0,0 +1 @@ +https://www.butter.us/ diff --git a/content/notes/Calendar.md b/content/notes/Calendar.md new file mode 100644 index 0000000..bbf8200 --- /dev/null +++ b/content/notes/Calendar.md @@ -0,0 +1,3 @@ + + +We use [[Luma]] to power our calendar. You are welcome to submit and share events related to Tools for Thought! \ No newline at end of file diff --git a/content/notes/Charles Chamberlain.md b/content/notes/Charles Chamberlain.md new file mode 100644 index 0000000..67850f4 --- /dev/null +++ b/content/notes/Charles Chamberlain.md @@ -0,0 +1,5 @@ +https://birch.ink/ + +#twitter https://twitter.com/charlesetc + +#person #speaker diff --git a/pages/Christina Bowen.md b/content/notes/Christina Bowen.md similarity index 100% rename from pages/Christina Bowen.md rename to content/notes/Christina Bowen.md diff --git a/content/notes/Colophon.md b/content/notes/Colophon.md new file mode 100755 index 0000000..fb21853 --- /dev/null +++ b/content/notes/Colophon.md @@ -0,0 +1,9 @@ +This site is managed with [[Obsidian]] + +The source lives in the Github organization at https://github.com/ToolsForThoughtRocks/ToolsForThoughtLogSeq + +There is a build script that turns it into static pages that are published via Github Pages + +You can access it via https://notes.toolsforthought.rocks + +The [[Community Infrastructure]] page has links to other resources. diff --git a/content/notes/Community Infrastructure.md b/content/notes/Community Infrastructure.md new file mode 100755 index 0000000..3201d87 --- /dev/null +++ b/content/notes/Community Infrastructure.md @@ -0,0 +1,51 @@ +This is all of the infrastructure, from people, to tools, to access, that helps run the [[Tools for Thought Rocks]] community + +Current organizers are [[Christina Bowen]], [[Boris Mann]], and [[Jess Martin]] + +Anyone can propose / organize events! What else can we do together? + +## Calendar + +https://lu.ma/toolsforthought is the link for the community calendar powered by [[Luma]] + +It is embedded on this site on the [[Calendar]] page + +## Github + +https://github.com/ToolsForThoughtRocks + +A number of people are [owners / members of the org](https://github.com/orgs/ToolsForThoughtRocks/people) (not all are publicly listed) + - [[Boris Mann]], Owner + - [[Jess Martin]], Owner + - [[Michael Gartner]], Owner + +## Youtube + +https://youtube.com/@toolsforthought + +* *[[Boris Mann]], Owner +* [[Jess Martin]], Owner +* [[Christina Bowen]], Owner +* [[Michael Gartner]], Manager + +## Open Collective + +https://opencollective.com/toolsforthought + +## Mastodon + +https://toolsforthought.social + +More at [[Tools For Thought Mastodon]] + +## Domain Name + +The domain `toolsforthought.rocks` is registered at and has DNS hosted by [[Google Domains]] + +* *[[Boris Mann]] and [[Jess Martin]] have full permissions over it + +## Website + +https://notes.toolsforthought.rocks, until we shut down the old Mastodon server. + +The [[Colophon]] describes the infrastructure for this [[TFTRocks Notes]] site \ No newline at end of file diff --git a/pages/Dalton Banks.md b/content/notes/Dalton Banks.md similarity index 100% rename from pages/Dalton Banks.md rename to content/notes/Dalton Banks.md diff --git a/pages/David Brooks.md b/content/notes/David Brooks.md similarity index 100% rename from pages/David Brooks.md rename to content/notes/David Brooks.md diff --git a/content/notes/Early Tools for Thought Followup.md b/content/notes/Early Tools for Thought Followup.md new file mode 100644 index 0000000..33451f8 --- /dev/null +++ b/content/notes/Early Tools for Thought Followup.md @@ -0,0 +1,3 @@ +published:: [[Feb 5th, 2023]] + +- \ No newline at end of file diff --git a/pages/Early Tools for Thought%2FChat Log.md b/content/notes/Early Tools for Thought%2FChat Log.md similarity index 100% rename from pages/Early Tools for Thought%2FChat Log.md rename to content/notes/Early Tools for Thought%2FChat Log.md diff --git a/pages/Early Tools for Thought%2FReferences.md b/content/notes/Early Tools for Thought%2FReferences.md similarity index 100% rename from pages/Early Tools for Thought%2FReferences.md rename to content/notes/Early Tools for Thought%2FReferences.md diff --git a/pages/Early Tools for Thought.md b/content/notes/Early Tools for Thought.md similarity index 100% rename from pages/Early Tools for Thought.md rename to content/notes/Early Tools for Thought.md diff --git a/pages/Eastgate Systems, Inc..md b/content/notes/Eastgate Systems, Inc..md similarity index 100% rename from pages/Eastgate Systems, Inc..md rename to content/notes/Eastgate Systems, Inc..md diff --git a/content/notes/Events.md b/content/notes/Events.md new file mode 100755 index 0000000..31c743a --- /dev/null +++ b/content/notes/Events.md @@ -0,0 +1 @@ +We use [[Luma]] to maintain the [[Calendar]] \ No newline at end of file diff --git a/pages/Everything I know about Mastodon.md b/content/notes/Everything I know about Mastodon.md similarity index 73% rename from pages/Everything I know about Mastodon.md rename to content/notes/Everything I know about Mastodon.md index 287ce13..3b3f4c7 100755 --- a/pages/Everything I know about Mastodon.md +++ b/content/notes/Everything I know about Mastodon.md @@ -1,6 +1,9 @@ -author:: [[Danielle Navarro]] +https://blog.djnavarro.net/posts/2022-11-03_what-i-know-about-mastodon/ -- link:: https://blog.djnavarro.net/posts/2022-11-03_what-i-know-about-mastodon/ -- published:: [[November 3rd, 2022]] -- It’s written from the time-tested pedagogical perspective of “the writer who is only one chapter ahead of her audience in the textbook”. I’ve been on mastodon for a few days, but this isn’t my first rodeo over there: I signed up for it very early on several years ago, and tried again a few years after that. This time I’m a lot more enthusiastic about it than the last two, so I’m writing a quick introductory post to help my fellow data science folks test out the waters. I sort of know what I’m doing there but not completely! -- If you want a more detailed guide on navigating Mastodon and the fediverse, I recommend [fedi.tips](https://fedi.tips/). There’s a lot of answers to common questions over there, from someone who actually does know what they are doing! Alternatively you can read [this thread](https://mastodon.nz/@explorergrace/109272759764872091) which covers a lot of the same things I’m saying here! \ No newline at end of file +Author [[Danielle Navarro]] + +> It’s written from the time-tested pedagogical perspective of “the writer who is only one chapter ahead of her audience in the textbook”. I’ve been on mastodon for a few days, but this isn’t my first rodeo over there: I signed up for it very early on several years ago, and tried again a few years after that. This time I’m a lot more enthusiastic about it than the last two, so I’m writing a quick introductory post to help my fellow data science folks test out the waters. I sort of know what I’m doing there but not completely! +> +> If you want a more detailed guide on navigating Mastodon and the fediverse, I recommend [fedi.tips](https://fedi.tips/). There’s a lot of answers to common questions over there, from someone who actually does know what they are doing! Alternatively you can read [this thread](https://mastodon.nz/@explorergrace/109272759764872091) which covers a lot of the same things I’m saying here! + +#article #mastodon \ No newline at end of file diff --git a/pages/Export considered harmful.md b/content/notes/Export considered harmful.md similarity index 100% rename from pages/Export considered harmful.md rename to content/notes/Export considered harmful.md diff --git a/content/notes/February 2022 Meeting.md b/content/notes/February 2022 Meeting.md new file mode 100755 index 0000000..4502c30 --- /dev/null +++ b/content/notes/February 2022 Meeting.md @@ -0,0 +1,16 @@ +[[2022-02-23]] + +https://lu.ma/tftrocks-feb22 +## Speakers + +[[Alexander Obenauer]] [@alexanderobenauer](https://twitter.com/alexobenauer)will regale us with insights gleaned while writing his fabulous [Lab Notes](https://alexanderobenauer.com/labnotes/000/). + +[[Geoffrey Litt]] [@geoffreylitt](https://twitter.com/geoffreylitt) will talk about his browser extension [Twemex](https://twemex.app/), which makes Twitter into a better memex for exploring ideas. + +[[Dalton Banks]] [@pixelflipping](https://twitter.com/pixelflipping) will explore modeling, simulation, and control of dynamical systems as a unifying & underrated thread in tools-for-thought history. + +## Video + + + +#event diff --git a/pages/Federated Hosting by Spacebear.md b/content/notes/Federated Hosting by Spacebear.md similarity index 100% rename from pages/Federated Hosting by Spacebear.md rename to content/notes/Federated Hosting by Spacebear.md diff --git a/pages/Fedi Tips.md b/content/notes/Fedi Tips.md similarity index 100% rename from pages/Fedi Tips.md rename to content/notes/Fedi Tips.md diff --git a/pages/Fedifinder.md b/content/notes/Fedifinder.md similarity index 100% rename from pages/Fedifinder.md rename to content/notes/Fedifinder.md diff --git a/pages/Fediverse Party - Mastodon.md b/content/notes/Fediverse Party - Mastodon.md similarity index 100% rename from pages/Fediverse Party - Mastodon.md rename to content/notes/Fediverse Party - Mastodon.md diff --git a/pages/Fission Discord.md b/content/notes/Fission Discord.md similarity index 100% rename from pages/Fission Discord.md rename to content/notes/Fission Discord.md diff --git a/pages/GSheet Mastodon Open Servers English 5K+ Active Users.md b/content/notes/GSheet Mastodon Open Servers English 5K+ Active Users.md similarity index 100% rename from pages/GSheet Mastodon Open Servers English 5K+ Active Users.md rename to content/notes/GSheet Mastodon Open Servers English 5K+ Active Users.md diff --git a/pages/Geoffrey Litt.md b/content/notes/Geoffrey Litt.md similarity index 100% rename from pages/Geoffrey Litt.md rename to content/notes/Geoffrey Litt.md diff --git a/pages/GithubIssue.md b/content/notes/GithubIssue.md similarity index 100% rename from pages/GithubIssue.md rename to content/notes/GithubIssue.md diff --git a/pages/Gordon Brander.md b/content/notes/Gordon Brander.md similarity index 100% rename from pages/Gordon Brander.md rename to content/notes/Gordon Brander.md diff --git a/pages/How long can the TFTRocks Mastodon server be relied upon%3F.md b/content/notes/How long can the TFTRocks Mastodon server be relied upon%3F.md similarity index 100% rename from pages/How long can the TFTRocks Mastodon server be relied upon%3F.md rename to content/notes/How long can the TFTRocks Mastodon server be relied upon%3F.md diff --git a/pages/How to Make the Fediverse Your Own.md b/content/notes/How to Make the Fediverse Your Own.md similarity index 100% rename from pages/How to Make the Fediverse Your Own.md rename to content/notes/How to Make the Fediverse Your Own.md diff --git a/pages/How to add content to the TFTRocks Notes site.md b/content/notes/How to add content to the TFTRocks Notes site.md similarity index 100% rename from pages/How to add content to the TFTRocks Notes site.md rename to content/notes/How to add content to the TFTRocks Notes site.md diff --git a/pages/Howard Rheingold.md b/content/notes/Howard Rheingold.md similarity index 100% rename from pages/Howard Rheingold.md rename to content/notes/Howard Rheingold.md diff --git a/pages/Hunter Clarke.md b/content/notes/Hunter Clarke.md similarity index 100% rename from pages/Hunter Clarke.md rename to content/notes/Hunter Clarke.md diff --git a/pages/Hyperdraft.md b/content/notes/Hyperdraft.md similarity index 100% rename from pages/Hyperdraft.md rename to content/notes/Hyperdraft.md diff --git a/pages/Instances.Social.md b/content/notes/Instances.Social.md similarity index 100% rename from pages/Instances.Social.md rename to content/notes/Instances.Social.md diff --git a/pages/Isomorphic Git.md b/content/notes/Isomorphic Git.md similarity index 100% rename from pages/Isomorphic Git.md rename to content/notes/Isomorphic Git.md diff --git a/content/notes/Jacky Zhao.md b/content/notes/Jacky Zhao.md new file mode 100644 index 0000000..2a74019 --- /dev/null +++ b/content/notes/Jacky Zhao.md @@ -0,0 +1 @@ +https://jzhao.xyz/ \ No newline at end of file diff --git a/pages/Jakob.md b/content/notes/Jakob.md similarity index 100% rename from pages/Jakob.md rename to content/notes/Jakob.md diff --git a/pages/James Walker.md b/content/notes/James Walker.md similarity index 100% rename from pages/James Walker.md rename to content/notes/James Walker.md diff --git a/pages/January 2022 Meeting.md b/content/notes/January 2022 Meeting.md similarity index 100% rename from pages/January 2022 Meeting.md rename to content/notes/January 2022 Meeting.md diff --git a/pages/Jared Pereira.md b/content/notes/Jared Pereira.md similarity index 100% rename from pages/Jared Pereira.md rename to content/notes/Jared Pereira.md diff --git a/pages/Jess Martin.md b/content/notes/Jess Martin.md similarity index 100% rename from pages/Jess Martin.md rename to content/notes/Jess Martin.md diff --git a/pages/Joybox.md b/content/notes/Joybox.md similarity index 100% rename from pages/Joybox.md rename to content/notes/Joybox.md diff --git a/pages/July 2021 Meeting.md b/content/notes/July 2021 Meeting.md similarity index 100% rename from pages/July 2021 Meeting.md rename to content/notes/July 2021 Meeting.md diff --git a/pages/July Meeting Chat Log.md b/content/notes/July Meeting Chat Log.md similarity index 100% rename from pages/July Meeting Chat Log.md rename to content/notes/July Meeting Chat Log.md diff --git a/pages/July Meeting Recap.md b/content/notes/July Meeting Recap.md similarity index 100% rename from pages/July Meeting Recap.md rename to content/notes/July Meeting Recap.md diff --git a/pages/Kevin Lin.md b/content/notes/Kevin Lin.md similarity index 100% rename from pages/Kevin Lin.md rename to content/notes/Kevin Lin.md diff --git a/pages/Kosmik.md b/content/notes/Kosmik.md similarity index 100% rename from pages/Kosmik.md rename to content/notes/Kosmik.md diff --git a/pages/Linus Lee.md b/content/notes/Linus Lee.md similarity index 100% rename from pages/Linus Lee.md rename to content/notes/Linus Lee.md diff --git a/pages/LogSeq Publish.md b/content/notes/LogSeq Publish.md similarity index 100% rename from pages/LogSeq Publish.md rename to content/notes/LogSeq Publish.md diff --git "a/pages/LogSeq Schr\303\266dinger.md" "b/content/notes/LogSeq Schr\303\266dinger.md" similarity index 100% rename from "pages/LogSeq Schr\303\266dinger.md" rename to "content/notes/LogSeq Schr\303\266dinger.md" diff --git a/pages/Lucerne.md b/content/notes/Lucerne.md similarity index 100% rename from pages/Lucerne.md rename to content/notes/Lucerne.md diff --git a/content/notes/Luma.md b/content/notes/Luma.md new file mode 100755 index 0000000..77ebb2f --- /dev/null +++ b/content/notes/Luma.md @@ -0,0 +1,3 @@ +https://lu.ma + +#calendar #event #tool #community \ No newline at end of file diff --git a/pages/Maggie Appleton.md b/content/notes/Maggie Appleton.md similarity index 100% rename from pages/Maggie Appleton.md rename to content/notes/Maggie Appleton.md diff --git a/pages/March 2022 Meeting.md b/content/notes/March 2022 Meeting.md similarity index 100% rename from pages/March 2022 Meeting.md rename to content/notes/March 2022 Meeting.md diff --git a/pages/Mark Anderson.md b/content/notes/Mark Anderson.md similarity index 100% rename from pages/Mark Anderson.md rename to content/notes/Mark Anderson.md diff --git a/pages/Mark Bernstein.md b/content/notes/Mark Bernstein.md similarity index 100% rename from pages/Mark Bernstein.md rename to content/notes/Mark Bernstein.md diff --git a/pages/Mark Upton.md b/content/notes/Mark Upton.md similarity index 100% rename from pages/Mark Upton.md rename to content/notes/Mark Upton.md diff --git a/pages/Mastodon Resources.md b/content/notes/Mastodon Resources.md similarity index 100% rename from pages/Mastodon Resources.md rename to content/notes/Mastodon Resources.md diff --git a/pages/Mastodon.md b/content/notes/Mastodon.md similarity index 100% rename from pages/Mastodon.md rename to content/notes/Mastodon.md diff --git a/pages/May 2022 Meeting.md b/content/notes/May 2022 Meeting.md similarity index 100% rename from pages/May 2022 Meeting.md rename to content/notes/May 2022 Meeting.md diff --git a/pages/Michael Gartner.md b/content/notes/Michael Gartner.md similarity index 100% rename from pages/Michael Gartner.md rename to content/notes/Michael Gartner.md diff --git a/pages/Moa Party.md b/content/notes/Moa Party.md similarity index 100% rename from pages/Moa Party.md rename to content/notes/Moa Party.md diff --git a/pages/Monthly Meetings.md b/content/notes/Monthly Meetings.md similarity index 100% rename from pages/Monthly Meetings.md rename to content/notes/Monthly Meetings.md diff --git a/pages/November 2021 Chat Log.md b/content/notes/November 2021 Chat Log.md similarity index 100% rename from pages/November 2021 Chat Log.md rename to content/notes/November 2021 Chat Log.md diff --git a/pages/November 2021 Meeting.md b/content/notes/November 2021 Meeting.md similarity index 100% rename from pages/November 2021 Meeting.md rename to content/notes/November 2021 Meeting.md diff --git a/pages/October 2021 Chat Log.md b/content/notes/October 2021 Chat Log.md similarity index 100% rename from pages/October 2021 Chat Log.md rename to content/notes/October 2021 Chat Log.md diff --git a/content/notes/October 2021 Meeting.md b/content/notes/October 2021 Meeting.md new file mode 100755 index 0000000..e422f67 --- /dev/null +++ b/content/notes/October 2021 Meeting.md @@ -0,0 +1,36 @@ +--- +date: 2021-10-29 +--- +https://lu.ma/tftrocks-oct + +Speakers +* [[Rosano]] 'The wetware of writing and doing' about his own homegrown publishing and productivity tools. + +Lobby Chatter + +* Let's try different collaborative apps, like [[Sprout]] that Wei Wei builds + * [[Butter]] is one that [[Boris]] tried the other day +* Bandwidth is 5x more expensive than storage costs, per GB +* [[Cloudflare]] released [[Cloudflare R2]], an [[Amazon S3]] competitor with no bandwidth costs + +## Video + + + +## Notes + +* [[Rosano]] [[Wetware of Writing and Doing]] +* [[Joybox]] +* [[Hyperdraft]] +* [[Write of Passage]] by [[David Perell]] +* [[Cory Doctorow]], [[The Memex Method]] +* [[Bill Seitz]] [[POSSE]] and [[PESOS]] +* [[Bill Seitz]] [MashUp](http://webseitz.fluxent.com/wiki/MashUp) +* [[Marc-Antoine Parent]] +* [[Andy Matuschak]] [[Export considered harmful]] +* [[Linus Lee]] Twitter client, [[Lucerne]] + + +[[October 2021 Chat Log]] + +#event \ No newline at end of file diff --git a/logseq/graphs-txid.edn b/content/notes/Open Collective.md old mode 100755 new mode 100644 similarity index 100% rename from logseq/graphs-txid.edn rename to content/notes/Open Collective.md diff --git a/pages/Paul Rony.md b/content/notes/Paul Rony.md similarity index 100% rename from pages/Paul Rony.md rename to content/notes/Paul Rony.md diff --git a/pages/Paul Shen.md b/content/notes/Paul Shen.md similarity index 100% rename from pages/Paul Shen.md rename to content/notes/Paul Shen.md diff --git a/pages/People.md b/content/notes/People.md similarity index 100% rename from pages/People.md rename to content/notes/People.md diff --git a/pages/Pol Baladas.md b/content/notes/Pol Baladas.md similarity index 100% rename from pages/Pol Baladas.md rename to content/notes/Pol Baladas.md diff --git a/content/notes/Quartz.md b/content/notes/Quartz.md new file mode 100644 index 0000000..71d1482 --- /dev/null +++ b/content/notes/Quartz.md @@ -0,0 +1,7 @@ +https://quartz.jzhao.xyz/ + +> Quartz is a fast, batteries-included static-site generator that transforms Markdown content into fully functional websites. Thousands of students, developers, and teachers are [already using Quartz](https://quartz.jzhao.xyz/showcase) to publish personal notes, wikis, and [digital gardens](https://jzhao.xyz/posts/networked-thought) to the web. + +Built by [[Jacky Zhao]] + +#quartz \ No newline at end of file diff --git a/pages/Rafael Nepo.md b/content/notes/Rafael Nepo.md similarity index 100% rename from pages/Rafael Nepo.md rename to content/notes/Rafael Nepo.md diff --git a/pages/Robert Haisfield.md b/content/notes/Robert Haisfield.md similarity index 100% rename from pages/Robert Haisfield.md rename to content/notes/Robert Haisfield.md diff --git a/pages/Rosano.md b/content/notes/Rosano.md similarity index 100% rename from pages/Rosano.md rename to content/notes/Rosano.md diff --git a/pages/Ryan Atkinson.md b/content/notes/Ryan Atkinson.md similarity index 100% rename from pages/Ryan Atkinson.md rename to content/notes/Ryan Atkinson.md diff --git a/pages/Scaling Synthesis - Extending Tools for Thought.md b/content/notes/Scaling Synthesis - Extending Tools for Thought.md similarity index 100% rename from pages/Scaling Synthesis - Extending Tools for Thought.md rename to content/notes/Scaling Synthesis - Extending Tools for Thought.md diff --git a/pages/Scaling Synthesis - Multiplayer TFT.md b/content/notes/Scaling Synthesis - Multiplayer TFT.md similarity index 100% rename from pages/Scaling Synthesis - Multiplayer TFT.md rename to content/notes/Scaling Synthesis - Multiplayer TFT.md diff --git a/pages/September 2021 Meeting.md b/content/notes/September 2021 Meeting.md similarity index 100% rename from pages/September 2021 Meeting.md rename to content/notes/September 2021 Meeting.md diff --git a/pages/Sophia Xu.md b/content/notes/Sophia Xu.md similarity index 100% rename from pages/Sophia Xu.md rename to content/notes/Sophia Xu.md diff --git a/pages/Spacebear.md b/content/notes/Spacebear.md similarity index 100% rename from pages/Spacebear.md rename to content/notes/Spacebear.md diff --git a/pages/Spatial Canvases - Part 1.md b/content/notes/Spatial Canvases - Part 1.md similarity index 100% rename from pages/Spatial Canvases - Part 1.md rename to content/notes/Spatial Canvases - Part 1.md diff --git a/pages/Spatial Canvases - Part 2.md b/content/notes/Spatial Canvases - Part 2.md similarity index 100% rename from pages/Spatial Canvases - Part 2.md rename to content/notes/Spatial Canvases - Part 2.md diff --git a/pages/Spatial Canvases - Part 3.md b/content/notes/Spatial Canvases - Part 3.md similarity index 100% rename from pages/Spatial Canvases - Part 3.md rename to content/notes/Spatial Canvases - Part 3.md diff --git a/pages/Spatial Canvases - Part 4.md b/content/notes/Spatial Canvases - Part 4.md similarity index 100% rename from pages/Spatial Canvases - Part 4.md rename to content/notes/Spatial Canvases - Part 4.md diff --git a/pages/Speaker Listing.md b/content/notes/Speaker Listing.md similarity index 100% rename from pages/Speaker Listing.md rename to content/notes/Speaker Listing.md diff --git a/pages/Sprout.md b/content/notes/Sprout.md similarity index 100% rename from pages/Sprout.md rename to content/notes/Sprout.md diff --git a/pages/Taylor Hadden.md b/content/notes/Taylor Hadden.md similarity index 100% rename from pages/Taylor Hadden.md rename to content/notes/Taylor Hadden.md diff --git a/pages/Templates.md b/content/notes/Templates.md similarity index 100% rename from pages/Templates.md rename to content/notes/Templates.md diff --git a/pages/Tinderbox.md b/content/notes/Tinderbox.md similarity index 100% rename from pages/Tinderbox.md rename to content/notes/Tinderbox.md diff --git a/pages/Tony Confrey.md b/content/notes/Tony Confrey.md similarity index 100% rename from pages/Tony Confrey.md rename to content/notes/Tony Confrey.md diff --git a/content/notes/Tools For Thought Mastodon.md b/content/notes/Tools For Thought Mastodon.md new file mode 100755 index 0000000..0b09c5c --- /dev/null +++ b/content/notes/Tools For Thought Mastodon.md @@ -0,0 +1,8 @@ + +[[Boris Mann]] hosts and admins a [[Mastodon]] server for the TFTRocks community at https://toolsforthought.social + +It is open to any TFT interested people to create accounts, either by invitation from an existing member, or by [applying for an account](https://toolsforthought.social/auth/sign_up). + +Contribute to the [[Tools For Thought Rocks Open Collective]] to support running the server. + +The server was originally running at the root domain here https://toolsforthought.rocks. [More info about the move](https://commonscomputer.com/t/tft-rocks-mastodon-server-move/13) \ No newline at end of file diff --git a/pages/Tools For Thought Rocks Open Collective.md b/content/notes/Tools For Thought Rocks Open Collective.md similarity index 100% rename from pages/Tools For Thought Rocks Open Collective.md rename to content/notes/Tools For Thought Rocks Open Collective.md diff --git "a/pages/Tools for THINKing \342\200\246.md" "b/content/notes/Tools for THINKing \342\200\246.md" similarity index 100% rename from "pages/Tools for THINKing \342\200\246.md" rename to "content/notes/Tools for THINKing \342\200\246.md" diff --git a/pages/Tools for Thought App List.md b/content/notes/Tools for Thought App List.md similarity index 100% rename from pages/Tools for Thought App List.md rename to content/notes/Tools for Thought App List.md diff --git a/pages/Tools for Thought Rocks.md b/content/notes/Tools for Thought Rocks.md similarity index 100% rename from pages/Tools for Thought Rocks.md rename to content/notes/Tools for Thought Rocks.md diff --git a/pages/Tools for Thought as Cultural Practices, not Computational Objects, Maggie Appleton, April 2022 Meeting.md b/content/notes/Tools for Thought as Cultural Practices, not Computational Objects, Maggie Appleton, April 2022 Meeting.md similarity index 100% rename from pages/Tools for Thought as Cultural Practices, not Computational Objects, Maggie Appleton, April 2022 Meeting.md rename to content/notes/Tools for Thought as Cultural Practices, not Computational Objects, Maggie Appleton, April 2022 Meeting.md diff --git a/pages/Tools for Thought.md b/content/notes/Tools for Thought.md similarity index 100% rename from pages/Tools for Thought.md rename to content/notes/Tools for Thought.md diff --git a/pages/Tudor Girba.md b/content/notes/Tudor Girba.md similarity index 100% rename from pages/Tudor Girba.md rename to content/notes/Tudor Girba.md diff --git a/pages/Utilities.md b/content/notes/Utilities.md similarity index 100% rename from pages/Utilities.md rename to content/notes/Utilities.md diff --git a/pages/Volunteer Organizer Kick Off Meeting.md b/content/notes/Volunteer Organizer Kick Off Meeting.md similarity index 100% rename from pages/Volunteer Organizer Kick Off Meeting.md rename to content/notes/Volunteer Organizer Kick Off Meeting.md diff --git a/pages/Weiwei Hsu.md b/content/notes/Weiwei Hsu.md similarity index 100% rename from pages/Weiwei Hsu.md rename to content/notes/Weiwei Hsu.md diff --git a/pages/Wetware of Writing and Doing.md b/content/notes/Wetware of Writing and Doing.md similarity index 100% rename from pages/Wetware of Writing and Doing.md rename to content/notes/Wetware of Writing and Doing.md diff --git a/pages/Write of Passage.md b/content/notes/Write of Passage.md similarity index 100% rename from pages/Write of Passage.md rename to content/notes/Write of Passage.md diff --git a/pages/monthly-meeting.md b/content/notes/monthly-meeting.md similarity index 100% rename from pages/monthly-meeting.md rename to content/notes/monthly-meeting.md diff --git a/pages/Yoyogi.md b/content/notes/yoyogi.md similarity index 100% rename from pages/Yoyogi.md rename to content/notes/yoyogi.md diff --git a/docs/advanced/architecture.md b/docs/advanced/architecture.md new file mode 100644 index 0000000..33da89d --- /dev/null +++ b/docs/advanced/architecture.md @@ -0,0 +1,52 @@ +--- +title: Architecture +--- + +Quartz is a static site generator. How does it work? + +This question is best answered by tracing what happens when a user (you!) runs `npx quartz build` in the command line: + +## On the server + +1. After running `npx quartz build`, npm will look at `package.json` to find the `bin` entry for `quartz` which points at `./quartz/bootstrap-cli.mjs`. +2. This file has a [shebang]() line at the top which tells npm to execute it using Node. +3. `bootstrap-cli.mjs` is responsible for a few things: + 1. Parsing the command-line arguments using [yargs](http://yargs.js.org/). + 2. Transpiling and bundling the rest of Quartz (which is in Typescript) to regular JavaScript using [esbuild](https://esbuild.github.io/). The `esbuild` configuration here is slightly special as it also handles `.scss` file imports using [esbuild-sass-plugin v2](https://www.npmjs.com/package/esbuild-sass-plugin). Additionally, we bundle 'inline' client-side scripts (any `.inline.ts` file) that components declare using a custom `esbuild` plugin that runs another instance of `esbuild` which bundles for the browser instead of `node`. Modules of both types are imported as plain text. + 3. Running the local preview server if `--serve` is set. This starts two servers: + 1. A WebSocket server on port 3001 to handle hot-reload signals. This tracks all inbound connections and sends a 'rebuild' message a server-side change is detected (either content or configuration). + 2. An HTTP file-server on a user defined port (normally 8080) to serve the actual website files. + 4. If the `--serve` flag is set, it also starts a file watcher to detect source-code changes (e.g. anything that is `.ts`, `.tsx`, `.scss`, or packager files). On a change, we rebuild the module (step 2 above) using esbuild's [rebuild API](https://esbuild.github.io/api/#rebuild) which drastically reduces the build times. + 5. After transpiling the main Quartz build module (`quartz/build.ts`), we write it to a cache file `.quartz-cache/transpiled-build.mjs` and then dynamically import this using `await import(cacheFile)`. However, we need to be pretty smart about how to bust Node's [import cache](https://github.com/nodejs/modules/issues/307) so we add a random query string to fake Node into thinking it's a new module. This does, however, cause memory leaks so we just hope that the user doesn't hot-reload their configuration too many times in a single session :)) (it leaks about ~350kB memory on each reload). After importing the module, we then invoke it, passing in the command line arguments we parsed earlier along with a callback function to signal the client to refresh. +4. In `build.ts`, we start by installing source map support manually to account for the query string cache busting hack we introduced earlier. Then, we start processing content: + 1. Clean the output directory. + 2. Recursively glob all files in the `content` folder, respecting the `.gitignore`. + 3. Parse the Markdown files. + 1. Quartz detects the number of threads available and chooses to spawn worker threads if there are >128 pieces of content to parse (rough heuristic). If it needs to spawn workers, it will invoke esbuild again to transpile the worker script `quartz/worker.ts`. Then, a work-stealing [workerpool](https://www.npmjs.com/package/workerpool) is then created and batches of 128 files are assigned to workers. + 2. Each worker (or just the main thread if there is no concurrency) creates a [unified](https://github.com/unifiedjs/unified) parser based off of the plugins defined in the [[configuration]]. + 3. Parsing has three steps: + 1. Read the file into a [vfile](https://github.com/vfile/vfile). + 2. Applied plugin-defined text transformations over the content. + 3. Slugify the file path and store it in the data for the file. See the page on [[paths]] for more details about how path logic works in Quartz (spoiler: its complicated). + 4. Markdown parsing using [remark-parse](https://www.npmjs.com/package/remark-parse) (text to [mdast](https://github.com/syntax-tree/mdast)). + 5. Apply plugin-defined Markdown-to-Markdown transformations. + 6. Convert Markdown into HTML using [remark-rehype](https://github.com/remarkjs/remark-rehype) ([mdast](https://github.com/syntax-tree/mdast) to [hast](https://github.com/syntax-tree/hast)). + 7. Apply plugin-defined HTML-to-HTML transformations. + 4. Filter out unwanted content using plugins. + 5. Emit files using plugins. + 1. Gather all the static resources (e.g. external CSS, JS modules, etc.) each emitter plugin declares. + 2. Emitters that emit HTML files do a bit of extra work here as they need to transform the [hast](https://github.com/syntax-tree/hast) produced in the parse step to JSX. This is done using [hast-util-to-jsx-runtime](https://github.com/syntax-tree/hast-util-to-jsx-runtime) with the [Preact](https://preactjs.com/) runtime. Finally, the JSX is rendered to HTML using [preact-render-to-string](https://github.com/preactjs/preact-render-to-string) which statically renders the JSX to HTML (i.e. doesn't care about `useState`, `useEffect`, or any other React/Preact interactive bits). Here, we also do a bunch of fun stuff like assemble the page [[layout]] from `quartz.layout.ts`, assemble all the inline scripts that actually get shipped to the client, and all the transpiled styles. The bulk of this logic can be found in `quartz/components/renderPage.tsx`. Other fun things of note: + 1. CSS is minified and transformed using [Lightning CSS](https://github.com/parcel-bundler/lightningcss) to add vendor prefixes and do syntax lowering. + 2. Scripts are split into `beforeDOMLoaded` and `afterDOMLoaded` and are inserted in the `` and `` respectively. + 3. Finally, each emitter plugin is responsible for emitting and writing it's own emitted files to disk. + 6. If the `--serve` flag was detected, we also set up another file watcher to detect content changes (only `.md` files). We keep a content map that tracks the parsed AST and plugin data for each slug and update this on file changes. Newly added or modified paths are rebuilt and added to the content map. Then, all the filters and emitters are run over the resulting content map. This file watcher is debounced with a threshold of 250ms. On success, we send a client refresh signal using the passed in callback function. + +## On the client + +1. The browser opens a Quartz page and loads the HTML. The `` also links to page styles (emitted to `public/index.css`) and page-critical JS (emitted to `public/prescript.js`) +2. Then, once the body is loaded, the browser loads the non-critical JS (emitted to `public/postscript.js`) +3. Once the page is done loading, the page will then dispatch a custom synthetic browser event `"nav"`. This is used so client-side scripts declared by components can 'setup' anything that requires access to the page DOM. + 1. If the [[SPA Routing|enableSPA option]] is enabled in the [[configuration]], this `"nav"` event is also fired on any client-navigation to allow for components to unregister and reregister any event handlers and state. + 2. If it's not, we wire up the `"nav"` event to just be fired a single time after page load to allow for consistency across how state is setup across both SPA and non-SPA contexts. + +The architecture and design of the plugin system was intentionally left pretty vague here as this is described in much more depth in the guide on [[making plugins|making your own plugin]]. diff --git a/docs/advanced/creating components.md b/docs/advanced/creating components.md new file mode 100644 index 0000000..1496b15 --- /dev/null +++ b/docs/advanced/creating components.md @@ -0,0 +1,233 @@ +--- +title: Creating your own Quartz components +--- + +> [!warning] +> This guide assumes you have experience writing JavaScript and are familiar with TypeScript. + +Normally on the web, we write layout code using HTML which looks something like the following: + +```html +
+

An article header

+

Some content

+
+``` + +This piece of HTML represents an article with a leading header that says "An article header" and a paragraph that contains the text "Some content". This is combined with CSS to style the page and JavaScript to add interactivity. + +However, HTML doesn't let you create reusable templates. If you wanted to create a new page, you would need to copy and paste the above snippet and edit the header and content yourself. This isn't great if we have a lot of content on our site that shares a lot of similar layout. The smart people who created React also had similar complaints and invented the concept of Components -- JavaScript functions that return JSX -- to solve the code duplication problem. + +In effect, components allow you to write a JavaScript function that takes some data and produces HTML as an output. **While Quartz doesn't use React, it uses the same component concept to allow you to easily express layout templates in your Quartz site.** + +## An Example Component + +### Constructor + +Component files are written in `.tsx` files that live in the `quartz/components` folder. These are re-exported in `quartz/components/index.ts` so you can use them in layouts and other components more easily. + +Each component file should have a default export that satisfies the `QuartzComponentConstructor` function signature. It's a function that takes in a single optional parameter `opts` and returns a Quartz Component. The type of the parameters `opts` is defined by the interface `Options` which you as the component creator also decide. + +In your component, you can use the values from the configuration option to change the rendering behaviour inside of your component. For example, the component in the code snippet below will not render if the `favouriteNumber` option is below 0. + +```tsx {11-17} +interface Options { + favouriteNumber: number +} + +const defaultOptions: Options = { + favouriteNumber: 42, +} + +export default ((userOpts?: Options) => { + const opts = { ...userOpts, ...defaultOpts } + function YourComponent(props: QuartzComponentProps) { + if (opts.favouriteNumber < 0) { + return null + } + + return

My favourite number is {opts.favouriteNumber}

+ } + + return YourComponent +}) satisfies QuartzComponentConstructor +``` + +### Props + +The Quartz component itself (lines 11-17 highlighted above) looks like a React component. It takes in properties (sometimes called [props](https://react.dev/learn/passing-props-to-a-component)) and returns JSX. + +All Quartz components accept the same set of props: + +```tsx title="quartz/components/types.ts" +// simplified for sake of demonstration +export type QuartzComponentProps = { + fileData: QuartzPluginData + cfg: GlobalConfiguration + tree: Node + allFiles: QuartzPluginData[] + displayClass?: "mobile-only" | "desktop-only" +} +``` + +- `fileData`: Any metadata [[making plugins|plugins]] may have added to the current page. + - `fileData.slug`: slug of the current page. + - `fileData.frontmatter`: any frontmatter parsed. +- `cfg`: The `configuration` field in `quartz.config.ts`. +- `tree`: the resulting [HTML AST](https://github.com/syntax-tree/hast) after processing and transforming the file. This is useful if you'd like to render the content using [hast-util-to-jsx-runtime](https://github.com/syntax-tree/hast-util-to-jsx-runtime) (you can find an example of this in `quartz/components/pages/Content.tsx`). +- `allFiles`: Metadata for all files that have been parsed. Useful for doing page listings or figuring out the overall site structure. +- `displayClass`: a utility class that indicates a preference from the user about how to render it in a mobile or desktop setting. Helpful if you want to conditionally hide a component on mobile or desktop. + +### Styling + +Quartz components can also define a `.css` property on the actual function component which will get picked up by Quartz. This is expected to be a CSS string which can either be inlined or imported from a `.scss` file. + +Note that inlined styles **must** be plain vanilla CSS: + +```tsx {6-10} title="quartz/components/YourComponent.tsx" +export default (() => { + function YourComponent() { + return

Example Component

+ } + + YourComponent.css = ` + p.red-text { + color: red; + } + ` + + return YourComponent +}) satisfies QuartzComponentConstructor +``` + +Imported styles, however, can be from SCSS files: + +```tsx {1-2,9} title="quartz/components/YourComponent.tsx" +// assuming your stylesheet is in quartz/components/styles/YourComponent.scss +import styles from "./styles/YourComponent.scss" + +export default (() => { + function YourComponent() { + return

Example Component

+ } + + YourComponent.css = styles + return YourComponent +}) satisfies QuartzComponentConstructor +``` + +> [!warning] +> Quartz does not use CSS modules so any styles you declare here apply _globally_. If you only want it to apply to your component, make sure you use specific class names and selectors. + +### Scripts and Interactivity + +What about interactivity? Suppose you want to add an-click handler for example. Like the `.css` property on the component, you can also declare `.beforeDOMLoaded` and `.afterDOMLoaded` properties that are strings that contain the script. + +```tsx title="quartz/components/YourComponent.tsx" +export default (() => { + function YourComponent() { + return + } + + YourComponent.beforeDOM = ` + console.log("hello from before the page loads!") + ` + + YourComponent.afterDOM = ` + document.getElementById('btn').onclick = () => { + alert('button clicked!') + } + ` + return YourComponent +}) satisfies QuartzComponentConstructor +``` + +> [!hint] +> For those coming from React, Quartz components are different from React components in that it only uses JSX for templating and layout. Hooks like `useEffect`, `useState`, etc. are not rendered and other properties that accept functions like `onClick` handlers will not work. Instead, do it using a regular JS script that modifies the DOM element directly. + +As the names suggest, the `.beforeDOMLoaded` scripts are executed _before_ the page is done loading so it doesn't have access to any elements on the page. This is mostly used to prefetch any critical data. + +The `.afterDOMLoaded` script executes once the page has been completely loaded. This is a good place to setup anything that should last for the duration of a site visit (e.g. getting something saved from local storage). + +If you need to create an `afterDOMLoaded` script that depends on _page specific_ elements that may change when navigating to a new page, you can listen for the `"nav"` event that gets fired whenever a page loads (which may happen on navigation if [[SPA Routing]] is enabled). + +```ts +document.addEventListener("nav", () => { + // do page specific logic here + // e.g. attach event listeners + const toggleSwitch = document.querySelector("#switch") as HTMLInputElement + toggleSwitch.removeEventListener("change", switchTheme) + toggleSwitch.addEventListener("change", switchTheme) +}) +``` + +It is best practice to also unmount any existing event handlers to prevent memory leaks. + +#### Importing Code + +Of course, it isn't always practical (nor desired!) to write your code as a string literal in the component. + +Quartz supports importing component code through `.inline.ts` files. + +```tsx title="quartz/components/YourComponent.tsx" +// @ts-ignore: typescript doesn't know about our inline bundling system +// so we need to silence the error +import script from "./scripts/graph.inline" + +export default (() => { + function YourComponent() { + return + } + + YourComponent.afterDOM = script + return YourComponent +}) satisfies QuartzComponentConstructor +``` + +```ts title="quartz/components/scripts/graph.inline.ts" +// any imports here are bundled for the browser +import * as d3 from "d3" + +document.getElementById("btn").onclick = () => { + alert("button clicked!") +} +``` + +Additionally, like what is shown in the example above, you can import packages in `.inline.ts` files. This will be bundled by Quartz and included in the actual script. + +### Using a Component + +After creating your custom component, re-export it in `quartz/components/index.ts`: + +```ts title="quartz/components/index.ts" {4,10} +import ArticleTitle from "./ArticleTitle" +import Content from "./pages/Content" +import Darkmode from "./Darkmode" +import YourComponent from "./YourComponent" + +export { ArticleTitle, Content, Darkmode, YourComponent } +``` + +Then, you can use it like any other component in `quartz.layout.ts` via `Component.YourComponent()`. See the [[configuration#Layout|layout]] section for more details. + +As Quartz components are just functions that return React components, you can compositionally use them in other Quartz components. + +```tsx title="quartz/components/AnotherComponent.tsx" +import YourComponent from "./YourComponent" + +export default (() => { + function AnotherComponent(props: QuartzComponentProps) { + return ( +
+

It's nested!

+ +
+ ) + } + + return AnotherComponent +}) satisfies QuartzComponentConstructor +``` + +> [!hint] +> Look in `quartz/components` for more examples of components in Quartz as reference for your own components! diff --git a/docs/advanced/index.md b/docs/advanced/index.md new file mode 100644 index 0000000..4822589 --- /dev/null +++ b/docs/advanced/index.md @@ -0,0 +1,3 @@ +--- +title: "Advanced" +--- diff --git a/docs/advanced/making plugins.md b/docs/advanced/making plugins.md new file mode 100644 index 0000000..1f1616f --- /dev/null +++ b/docs/advanced/making plugins.md @@ -0,0 +1,302 @@ +--- +title: Making your own plugins +--- + +> [!warning] +> This part of the documentation will assume you have working knowledge in TypeScript and will include code snippets that describe the interface of what Quartz plugins should look like. + +Quartz's plugins are a series of transformations over content. This is illustrated in the diagram of the processing pipeline below: + +![[quartz transform pipeline.png]] + +All plugins are defined as a function that takes in a single parameter for options `type OptionType = object | undefined` and return an object that corresponds to the type of plugin it is. + +```ts +type OptionType = object | undefined +type QuartzPlugin = (opts?: Options) => QuartzPluginInstance +type QuartzPluginInstance = + | QuartzTransformerPluginInstance + | QuartzFilterPluginInstance + | QuartzEmitterPluginInstance +``` + +The following sections will go into detail for what methods can be implemented for each plugin type. Before we do that, let's clarify a few more ambiguous types: + +- `BuildCtx` is defined in `quartz/ctx.ts`. It consists of + - `argv`: The command line arguments passed to the Quartz [[build]] command + - `cfg`: The full Quartz [[configuration]] + - `allSlugs`: a list of all the valid content slugs (see [[paths]] for more information on what a `ServerSlug` is) +- `StaticResources` is defined in `quartz/resources.tsx`. It consists of + - `css`: a list of URLs for stylesheets that should be loaded + - `js`: a list of scripts that should be loaded. A script is described with the `JSResource` type which is also defined in `quartz/resources.tsx`. It allows you to define a load time (either before or after the DOM has been loaded), whether it should be a module, and either the source URL or the inline content of the script. + +## Transformers + +Transformers **map** over content, taking a Markdown file and outputting modified content or adding metadata to the file itself. + +```ts +export type QuartzTransformerPluginInstance = { + name: string + textTransform?: (ctx: BuildCtx, src: string | Buffer) => string | Buffer + markdownPlugins?: (ctx: BuildCtx) => PluggableList + htmlPlugins?: (ctx: BuildCtx) => PluggableList + externalResources?: (ctx: BuildCtx) => Partial +} +``` + +All transformer plugins must define at least a `name` field to register the plugin and a few optional functions that allow you to hook into various parts of transforming a single Markdown file. + +- `textTransform` performs a text-to-text transformation _before_ a file is parsed into the [Markdown AST](https://github.com/syntax-tree/mdast). +- `markdownPlugins` defines a list of [remark plugins](https://github.com/remarkjs/remark/blob/main/doc/plugins.md). `remark` is a tool that transforms Markdown to Markdown in a structured way. +- `htmlPlugins` defines a list of [rehype plugins](https://github.com/rehypejs/rehype/blob/main/doc/plugins.md). Similar to how `remark` works, `rehype` is a tool that transforms HTML to HTML in a structured way. +- `externalResources` defines any external resources the plugin may need to load on the client-side for it to work properly. + +Normally for both `remark` and `rehype`, you can find existing plugins that you can use to . If you'd like to create your own `remark` or `rehype` plugin, checkout the [guide to creating a plugin](https://unifiedjs.com/learn/guide/create-a-plugin/) using `unified` (the underlying AST parser and transformer library). + +A good example of a transformer plugin that borrows from the `remark` and `rehype` ecosystems is the [[Latex]] plugin: + +```ts title="quartz/plugins/transformers/latex.ts" +import remarkMath from "remark-math" +import rehypeKatex from "rehype-katex" +import rehypeMathjax from "rehype-mathjax/svg.js" +import { QuartzTransformerPlugin } from "../types" + +interface Options { + renderEngine: "katex" | "mathjax" +} + +export const Latex: QuartzTransformerPlugin = (opts?: Options) => { + const engine = opts?.renderEngine ?? "katex" + return { + name: "Latex", + markdownPlugins() { + return [remarkMath] + }, + htmlPlugins() { + if (engine === "katex") { + // if you need to pass options into a plugin, you + // can use a tuple of [plugin, options] + return [[rehypeKatex, { output: "html" }]] + } else { + return [rehypeMathjax] + } + }, + externalResources() { + if (engine === "katex") { + return { + css: ["https://cdn.jsdelivr.net/npm/katex@0.16.0/dist/katex.min.css"], + js: [ + { + src: "https://cdn.jsdelivr.net/npm/katex@0.16.7/dist/contrib/copy-tex.min.js", + loadTime: "afterDOMReady", + contentType: "external", + }, + ], + } + } else { + return {} + } + }, + } +} +``` + +Another common thing that transformer plugins will do is parse a file and add extra data for that file: + +```ts +export const AddWordCount: QuartzTransformerPlugin = () => { + return { + name: "AddWordCount", + markdownPlugins() { + return [ + () => { + return (tree, file) => { + // tree is an `mdast` root element + // file is a `vfile` + const text = file.value + const words = text.split(" ").length + file.data.wordcount = words + } + }, + ] + }, + } +} + +// tell typescript about our custom data fields we are adding +// other plugins will then also be aware of this data field +declare module "vfile" { + interface DataMap { + wordcount: number + } +} +``` + +Finally, you can also perform transformations over Markdown or HTML ASTs using the `visit` function from the `unist-util-visit` package or the `findAndReplace` function from the `mdast-util-find-and-replace` package. + +```ts +export const TextTransforms: QuartzTransformerPlugin = () => { + return { + name: "TextTransforms", + markdownPlugins() { + return [() => { + return (tree, file) => { + // replace _text_ with the italics version + findAndReplace(tree, /_(.+)_/, (_value: string, ...capture: string[]) => { + // inner is the text inside of the () of the regex + const [inner] = capture + // return an mdast node + // https://github.com/syntax-tree/mdast + return { + type: "emphasis", + children: [{ type: 'text', value: inner }] + } + }) + + // remove all links (replace with just the link content) + // match by 'type' field on an mdast node + // https://github.com/syntax-tree/mdast#link in this example + visit(tree, "link", (link: Link) => { + return { + type: "paragraph" + children: [{ type: 'text', value: link.title }] + } + }) + } + }] + } + } +} +``` + +All transformer plugins can be found under `quartz/plugins/transformers`. If you decide to write your own transformer plugin, don't forget to re-export it under `quartz/plugins/transformers/index.ts` + +A parting word: transformer plugins are quite complex so don't worry if you don't get them right away. Take a look at the built in transformers and see how they operate over content to get a better sense for how to accomplish what you are trying to do. + +## Filters + +Filters **filter** content, taking the output of all the transformers and determining what files to actually keep and what to discard. + +```ts +export type QuartzFilterPlugin = ( + opts?: Options, +) => QuartzFilterPluginInstance + +export type QuartzFilterPluginInstance = { + name: string + shouldPublish(ctx: BuildCtx, content: ProcessedContent): boolean +} +``` + +A filter plugin must define a `name` field and a `shouldPublish` function that takes in a piece of content that has been processed by all the transformers and returns a `true` or `false` depending on whether it should be passed to the emitter plugins or not. + +For example, here is the built-in plugin for removing drafts: + +```ts title="quartz/plugins/filters/draft.ts" +import { QuartzFilterPlugin } from "../types" + +export const RemoveDrafts: QuartzFilterPlugin<{}> = () => ({ + name: "RemoveDrafts", + shouldPublish(_ctx, [_tree, vfile]) { + // uses frontmatter parsed from transformers + const draftFlag: boolean = vfile.data?.frontmatter?.draft ?? false + return !draftFlag + }, +}) +``` + +## Emitters + +Emitters **reduce** over content, taking in a list of all the transformed and filtered content and creating output files. + +```ts +export type QuartzEmitterPlugin = ( + opts?: Options, +) => QuartzEmitterPluginInstance + +export type QuartzEmitterPluginInstance = { + name: string + emit( + ctx: BuildCtx, + content: ProcessedContent[], + resources: StaticResources, + emitCallback: EmitCallback, + ): Promise + getQuartzComponents(ctx: BuildCtx): QuartzComponent[] +} +``` + +An emitter plugin must define a `name` field an `emit` function and a `getQuartzComponents` function. `emit` is responsible for looking at all the parsed and filtered content and then appropriately creating files and returning a list of paths to files the plugin created. + +Creating new files can be done via regular Node [fs module](https://nodejs.org/api/fs.html) (i.e. `fs.cp` or `fs.writeFile`) or via the `emitCallback` if you are creating files that contain text. The `emitCallback` function is the 4th argument of the emit function. It's interface looks something like this: + +```ts +export type EmitCallback = (data: { + // the name of the file to emit (not including the file extension) + slug: ServerSlug + // the file extension + ext: `.${string}` | "" + // the file content to add + content: string +}) => Promise +``` + +This is a thin wrapper around writing to the appropriate output folder and ensuring that intermediate directories exist. If you choose to use the native Node `fs` APIs, ensure you emit to the `argv.output` folder as well. + +If you are creating an emitter plugin that needs to render components, there are three more things to be aware of: + +- Your component should use `getQuartzComponents` to declare a list of `QuartzComponents` that it uses to construct the page. See the page on [[creating components]] for more information. +- You can use the `renderPage` function defined in `quartz/components/renderPage.tsx` to render Quartz components into HTML. +- If you need to render an HTML AST to JSX, you can use the `toJsxRuntime` function from `hast-util-to-jsx-runtime` library. An example of this can be found in `quartz/components/pages/Content.tsx`. + +For example, the following is a simplified version of the content page plugin that renders every single page. + +```tsx title="quartz/plugins/emitters/contentPage.tsx" +export const ContentPage: QuartzEmitterPlugin = () => { + // construct the layout + const layout: FullPageLayout = { + ...sharedPageComponents, + ...defaultContentPageLayout, + pageBody: Content(), + } + const { head, header, beforeBody, pageBody, left, right, footer } = layout + return { + name: "ContentPage", + getQuartzComponents() { + return [head, ...header, ...beforeBody, pageBody, ...left, ...right, footer] + }, + async emit(ctx, content, resources, emit): Promise { + const cfg = ctx.cfg.configuration + const fps: FilePath[] = [] + const allFiles = content.map((c) => c[1].data) + for (const [tree, file] of content) { + const slug = canonicalizeServer(file.data.slug!) + const externalResources = pageResources(slug, resources) + const componentData: QuartzComponentProps = { + fileData: file.data, + externalResources, + cfg, + children: [], + tree, + allFiles, + } + + const content = renderPage(slug, componentData, opts, externalResources) + const fp = await emit({ + content, + slug: file.data.slug!, + ext: ".html", + }) + + fps.push(fp) + } + return fps + }, + } +} +``` + +Note that it takes in a `FullPageLayout` as the options. It's made by combining a `SharedLayout` and a `PageLayout` both of which are provided through the `quartz.layout.ts` file. + +> [!hint] +> Look in `quartz/plugins` for more examples of plugins in Quartz as reference for your own plugins! diff --git a/docs/advanced/paths.md b/docs/advanced/paths.md new file mode 100644 index 0000000..9455b98 --- /dev/null +++ b/docs/advanced/paths.md @@ -0,0 +1,51 @@ +--- +title: Paths in Quartz +--- + +Paths are pretty complex to reason about because, especially for a static site generator, they can come from so many places. + +A full file path to a piece of content? Also a path. What about a slug for a piece of content? Yet another path. + +It would be silly to type these all as `string` and call it a day as it's pretty common to accidentally mistake one type of path for another. Unfortunately, TypeScript does not have [nominal types](https://en.wikipedia.org/wiki/Nominal_type_system) for type aliases meaning even if you made custom types of a server-side slug or a client-slug slug, you can still accidentally assign one to another and TypeScript wouldn't catch it. + +Luckily, we can mimic nominal typing using [brands](https://www.typescriptlang.org/play#example/nominal-typing). + +```typescript +// instead of +type FullSlug = string + +// we do +type FullSlug = string & { __brand: "full" } + +// that way, the following will fail typechecking +const slug: FullSlug = "some random string" +``` + +While this prevents most typing mistakes _within_ our nominal typing system (e.g. mistaking a server slug for a client slug), it doesn't prevent us from _accidentally_ mistaking a string for a client slug when we forcibly cast it. + +Thus, we still need to be careful when casting from a string to one of these nominal types in the 'entrypoints', illustrated with hexagon shapes in the diagram below. + +The following diagram draws the relationships between all the path sources, nominal path types, and what functions in `quartz/path.ts` convert between them. + +```mermaid +graph LR + Browser{{Browser}} --> Window{{Body}} & LinkElement{{Link Element}} + Window --"getFullSlug()"--> FullSlug[Full Slug] + LinkElement --".href"--> Relative[Relative URL] + FullSlug --"simplifySlug()" --> SimpleSlug[Simple Slug] + SimpleSlug --"pathToRoot()"--> Relative + SimpleSlug --"resolveRelative()" --> Relative + MD{{Markdown File}} --> FilePath{{File Path}} & Links[Markdown links] + Links --"transformLink()"--> Relative + FilePath --"slugifyFilePath()"--> FullSlug[Full Slug] + style FullSlug stroke-width:4px +``` + +Here are the main types of slugs with a rough description of each type of path: + +- `FilePath`: a real file path to a file on disk. Cannot be relative and must have a file extension. +- `FullSlug`: cannot be relative and may not have leading or trailing slashes. It can have `index` as it's last segment. Use this wherever possible is it's the most 'general' interpretation of a slug. +- `SimpleSlug`: cannot be relative and shouldn't have `/index` as an ending or a file extension. It _can_ however have a trailing slash to indicate a folder path. +- `RelativeURL`: must start with `.` or `..` to indicate it's a relative URL. Shouldn't have `/index` as an ending or a file extension but can contain a trailing slash. + +To get a clearer picture of how these relate to each other, take a look at the path tests in `quartz/path.test.ts`. diff --git a/docs/authoring content.md b/docs/authoring content.md new file mode 100644 index 0000000..fa6eea2 --- /dev/null +++ b/docs/authoring content.md @@ -0,0 +1,48 @@ +--- +title: Authoring Content +--- + +All of the content in your Quartz should go in the `/content` folder. The content for the home page of your Quartz lives in `content/index.md`. If you've [[index#🪴 Get Started|setup Quartz]] already, this folder should already be initailized. Any Markdown in this folder will get processed by Quartz. + +It is recommended that you use [Obsidian](https://obsidian.md/) as a way to edit and maintain your Quartz. It comes with a nice editor and graphical interface to preview, edit, and link your local files and attachments. + +Got everything setup? Let's [[build]] and preview your Quartz locally! + +## Syntax + +As Quartz uses Markdown files as the main way of writing content, it fully supports Markdown syntax. By default, Quartz also ships with a few syntax extensions like [Github Flavored Markdown](https://docs.github.com/en/get-started/writing-on-github/getting-started-with-writing-and-formatting-on-github/basic-writing-and-formatting-syntax) (footnotes, strikethrough, tables, tasklists) and [Obsidian Flavored Markdown](https://help.obsidian.md/Editing+and+formatting/Obsidian+Flavored+Markdown) ([[callouts]], [[wikilinks]]). + +Additionally, Quartz also allows you to specify additional metadata in your notes called **frontmatter**. + +```md title="content/note.md" +--- +title: Example Title +draft: false +tags: + - example-tag +--- + +The rest of your content lives here. You can use **Markdown** here :) +``` + +Some common frontmatter fields that are natively supported by Quartz: + +- `title`: Title of the page. If it isn't provided, Quartz will use the name of the file as the title. +- `aliases`: Other names for this note. This is a list of strings. +- `draft`: Whether to publish the page or not. This is one way to make [[private pages|pages private]] in Quartz. +- `date`: A string representing the day the note was published. Normally uses `YYYY-MM-DD` format. + +## Syncing your Content + +When your Quartz is at a point you're happy with, you can save your changes to GitHub by doing `npx quartz sync`. + +> [!hint] Flags and options +> For full help options, you can run `npx quartz sync --help`. +> +> Most of these have sensible defaults but you can override them if you have a custom setup: +> +> - `-d` or `--directory`: the content folder. This is normally just `content` +> - `-v` or `--verbose`: print out extra logging information +> - `--commit` or `--no-commit`: whether to make a `git` commit for your changes +> - `--push` or `--no-push`: whether to push updates to your GitHub fork of Quartz +> - `--pull` or `--no-pull`: whether to try and pull in any updates from your GitHub fork (i.e. from other devices) before pushing diff --git a/docs/build.md b/docs/build.md new file mode 100644 index 0000000..1447687 --- /dev/null +++ b/docs/build.md @@ -0,0 +1,23 @@ +--- +title: "Building your Quartz" +--- + +Once you've [[index#🪴 Get Started|initialized]] Quartz, let's see what it looks like locally: + +```bash +npx quartz build --serve +``` + +This will start a local web server to run your Quartz on your computer. Open a web browser and visit `http://localhost:8080/` to view it. + +> [!hint] Flags and options +> For full help options, you can run `npx quartz build --help`. +> +> Most of these have sensible defaults but you can override them if you have a custom setup: +> +> - `-d` or `--directory`: the content folder. This is normally just `content` +> - `-v` or `--verbose`: print out extra logging information +> - `-o` or `--output`: the output folder. This is normally just `public` +> - `--serve`: run a local hot-reloading server to preview your Quartz +> - `--port`: what port to run the local preview server on +> - `--concurrency`: how many threads to use to parse notes diff --git a/docs/configuration.md b/docs/configuration.md new file mode 100644 index 0000000..047f6ca --- /dev/null +++ b/docs/configuration.md @@ -0,0 +1,82 @@ +--- +title: Configuration +--- + +Quartz is meant to be extremely configurable, even if you don't know any coding. Most of the configuration you should need can be done by just editing `quartz.config.ts` or changing [[layout|the layout]] in `quartz.layout.ts`. + +> [!tip] +> If you edit Quartz configuration using a text-editor that has TypeScript language support like VSCode, it will warn you when you you've made an error in your configuration, helping you avoid configuration mistakes! + +The configuration of Quartz can be broken down into two main parts: + +```ts title="quartz.config.ts" +const config: QuartzConfig = { + configuration: { ... }, + plugins: { ... }, +} +``` + +## General Configuration + +This part of the configuration concerns anything that can affect the whole site. The following is a list breaking down all the things you can configure: + +- `pageTitle`: title of the site. This is also used when generating the [[RSS Feed]] for your site. +- `enableSPA`: whether to enable [[SPA Routing]] on your site. +- `enablePopovers`: whether to enable [[popover previews]] on your site. +- `analytics`: what to use for analytics on your site. Values can be + - `null`: don't use analytics; + - `{ provider: 'plausible' }`: use [Plausible](https://plausible.io/), a privacy-friendly alternative to Google Analytics; or + - `{ provider: 'google', tagId: }`: use Google Analytics +- `baseUrl`: this is used for sitemaps and RSS feeds that require an absolute URL to know where the canonical 'home' of your site lives. This is normally the deployed URL of your site (e.g. `quartz.jzhao.xyz` for this site). Do not include the protocol (i.e. `https://`) or any leading or trailing slashes. + - This should also include the subpath if you are [[hosting]] on GitHub pages without a custom domain. For example, if my repository is `jackyzha0/quartz`, GitHub pages would deploy to `https://jackyzha0.github.io/quartz` and the `baseUrl` would be `jackyzha0.github.io/quartz` + - Note that Quartz 4 will avoid using this as much as possible and use relative URLs whenever it can to make sure your site works no matter _where_ you end up actually deploying it. +- `ignorePatterns`: a list of [glob]() patterns that Quartz should ignore and not search through when looking for files inside the `content` folder. See [[private pages]] for more details. +- `defaultDateType`: whether to use created, modified, or published as the default date to display on pages and page listings. +- `theme`: configure how the site looks. + - `typography`: what fonts to use. Any font available on [Google Fonts](https://fonts.google.com/) works here. + - `header`: Font to use for headers + - `code`: Font for inline and block quotes. + - `body`: Font for everything + - `colors`: controls the theming of the site. + - `light`: page background + - `lightgray`: borders + - `gray`: graph links, heavier borders + - `darkgray`: body text + - `dark`: header text and icons + - `secondary`: link colour, current [[graph view|graph]] node + - `tertiary`: hover states and visited [[graph view|graph]] nodes + - `highlight`: internal link background, highlighted text, [[syntax highlighting|highlighted lines of code]] + +## Plugins + +You can think of Quartz plugins as a series of transformations over content. + +![[quartz transform pipeline.png]] + +```ts +plugins: { + transformers: [...], + filters: [...], + emitters: [...], +} +``` + +- [[making plugins#Transformers|Transformers]] **map** over content (e.g. parsing frontmatter, generating a description) +- [[making plugins#Filters|Filters]] **filter** content (e.g. filtering out drafts) +- [[making plugins#Emitters|Emitters]] **reduce** over content (e.g. creating an RSS feed or pages that list all files with a specific tag) + +By adding, removing, and reordering plugins from the `tranformers`, `filters`, and `emitters` fields, you can customize the behaviour of Quartz. + +> [!note] +> Each node is modified by every transformer _in order_. Some transformers are position-sensitive so you may need to take special note of whether it needs come before or after any other particular plugins. + +Additionally, plugins may also have their own configuration settings that you can pass in. For example, the [[Latex]] plugin allows you to pass in a field specifying the `renderEngine` to choose between Katex and MathJax. + +```ts +transformers: [ + Plugin.FrontMatter(), // uses default options + Plugin.Latex({ renderEngine: "katex" }), // specify some options +] +``` + +If you'd like to make your own plugins, read the guide on [[making plugins]] for more information. diff --git a/docs/features/Docker Support.md b/docs/features/Docker Support.md new file mode 100644 index 0000000..cf73b7f --- /dev/null +++ b/docs/features/Docker Support.md @@ -0,0 +1,7 @@ +Quartz comes shipped with a Docker image that will allow you to preview your Quartz locally without installing Node. + +You can run the below one-liner to run Quartz in Docker. + +```sh +docker run --rm -itp 8080:8080 $(docker build -q .) +``` diff --git a/docs/features/Latex.md b/docs/features/Latex.md new file mode 100644 index 0000000..3c8f6ff --- /dev/null +++ b/docs/features/Latex.md @@ -0,0 +1,63 @@ +--- +tags: + - plugin/transformer +--- + +Quartz uses [Katex](https://katex.org/) by default to typeset both inline and block math expressions at build time. + +## Syntax + +### Block Math + +Block math can be rendered by delimiting math expression with `$$`. + +``` +$$ +f(x) = \int_{-\infty}^\infty + f\hat(\xi),e^{2 \pi i \xi x} + \,d\xi +$$ +``` + +$$ +f(x) = \int_{-\infty}^\infty + f\hat(\xi),e^{2 \pi i \xi x} + \,d\xi +$$ + +$$ +\begin{aligned} +a &= b + c \\ &= e + f \\ +\end{aligned} +$$ + +$$ +\begin{bmatrix} +1 & 2 & 3 \\ +a & b & c +\end{bmatrix} +$$ + +### Inline Math + +Similarly, inline math can be rendered by delimiting math expression with a single `$`. For example, `$e^{i\pi} = -1$` produces $e^{i\pi} = -1$ + +### Escaping symbols + +There will be cases where you may have more than one `$` in a paragraph at once which may accidentally trigger MathJax/Katex. + +To get around this, you can escape the dollar sign by doing `\$` instead. + +For example: + +- Incorrect: `I have $1 and you have $2` produces I have $1 and you have $2 +- Correct: `I have \$1 and you have \$2` produces I have \$1 and you have \$2 + +## MathJax + +In `quartz.config.ts`, you can configure Quartz to use [MathJax SVG rendering](https://docs.mathjax.org/en/latest/output/svg.html) by replacing `Plugin.Latex({ renderEngine: 'katex' })` with `Plugin.Latex({ renderEngine: 'mathjax' })` + +## Customization + +- Removing Latex support: remove all instances of `Plugin.Latex()` from `quartz.config.ts`. +- Plugin: `quartz/plugins/transformers/latex.ts` diff --git a/docs/features/Mermaid diagrams.md b/docs/features/Mermaid diagrams.md new file mode 100644 index 0000000..f14a558 --- /dev/null +++ b/docs/features/Mermaid diagrams.md @@ -0,0 +1,28 @@ +Quartz supports Mermaid which allows you to add diagrams and charts to your notes. Mermaid supports a range of diagrams, such as [flow charts](https://mermaid.js.org/syntax/flowchart.html), [sequence diagrams](https://mermaid.js.org/syntax/sequenceDiagram.html), and [timelines](https://mermaid.js.org/syntax/timeline.html). This is enabled as a part of [[Obsidian compatibility]] and can be configured and enabled/disabled from that plugin. + +By default, Quartz will render Mermaid diagrams to match the site theme. + +> [!warning] +> Wondering why Mermaid diagrams may not be showing up even if you have them enabled? You may need to reorder your plugins so that `Plugin.ObsidianFlavoredMarkdown()` is _after_ `Plugin.SyntaxHighlighting()`. + +## Syntax + +To add a Mermaid diagram, create a mermaid code block. + +```` +```mermaid +sequenceDiagram + Alice->>+John: Hello John, how are you? + Alice->>+John: John, can you hear me? + John-->>-Alice: Hi Alice, I can hear you! + John-->>-Alice: I feel great! +``` +```` + +```mermaid +sequenceDiagram + Alice->>+John: Hello John, how are you? + Alice->>+John: John, can you hear me? + John-->>-Alice: Hi Alice, I can hear you! + John-->>-Alice: I feel great! +``` diff --git a/docs/features/Obsidian compatibility.md b/docs/features/Obsidian compatibility.md new file mode 100644 index 0000000..61fcc06 --- /dev/null +++ b/docs/features/Obsidian compatibility.md @@ -0,0 +1,31 @@ +--- +tags: + - plugin/transformer +--- + +Quartz was originally designed as a tool to publish Obsidian vaults as websites. Even as the scope of Quartz has widened over time, it hasn't lost the ability to seamlessly interoperate with Obsidian. + +By default, Quartz ships with `Plugin.ObsidianFlavoredMarkdown` which is a transformer plugin that adds support for [Obsidian Flavored Markdown](https://help.obsidian.md/Editing+and+formatting/Obsidian+Flavored+Markdown). This includes support for features like [[wikilinks]] and [[Mermaid diagrams]]. + +It also ships with support for [frontmatter parsing](https://help.obsidian.md/Editing+and+formatting/Properties) with the same fields that Obsidian uses through the `Plugin.FrontMatter` transformer plugin. + +Finally, Quartz also provides `Plugin.CrawlLinks` which allows you to customize Quartz's link resolution behaviour to match Obsidian. + +## Configuration + +- Frontmatter parsing: + - Disabling: remove all instances of `Plugin.FrontMatter()` from `quartz.config.ts`. + - Customize default values for frontmatter: edit `quartz/plugins/transformers/frontmatter.ts` +- Obsidian Flavored Markdown: + - Disabling: remove all instances of `Plugin.ObsidianFlavoredMarkdown()` from `quartz.config.ts` + - Customizing features: `Plugin.ObsidianFlavoredMarkdown` has several other options to toggle on and off: + - `comments`: whether to enable `%%` style Obsidian comments. Defaults to `true` + - `highlight`: whether to enable `==` style highlights. Defaults to `true` + - `wikilinks`: whether to enable turning [[wikilinks]] into regular links. Defaults to `true` + - `callouts`: whether to enable [[callouts]]. Defaults to `true` + - `mermaid`: whether to enable [[Mermaid diagrams]]. Defaults to `true` + - `parseTags`: whether to try and parse tags in the content body. Defaults to `true` + - `enableInHtmlEmbed`: whether to try and parse Obsidian flavoured markdown in raw HTML. Defaults to `false` +- Link resolution behaviour: + - Disabling: remove all instances of `Plugin.CrawlLinks()` from `quartz.config.ts` + - Changing link resolution preference: set `markdownLinkResolution` to one of `absolute`, `relative` or `shortest` diff --git a/docs/features/OxHugo compatibility.md b/docs/features/OxHugo compatibility.md new file mode 100644 index 0000000..3143eb1 --- /dev/null +++ b/docs/features/OxHugo compatibility.md @@ -0,0 +1,39 @@ +--- +tags: + - plugin/transformer +--- + +[org-roam](https://www.orgroam.com/) is a plain-text personal knowledge management system for [emacs](https://en.wikipedia.org/wiki/Emacs). [ox-hugo](https://github.com/kaushalmodi/ox-hugo) is org exporter backend that exports `org-mode` files to [Hugo](https://gohugo.io/) compatible Markdown. + +Because the Markdown generated by ox-hugo is not pure Markdown but Hugo specific, we need to transform it to fit into Quartz. This is done by `Plugin.OxHugoFlavouredMarkdown`. Even though this [[making plugins|plugin]] was written with `ox-hugo` in mind, it should work for any Hugo specific Markdown. + +```typescript title="quartz.config.ts" +plugins: { + transformers: [ + Plugin.FrontMatter({ delims: "+++", language: "toml" }), // if toml frontmatter + // ... + Plugin.OxHugoFlavouredMarkdown(), + Plugin.GitHubFlavoredMarkdown(), + // ... + ], +}, +``` + +## Usage + +Quartz by default doesn't understand `org-roam` files as they aren't Markdown. You're responsible for using an external tool like `ox-hugo` to export the `org-roam` files as Markdown content to Quartz and managing the static assets so that they're available in the final output. + +## Configuration + +- Link resolution + - `wikilinks`: Whether to replace `{{ relref }}` with Quartz [[wikilinks]] + - `removePredefinedAnchor`: Whether to remove [pre-defined anchor set by ox-hugo](https://ox-hugo.scripter.co/doc/anchors/). +- Image handling + - `replaceFigureWithMdImg`: Whether to replace `
` with `![]()` +- Formatting + - `removeHugoShortcode`: Whether to remove hugo shortcode syntax (`{{}}`) + - `replaceOrgLatex`: Whether to replace org-mode formatting for latex fragments with what `Plugin.Latex` supports. + +> [!warning] +> +> While you can use `Plugin.OxHugoFlavoredMarkdown` and `Plugin.ObsidianFlavoredMarkdown` together, it's not recommended because it might mutate the file in unexpected ways. Use with caution. diff --git a/docs/features/RSS Feed.md b/docs/features/RSS Feed.md new file mode 100644 index 0000000..bfeb399 --- /dev/null +++ b/docs/features/RSS Feed.md @@ -0,0 +1,7 @@ +Quartz creates an RSS feed for all the content on your site by generating an `index.xml` file that RSS readers can subscribe to. Because of the RSS spec, this requires the `baseUrl` property in your [[configuration]] to be set properly for RSS readers to pick it up properly. + +## Configuration + +- Remove RSS feed: set the `enableRSS` field of `Plugin.ContentIndex` in `quartz.config.ts` to be `false`. +- Change number of entries: set the `rssLimit` field of `Plugin.ContentIndex` to be the desired value. It defaults to latest 10 items. +- Use rich HTML output in RSS: set `rssFullHtml` field of `Plugin.ContentIndex` to be `true`. diff --git a/docs/features/SPA Routing.md b/docs/features/SPA Routing.md new file mode 100644 index 0000000..3004af9 --- /dev/null +++ b/docs/features/SPA Routing.md @@ -0,0 +1,7 @@ +Single-page-app style rendering. This prevents flashes of unstyled content and improves the smoothness of Quartz. + +Under the hood, this is done by hijacking page navigations and instead fetching the HTML via a `GET` request and then diffing and selectively replacing parts of the page using [micromorph](https://github.com/natemoo-re/micromorph). This allows us to change the content of the page without fully refreshing the page, reducing the amount of content that the browser needs to load. + +## Configuration + +- Disable SPA Routing: set the `enableSPA` field of the [[configuration]] in `quartz.config.ts` to be `false`. diff --git a/docs/features/backlinks.md b/docs/features/backlinks.md new file mode 100644 index 0000000..f558f4a --- /dev/null +++ b/docs/features/backlinks.md @@ -0,0 +1,14 @@ +--- +title: Backlinks +tags: + - component +--- + +A backlink for a note is a link from another note to that note. Links in the backlink pane also feature rich [[popover previews]] if you have that feature enabled. + +## Customization + +- Removing backlinks: delete all usages of `Component.Backlinks()` from `quartz.layout.ts`. +- Component: `quartz/components/Backlinks.tsx` +- Style: `quartz/components/styles/backlinks.scss` +- Script: `quartz/components/scripts/search.inline.ts` diff --git a/docs/features/breadcrumbs.md b/docs/features/breadcrumbs.md new file mode 100644 index 0000000..94db66a --- /dev/null +++ b/docs/features/breadcrumbs.md @@ -0,0 +1,35 @@ +--- +title: "Breadcrumbs" +tags: + - component +--- + +Breadcrumbs provide a way to navigate a hierarchy of pages within your site using a list of its parent folders. + +By default, the element at the very top of your page is the breadcrumb navigation bar (can also be seen at the top on this page!). + +## Customization + +Most configuration can be done by passing in options to `Component.Breadcrumbs()`. + +For example, here's what the default configuration looks like: + +```typescript title="quartz.layout.ts" +Component.Breadcrumbs({ + spacerSymbol: ">", // symbol between crumbs + rootName: "Home", // name of first/root element + resolveFrontmatterTitle: false, // wether to resolve folder names through frontmatter titles (more computationally expensive) + hideOnRoot: true, // wether to hide breadcrumbs on root `index.md` page +}) +``` + +When passing in your own options, you can omit any or all of these fields if you'd like to keep the default value for that field. + +You can also adjust where the breadcrumbs will be displayed by adjusting the [[layout]] (moving `Component.Breadcrumbs()` up or down) + +Want to customize it even more? + +- Removing breadcrumbs: delete all usages of `Component.Breadcrumbs()` from `quartz.layout.ts`. +- Component: `quartz/components/Breadcrumbs.tsx` +- Style: `quartz/components/styles/breadcrumbs.scss` +- Script: inline at `quartz/components/Breadcrumbs.tsx` diff --git a/docs/features/callouts.md b/docs/features/callouts.md new file mode 100644 index 0000000..63051ad --- /dev/null +++ b/docs/features/callouts.md @@ -0,0 +1,86 @@ +--- +title: Callouts +tags: + - plugin/transformer +--- + +Quartz supports the same Admonition-callout syntax as Obsidian. + +This includes + +- 12 Distinct callout types (each with several aliases) +- Collapsable callouts + +``` +> [!info] Title +> This is a callout! +``` + +See [documentation on supported types and syntax here](https://help.obsidian.md/Editing+and+formatting/Callouts). + +> [!warning] +> Wondering why callouts may not be showing up even if you have them enabled? You may need to reorder your plugins so that `Plugin.ObsidianFlavoredMarkdown()` is _after_ `Plugin.SyntaxHighlighting()`. + +## Customization + +- Disable callouts: simply pass `callouts: false` to the plugin: `Plugin.ObsidianFlavoredMarkdown({ callouts: false })` +- Editing icons: `quartz/plugins/transformers/ofm.ts` + +## Showcase + +> [!info] +> Default title + +> [!question]+ Can callouts be nested? +> +> > [!todo]- Yes!, they can. +> > +> > > [!example] You can even use multiple layers of nesting. + +> [!EXAMPLE] Examples +> +> Aliases: example + +> [!note] Notes +> +> Aliases: note + +> [!abstract] Summaries +> +> Aliases: abstract, summary, tldr + +> [!info] Info +> +> Aliases: info, todo + +> [!tip] Hint +> +> Aliases: tip, hint, important + +> [!success] Success +> +> Aliases: success, check, done + +> [!question] Question +> +> Aliases: question, help, faq + +> [!warning] Warning +> +> Aliases: warning, caution, attention + +> [!failure] Failure +> +> Aliases: failure, fail, missing + +> [!danger] Error +> +> Aliases: danger, error + +> [!bug] Bug +> +> Aliases: bug + +> [!quote] Quote +> +> Aliases: quote, cite diff --git a/docs/features/darkmode.md b/docs/features/darkmode.md new file mode 100644 index 0000000..dfa2314 --- /dev/null +++ b/docs/features/darkmode.md @@ -0,0 +1,14 @@ +--- +title: "Darkmode" +tags: + - component +--- + +Quartz supports darkmode out of the box that respects the user's theme preference. Any future manual toggles of the darkmode switch will be saved in the browser's local storage so it can be persisted across future page loads. + +## Customization + +- Removing darkmode: delete all usages of `Component.Darkmode()` from `quartz.layout.ts`. +- Component: `quartz/components/Darkmode.tsx` +- Style: `quartz/components/styles/darkmode.scss` +- Script: `quartz/components/scripts/darkmode.inline.ts` diff --git a/docs/features/explorer.md b/docs/features/explorer.md new file mode 100644 index 0000000..fd656a8 --- /dev/null +++ b/docs/features/explorer.md @@ -0,0 +1,245 @@ +--- +title: "Explorer" +tags: + - component +--- + +Quartz features an explorer that allows you to navigate all files and folders on your site. It supports nested folders and is highly customizable. + +By default, it shows all folders and files on your page. To display the explorer in a different spot, you can edit the [[layout]]. + +Display names for folders get determined by the `title` frontmatter field in `folder/index.md` (more detail in [[authoring content | Authoring Content]]). If this file does not exist or does not contain frontmatter, the local folder name will be used instead. + +> [!info] +> The explorer uses local storage by default to save the state of your explorer. This is done to ensure a smooth experience when navigating to different pages. +> +> To clear/delete the explorer state from local storage, delete the `fileTree` entry (guide on how to delete a key from local storage in chromium based browsers can be found [here](https://docs.devolutions.net/kb/general-knowledge-base/clear-browser-local-storage/clear-chrome-local-storage/)). You can disable this by passing `useSavedState: false` as an argument. + +## Customization + +Most configuration can be done by passing in options to `Component.Explorer()`. + +For example, here's what the default configuration looks like: + +```typescript title="quartz.layout.ts" +Component.Explorer({ + title: "Explorer", // title of the explorer component + folderClickBehavior: "collapse", // what happens when you click a folder ("link" to navigate to folder page on click or "collapse" to collapse folder on click) + folderDefaultState: "collapsed", // default state of folders ("collapsed" or "open") + useSavedState: true, // wether to use local storage to save "state" (which folders are opened) of explorer + // Sort order: folders first, then files. Sort folders and files alphabetically + sortFn: (a, b) => { + ... // default implementation shown later + }, + filterFn: filterFn: (node) => node.name !== "tags", // filters out 'tags' folder + mapFn: undefined, + // what order to apply functions in + order: ["filter", "map", "sort"], +}) +``` + +When passing in your own options, you can omit any or all of these fields if you'd like to keep the default value for that field. + +Want to customize it even more? + +- Removing table of contents: remove `Component.Explorer()` from `quartz.layout.ts` + - (optional): After removing the explorer component, you can move the [[table of contents | Table of Contents]] component back to the `left` part of the layout +- Changing `sort`, `filter` and `map` behavior: explained in [[#Advanced customization]] +- Component: + - Wrapper (Outer component, generates file tree, etc): `quartz/components/Explorer.tsx` + - Explorer node (recursive, either a folder or a file): `quartz/components/ExplorerNode.tsx` +- Style: `quartz/components/styles/explorer.scss` +- Script: `quartz/components/scripts/explorer.inline.ts` + +## Advanced customization + +This component allows you to fully customize all of its behavior. You can pass a custom `sort`, `filter` and `map` function. +All functions you can pass work with the `FileNode` class, which has the following properties: + +```ts title="quartz/components/ExplorerNode.tsx" {2-5} +export class FileNode { + children: FileNode[] // children of current node + name: string // last part of slug + displayName: string // what actually should be displayed in the explorer + file: QuartzPluginData | null // set if node is a file, see `QuartzPluginData` for more detail + depth: number // depth of current node + + ... // rest of implementation +} +``` + +Every function you can pass is optional. By default, only a `sort` function will be used: + +```ts title="Default sort function" +// Sort order: folders first, then files. Sort folders and files alphabetically +Component.Explorer({ + sortFn: (a, b) => { + if ((!a.file && !b.file) || (a.file && b.file)) { + // sensitivity: "base": Only strings that differ in base letters compare as unequal. Examples: a ≠ b, a = á, a = A + // numeric: true: Whether numeric collation should be used, such that "1" < "2" < "10" + return a.displayName.localeCompare(b.displayName, undefined, { + numeric: true, + sensitivity: "base", + }) + } + if (a.file && !b.file) { + return 1 + } else { + return -1 + } + }, +}) +``` + +--- + +You can pass your own functions for `sortFn`, `filterFn` and `mapFn`. All functions will be executed in the order provided by the `order` option (see [[#Customization]]). These functions behave similarly to their `Array.prototype` counterpart, except they modify the entire `FileNode` tree in place instead of returning a new one. + +For more information on how to use `sort`, `filter` and `map`, you can check [Array.prototype.sort()](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/sort), [Array.prototype.filter()](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/filter) and [Array.prototype.map()](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/map). + +Type definitions look like this: + +```ts +sortFn: (a: FileNode, b: FileNode) => number +filterFn: (node: FileNode) => boolean +mapFn: (node: FileNode) => void +``` + +> [!tip] +> You can check if a `FileNode` is a folder or a file like this: +> +> ```ts +> if (node.file) { +> // node is a file +> } else { +> // node is a folder +> } +> ``` + +## Basic examples + +These examples show the basic usage of `sort`, `map` and `filter`. + +### Use `sort` to put files first + +Using this example, the explorer will alphabetically sort everything, but put all **files** above all **folders**. + +```ts title="quartz.layout.ts" +Component.Explorer({ + sortFn: (a, b) => { + if ((!a.file && !b.file) || (a.file && b.file)) { + return a.displayName.localeCompare(b.displayName) + } + if (a.file && !b.file) { + return -1 + } else { + return 1 + } + }, +}) +``` + +### Change display names (`map`) + +Using this example, the display names of all `FileNodes` (folders + files) will be converted to full upper case. + +```ts title="quartz.layout.ts" +Component.Explorer({ + mapFn: (node) => { + node.displayName = node.displayName.toUpperCase() + }, +}) +``` + +### Remove list of elements (`filter`) + +Using this example, you can remove elements from your explorer by providing an array of folders/files using the `omit` set. + +```ts title="quartz.layout.ts" +Component.Explorer({ + filterFn: (node) => { + // set containing names of everything you want to filter out + const omit = new Set(["authoring content", "tags", "hosting"]) + return !omit.has(node.name.toLowerCase()) + }, +}) +``` + +You can customize this by changing the entries of the `omit` set. Simply add all folder or file names you want to remove. + +### Show every element in explorer + +To override the default filter function that removes the `tags` folder from the explorer, you can set the filter function to `undefined`. + +```ts title="quartz.layout.ts" +Component.Explorer({ + filterFn: undefined, // apply no filter function, every file and folder will visible +}) +``` + +## Advanced examples + +### Add emoji prefix + +To add emoji prefixes (📁 for folders, 📄 for files), you could use a map function like this: + +```ts title="quartz.layout.ts" +Component.Explorer({ + mapFn: (node) => { + // dont change name of root node + if (node.depth > 0) { + // set emoji for file/folder + if (node.file) { + node.displayName = "📄 " + node.displayName + } else { + node.displayName = "📁 " + node.displayName + } + } + }, +}) +``` + +### Putting it all together + +In this example, we're going to customize the explorer by using functions from examples above to [[#Add emoji prefix | add emoji prefixes]], [[#remove-list-of-elements-filter| filter out some folders]] and [[#use-sort-to-put-files-first | sort with files above folders]]. + +```ts title="quartz.layout.ts" +Component.Explorer({ + filterFn: sampleFilterFn, + mapFn: sampleMapFn, + sortFn: sampleSortFn, + order: ["filter", "sort", "map"], +}) +``` + +Notice how we customized the `order` array here. This is done because the default order applies the `sort` function last. While this normally works well, it would cause unintended behavior here, since we changed the first characters of all display names. In our example, `sort` would be applied based off the emoji prefix instead of the first _real_ character. + +To fix this, we just changed around the order and apply the `sort` function before changing the display names in the `map` function. + +> [!tip] +> When writing more complicated functions, the `layout` file can start to look very cramped. +> You can fix this by defining your functions in another file. +> +> ```ts title="functions.ts" +> import { Options } from "./quartz/components/ExplorerNode" +> export const mapFn: Options["mapFn"] = (node) => { +> // implement your function here +> } +> export const filterFn: Options["filterFn"] = (node) => { +> // implement your function here +> } +> export const sortFn: Options["sortFn"] = (a, b) => { +> // implement your function here +> } +> ``` +> +> You can then import them like this: +> +> ```ts title="quartz.layout.ts" +> import { mapFn, filterFn, sortFn } from "./functions.ts" +> Component.Explorer({ +> mapFn: mapFn, +> filterFn: filterFn, +> sortFn: sortFn, +> }) +> ``` diff --git a/docs/features/folder and tag listings.md b/docs/features/folder and tag listings.md new file mode 100644 index 0000000..dfde7c2 --- /dev/null +++ b/docs/features/folder and tag listings.md @@ -0,0 +1,32 @@ +--- +title: Folder and Tag Listings +tags: + - plugin/emitter +--- + +Quartz creates listing pages for any folders and tags you have. + +## Folder Listings + +Quartz will generate an index page for all the pages under that folder. This includes any content that is multiple levels deep. + +Additionally, Quartz will also generate pages for subfolders. Say you have a note in a nested folder `content/abc/def/note.md`. Then, Quartz would generate a page for all the notes under `abc` _and_ a page for all the notes under `abc/def`. + +By default, Quartz will title the page `Folder: ` and no description. You can override this by creating an `index.md` file in the folder with the `title` [[authoring content#Syntax|frontmatter]] field. Any content you write in this file will also be used in the description of the folder. + +For example, for the folder `content/posts`, you can add another file `content/posts/index.md` to add a specific description for it. + +## Tag Listings + +Quartz will also create an index page for each unique tag in your vault and render a list of all notes with that tag. + +Quartz also supports tag hierarchies as well (e.g. `plugin/emitter`) and will also render a separate tag page for each layer of the tag hierarchy. It will also create a default global tag index page at `/tags` that displays a list of all the tags in your Quartz. + +Like folder listings, you can also provide a description and title for a tag page by creating a file for each tag. For example, if you wanted to create a custom description for the #component tag, you would create a file at `content/tags/component.md` with a title and description. + +## Customization + +The layout for both the folder and content pages can be customized. By default, they use the `defaultListPageLayout` in `quartz.layouts.ts`. If you'd like to make more involved changes to the layout and don't mind editing some [[creating components|Quartz components]], you can take a look at `quartz/components/pages/FolderContent.tsx` and `quartz/components/pages/TagContent.tsx` respectively. + +- Removing folder listings: remove `Plugin.FolderPage()` from `emitters` in `quartz.config.ts` +- Removing tag listings: remove `Plugin.TagPage()` from `emitters` in `quartz.config.ts` diff --git a/docs/features/full-text search.md b/docs/features/full-text search.md new file mode 100644 index 0000000..85ec030 --- /dev/null +++ b/docs/features/full-text search.md @@ -0,0 +1,30 @@ +--- +title: Full-text Search +tags: + - component +--- + +Full-text search in Quartz is powered by [Flexsearch](https://github.com/nextapps-de/flexsearch). It's fast enough to return search results in under 10ms for Quartzs as large as half a million words. + +It can be opened by either clicking on the search bar or pressing `⌘`/`ctrl` + `K`. The top 5 search results are shown on each query. Matching subterms are highlighted and the most relevant 30 words are excerpted. Clicking on a search result will navigate to that page. + +To search content by tags, you can either press `⌘`/`ctrl` + `shift` + `K` or start your query with `#` (e.g. `#components`). + +This component is also keyboard accessible: Tab and Shift+Tab will cycle forward and backward through search results and Enter will navigate to the highlighted result (first result by default). You are also able to navigate search results using `ArrowUp` and `ArrowDown`. + +> [!info] +> Search requires the `ContentIndex` emitter plugin to be present in the [[configuration]]. + +### Indexing Behaviour + +By default, it indexes every page on the site with **Markdown syntax removed**. This means link URLs for instance are not indexed. + +It properly tokenizes Chinese, Korean, and Japenese characters and constructs separate indexes for the title, content and tags, weighing title matches above content matches. + +## Customization + +- Removing search: delete all usages of `Component.Search()` from `quartz.layout.ts`. +- Component: `quartz/components/Search.tsx` +- Style: `quartz/components/styles/search.scss` +- Script: `quartz/components/scripts/search.inline.ts` + - You can edit `contextWindowWords`, `numSearchResults` or `numTagResults` to suit your needs diff --git a/docs/features/graph view.md b/docs/features/graph view.md new file mode 100644 index 0000000..c7ddb03 --- /dev/null +++ b/docs/features/graph view.md @@ -0,0 +1,59 @@ +--- +title: "Graph View" +tags: + - component +--- + +Quartz features a graph-view that can show both a local graph view and a global graph view. + +- The local graph view shows files that either link to the current file or are linked from the current file. In other words, it shows all notes that are _at most_ one hop away. +- The global graph view can be toggled by clicking the graph icon on the top-right of the local graph view. It shows _all_ the notes in your graph and how they connect to each other. + +By default, the node radius is proportional to the total number of incoming and outgoing internal links from that file. + +Additionally, similar to how browsers highlight visited links a different colour, the graph view will also show nodes that you have visited in a different colour. + +> [!info] +> Graph View requires the `ContentIndex` emitter plugin to be present in the [[configuration]]. + +## Customization + +Most configuration can be done by passing in options to `Component.Graph()`. + +For example, here's what the default configuration looks like: + +```typescript title="quartz.layout.ts" +Component.Graph({ + localGraph: { + drag: true, // whether to allow panning the view around + zoom: true, // whether to allow zooming in and out + depth: 1, // how many hops of notes to display + scale: 1.1, // default view scale + repelForce: 0.5, // how much nodes should repel each other + centerForce: 0.3, // how much force to use when trying to center the nodes + linkDistance: 30, // how long should the links be by default? + fontSize: 0.6, // what size should the node labels be? + opacityScale: 1, // how quickly do we fade out the labels when zooming out? + }, + globalGraph: { + drag: true, + zoom: true, + depth: -1, + scale: 0.9, + repelForce: 0.5, + centerForce: 0.3, + linkDistance: 30, + fontSize: 0.6, + opacityScale: 1, + }, +}) +``` + +When passing in your own options, you can omit any or all of these fields if you'd like to keep the default value for that field. + +Want to customize it even more? + +- Removing graph view: delete all usages of `Component.Graph()` from `quartz.layout.ts`. +- Component: `quartz/components/Graph.tsx` +- Style: `quartz/components/styles/graph.scss` +- Script: `quartz/components/scripts/graph.inline.ts` diff --git a/docs/features/index.md b/docs/features/index.md new file mode 100644 index 0000000..2997b3a --- /dev/null +++ b/docs/features/index.md @@ -0,0 +1,3 @@ +--- +title: Feature List +--- diff --git a/docs/features/popover previews.md b/docs/features/popover previews.md new file mode 100644 index 0000000..f882224 --- /dev/null +++ b/docs/features/popover previews.md @@ -0,0 +1,15 @@ +--- +title: Popover Previews +--- + +Like Wikipedia, when you hover over a link in Quartz, there is a popup of a page preview that you can scroll to see the entire content. Links to headers will also scroll the popup to show that specific header in view. + +By default, Quartz only fetches previews for pages inside your vault due to [CORS](https://developer.mozilla.org/en-US/docs/Web/HTTP/CORS). It does this by selecting all HTML elements with the `popover-hint` class. For most pages, this includes the page title, page metadata like words and time to read, tags, and the actual page content. + +When [[creating components|creating your own components]], you can include this `popover-hint` class to also include it in the popover. + +## Configuration + +- Remove popovers: set the `enablePopovers` field in `quartz.config.ts` to be `false`. +- Style: `quartz/components/styles/popover.scss` +- Script: `quartz/components/scripts/popover.inline.ts` diff --git a/docs/features/private pages.md b/docs/features/private pages.md new file mode 100644 index 0000000..5c3940b --- /dev/null +++ b/docs/features/private pages.md @@ -0,0 +1,27 @@ +--- +title: Private Pages +tags: + - plugin/filter +--- + +There may be some notes you want to avoid publishing as a website. Quartz supports this through two mechanisms which can be used in conjunction: + +## Filter Plugins + +[[making plugins#Filters|Filter plugins]] are plugins that filter out content based off of certain criteria. By default, Quartz uses the `Plugin.RemoveDrafts` plugin which filters out any note that has `draft: true` in the frontmatter. + +If you'd like to only publish a select number of notes, you can instead use `Plugin.ExplicitPublish` which will filter out all notes except for any that have `publish: true` in the frontmatter. + +## `ignorePatterns` + +This is a field in `quartz.config.ts` under the main [[configuration]] which allows you to specify a list of patterns to effectively exclude from parsing all together. Any valid [glob](https://github.com/mrmlnc/fast-glob#pattern-syntax) pattern works here. + +Common examples include: + +- `some/folder`: exclude the entire of `some/folder` +- `*.md`: exclude all files with a `.md` extension +- `!*.md` exclude all files that _don't_ have a `.md` extension +- `**/private`: exclude any files or folders named `private` at any level of nesting + +> [!warning] +> Marking something as private via either a plugin or through the `ignorePatterns` pattern will only prevent a page from being included in the final built site. If your GitHub repository is public, also be sure to include an ignore for those in the `.gitignore` of your Quartz. See the `git` [documentation](https://git-scm.com/docs/gitignore#_pattern_format) for more information. diff --git a/docs/features/recent notes.md b/docs/features/recent notes.md new file mode 100644 index 0000000..439d6d0 --- /dev/null +++ b/docs/features/recent notes.md @@ -0,0 +1,16 @@ +--- +title: Recent Notes +tags: component +--- + +Quartz can generate a list of recent notes for based on some filtering and sorting criteria. Though this component isn't included in any [[layout]] by default, you can add it by using `Component.RecentNotes`. + +## Customization + +- Changing the title from "Recent notes": pass in an additional parameter to `Component.RecentNotes({ title: "Recent writing" })` +- Changing the number of recent notes: pass in an additional parameter to `Component.RecentNotes({ limit: 5 })` +- Show a 'see more' link: pass in an additional parameter to `Component.RecentNotes({ linkToMore: "tags/components" })`. This field should be a full slug to a page that exists. +- Customize filtering: pass in an additional parameter to `Component.RecentNotes({ filter: someFilterFunction })`. The filter function should be a function that has the signature `(f: QuartzPluginData) => boolean`. +- Customize sorting: pass in an additional parameter to `Component.RecentNotes({ sort: someSortFunction })`. By default, Quartz will sort by date and then tie break lexographically. The sort function should be a function that has the signature `(f1: QuartzPluginData, f2: QuartzPluginData) => number`. See `byDateAndAlphabetical` in `quartz/components/PageList.tsx` for an example. +- Component: `quartz/components/RecentNotes.tsx` +- Style: `quartz/components/styles/recentNotes.scss` diff --git a/docs/features/syntax highlighting.md b/docs/features/syntax highlighting.md new file mode 100644 index 0000000..68436c2 --- /dev/null +++ b/docs/features/syntax highlighting.md @@ -0,0 +1,135 @@ +--- +title: Syntax Highlighting +tags: + - plugin/transformer +--- + +Syntax highlighting in Quartz is completely done at build-time. This means that Quartz only ships pre-calculated CSS to highlight the right words so there is no heavy client-side bundle that does the syntax highlighting. + +And, unlike some client-side highlighters, it has a full TextMate parser grammar instead of using Regexes, allowing for highly accurate code highlighting. + +In short, it generates HTML that looks exactly like your code in an editor like VS Code. Under the hood, it's powered by [Rehype Pretty Code](https://rehype-pretty-code.netlify.app/) which uses [Shiki](https://github.com/shikijs/shiki). + +> [!warning] +> Syntax highlighting does have an impact on build speed if you have a lot of code snippets in your notes. + +## Formatting + +Text inside `backticks` on a line will be formatted like code. + +```` +```ts +export function trimPathSuffix(fp: string): string { + fp = clientSideSlug(fp) + let [cleanPath, anchor] = fp.split("#", 2) + anchor = anchor === undefined ? "" : "#" + anchor + + return cleanPath + anchor +} +``` +```` + +```ts +export function trimPathSuffix(fp: string): string { + fp = clientSideSlug(fp) + let [cleanPath, anchor] = fp.split("#", 2) + anchor = anchor === undefined ? "" : "#" + anchor + + return cleanPath + anchor +} +``` + +### Titles + +Add a file title to your code block, with text inside double quotes (`""`): + +```` +```js title="..." + +``` +```` + +```ts title="quartz/path.ts" +export function trimPathSuffix(fp: string): string { + fp = clientSideSlug(fp) + let [cleanPath, anchor] = fp.split("#", 2) + anchor = anchor === undefined ? "" : "#" + anchor + + return cleanPath + anchor +} +``` + +### Line highlighting + +Place a numeric range inside `{}`. + +```` +```js {1-3,4} + +``` +```` + +```ts {2-3,6} +export function trimPathSuffix(fp: string): string { + fp = clientSideSlug(fp) + let [cleanPath, anchor] = fp.split("#", 2) + anchor = anchor === undefined ? "" : "#" + anchor + + return cleanPath + anchor +} +``` + +### Word highlighting + +A series of characters, like a literal regex. + +```` +```js /useState/ +const [age, setAge] = useState(50); +const [name, setName] = useState('Taylor'); +``` +```` + +```js /useState/ +const [age, setAge] = useState(50) +const [name, setName] = useState("Taylor") +``` + +### Line numbers + +Syntax highlighting has line numbers configured automatically. If you want to start line numbers at a specific number, use `showLineNumbers{number}`: + +```` +```js showLineNumbers{number} + +``` +```` + +```ts showLineNumbers{20} +export function trimPathSuffix(fp: string): string { + fp = clientSideSlug(fp) + let [cleanPath, anchor] = fp.split("#", 2) + anchor = anchor === undefined ? "" : "#" + anchor + + return cleanPath + anchor +} +``` + +### Escaping code blocks + +You can format a codeblock inside of a codeblock by wrapping it with another level of backtick fences that has one more backtick than the previous fence. + +````` +```` +```js /useState/ +const [age, setAge] = useState(50); +const [name, setName] = useState('Taylor'); +``` +```` +````` + +## Customization + +- Removing syntax highlighting: delete all usages of `Plugin.SyntaxHighlighting()` from `quartz.config.ts`. +- Style: By default, Quartz uses derivatives of the GitHub light and dark themes. You can customize the colours in the `quartz/styles/syntax.scss` file. +- Plugin: `quartz/plugins/transformers/syntax.ts` diff --git a/docs/features/table of contents.md b/docs/features/table of contents.md new file mode 100644 index 0000000..a66c850 --- /dev/null +++ b/docs/features/table of contents.md @@ -0,0 +1,25 @@ +--- +title: "Table of Contents" +tags: + - component + - plugin/transformer +--- + +Quartz can automatically generate a table of contents from a list of headings on each page. It will also show you your current scroll position on the site by marking headings you've scrolled through with a different colour. + +By default, it will show all headers from H1 (`# Title`) all the way to H3 (`### Title`) and will only show the table of contents if there is more than 1 header on the page. +You can also hide the table of contents on a page by adding `showToc: false` to the frontmatter for that page. + +> [!info] +> This feature requires both `Plugin.TableOfContents` in your `quartz.config.ts` and `Component.TableOfContents` in your `quartz.layout.ts` to function correctly. + +## Customization + +- Removing table of contents: remove all instances of `Plugin.TableOfContents()` from `quartz.config.ts`. and `Component.TableOfContents()` from `quartz.layout.ts` +- Changing the max depth: pass in a parameter to `Plugin.TableOfContents({ maxDepth: 4 })` +- Changing the minimum number of entries in the Table of Contents before it renders: pass in a parameter to `Plugin.TableOfContents({ minEntries: 3 })` +- Component: `quartz/components/TableOfContents.tsx` +- Style: + - Modern (default): `quartz/components/styles/toc.scss` + - Legacy Quartz 3 style: `quartz/components/styles/legacyToc.scss` +- Script: `quartz/components/scripts/toc.inline.ts` diff --git a/docs/features/upcoming features.md b/docs/features/upcoming features.md new file mode 100644 index 0000000..76adda0 --- /dev/null +++ b/docs/features/upcoming features.md @@ -0,0 +1,23 @@ +--- +draft: true +--- + +## high priority backlog + +- static dead link detection +- block links: https://help.obsidian.md/Linking+notes+and+files/Internal+links#Link+to+a+block+in+a+note +- note/header/block transcludes: https://help.obsidian.md/Linking+notes+and+files/Embedding+files +- docker support + +## misc backlog + +- breadcrumbs component +- cursor chat extension +- https://giscus.app/ extension +- sidenotes? https://github.com/capnfabs/paperesque +- direct match in search using double quotes +- https://help.obsidian.md/Advanced+topics/Using+Obsidian+URI +- audio/video embed styling +- Canvas +- parse all images in page: use this for page lists if applicable? +- CV mode? with print stylesheet diff --git a/docs/features/wikilinks.md b/docs/features/wikilinks.md new file mode 100644 index 0000000..50bbb1b --- /dev/null +++ b/docs/features/wikilinks.md @@ -0,0 +1,16 @@ +--- +title: Wikilinks +--- + +Wikilinks were pioneered by earlier internet wikis to make it easier to write links across pages without needing to write Markdown or HTML links each time. + +Quartz supports Wikilinks by default and these links are resolved by Quartz using `Plugin.CrawlLinks`. See the [Obsidian Help page on Internal Links](https://help.obsidian.md/Linking+notes+and+files/Internal+links) for more information on Wikilink syntax. + +This is enabled as a part of [[Obsidian compatibility]] and can be configured and enabled/disabled from that plugin. + +## Syntax + +- `[[Path to file]]`: produces a link to `Path to file.md` (or `Path-to-file.md`) with the text `Path to file` +- `[[Path to file | Here's the title override]]`: produces a link to `Path to file.md` with the text `Here's the title override` +- `[[Path to file#Anchor]]`: produces a link to the anchor `Anchor` in the file `Path to file.md` +- `[[Path to file#^block-ref]]`: produces a link to the specific block `block-ref` in the file `Path to file.md` diff --git a/docs/hosting.md b/docs/hosting.md new file mode 100644 index 0000000..01d130f --- /dev/null +++ b/docs/hosting.md @@ -0,0 +1,168 @@ +--- +title: Hosting +--- + +Quartz effectively turns your Markdown files and other resources into a bundle of HTML, JS, and CSS files (a website!). + +However, if you'd like to publish your site to the world, you need a way to host it online. This guide will detail how to deploy with either GitHub Pages or Cloudflare pages but any service that allows you to deploy static HTML should work as well (e.g. Netlify, Replit, etc.) + +> [!hint] +> Some Quartz features (like [[RSS Feed]] and sitemap generation) require `baseUrl` to be configured properly in your [[configuration]] to work properly. Make sure you set this before deploying! + +## Cloudflare Pages + +1. Log in to the [Cloudflare dashboard](https://dash.cloudflare.com/) and select your account. +2. In Account Home, select **Workers & Pages** > **Create application** > **Pages** > **Connect to Git**. +3. Select the new GitHub repository that you created and, in the **Set up builds and deployments** section, provide the following information: + +| Configuration option | Value | +| ---------------------- | ------------------ | +| Production branch | `v4` | +| Framework preset | `None` | +| Build command | `npx quartz build` | +| Build output directory | `public` | + +Press "Save and deploy" and Cloudflare should have a deployed version of your site in about a minute. Then, every time you sync your Quartz changes to GitHub, your site should be updated. + +To add a custom domain, check out [Cloudflare's documentation](https://developers.cloudflare.com/pages/platform/custom-domains/). + +## GitHub Pages + +Like Quartz 3, you can deploy the site generated by Quartz 4 via GitHub Pages. + +> [!warning] +> Quartz generates files in the format of `file.html` instead of `file/index.html` which means the trailing slashes for _non-folder paths_ are dropped. As GitHub pages does not do this redirect, this may cause existing links to your site that use trailing slashes to break. If not breaking existing links is important to you, consider using [[#Cloudflare Pages]]. + +In your local Quartz, create a new file `quartz/.github/workflows/deploy.yml`. + +```yaml title="quartz/.github/workflows/deploy.yml" +name: Deploy Quartz site to GitHub Pages + +on: + push: + branches: + - v4 + +permissions: + contents: read + pages: write + id-token: write + +concurrency: + group: "pages" + cancel-in-progress: false + +jobs: + build: + runs-on: ubuntu-22.04 + steps: + - uses: actions/checkout@v3 + with: + fetch-depth: 0 # Fetch all history for git info + - uses: actions/setup-node@v3 + with: + node-version: 18.14 + - name: Install Dependencies + run: npm ci + - name: Build Quartz + run: npx quartz build + - name: Upload artifact + uses: actions/upload-pages-artifact@v2 + with: + path: public + + deploy: + needs: build + environment: + name: github-pages + url: ${{ steps.deployment.outputs.page_url }} + runs-on: ubuntu-latest + steps: + - name: Deploy to GitHub Pages + id: deployment + uses: actions/deploy-pages@v2 +``` + +Then: + +1. Head to "Settings" tab of your forked repository and in the sidebar, click "Pages". Under "Source", select "GitHub Actions". +2. Commit these changes by doing `npx quartz sync`. This should deploy your site to `.github.io/`. + +> [!hint] +> If you get an error about not being allowed to deploy to `github-pages` due to environment protection rules, make sure you remove any existing GitHub pages environments. +> +> You can do this by going to your Settings page on your GitHub fork and going to the Environments tab and pressing the trash icon. The GitHub action will recreate the environment for you correctly the next time you sync your Quartz. + +### Custom Domain + +Here's how to add a custom domain to your GitHub pages deployment. + +1. Head to the "Settings" tab of your forked repository. +2. In the "Code and automation" section of the sidebar, click "Pages". +3. Under "Custom Domain", type your custom domain and click "Save". +4. This next step depends on whether you are using an apex domain (`example.com`) or a subdomain (`subdomain.example.com`). + - If you are using an apex domain, navigate to your DNS provider and create an `A` record that points your apex domain to GitHub's name servers which have the following IP addresses: + - `185.199.108.153` + - `185.199.109.153` + - `185.199.110.153` + - `185.199.111.153` + - If you are using a subdomain, navigate to your DNS provider and create a `CNAME` record that points your subdomain to the default domain for your site. For example, if you want to use the subdomain `quartz.example.com` for your user site, create a `CNAME` record that points `quartz.example.com` to `.github.io`. + +![[dns records.png]]_The above shows a screenshot of Google Domains configured for both `jzhao.xyz` (an apex domain) and `quartz.jzhao.xyz` (a subdomain)._ + +See the [GitHub documentation](https://docs.github.com/en/pages/configuring-a-custom-domain-for-your-github-pages-site/managing-a-custom-domain-for-your-github-pages-site#configuring-a-subdomain) for more detail about how to setup your own custom domain with GitHub Pages. + +> [!question] Why aren't my changes showing up? +> There could be many different reasons why your changes aren't showing up but the most likely reason is that you forgot to push your changes to GitHub. +> +> Make sure you save your changes to Git and sync it to GitHub by doing `npx quartz sync`. This will also make sure to pull any updates you may have made from other devices so you have them locally. + +## Vercel + +### Fix URLs + +Before deploying to Vercel, a `vercel.json` file is required at the root of the project directory. It needs to contain the following configuration so that URLs don't require the `.html` extension: + +```json title="vercel.json" +{ + "cleanUrls": true +} +``` + +### Deploy to Vercel + +1. Log in to the [Vercel Dashboard](https://vercel.com/dashboard) and click "Add New..." > Project +2. Import the Git repository containing your Quartz project. +3. Give the project a name (lowercase characters and hyphens only) +4. Check that these configuration options are set: + +| Configuration option | Value | +| ----------------------------------------- | ------------------ | +| Framework Preset | `Other` | +| Root Directory | `./` | +| Build and Output Settings > Build Command | `npx quartz build` | + +5. Press Deploy. Once it's live, you'll have 2 `*.vercel.app` URLs to view the page. + +### Custom Domain + +> [!note] +> If there is something already hosted on the domain, these steps will not work without replacing the previous content. As a workaround, you could use Next.js rewrites or use the next section to create a subdomain. + +1. Update the `baseUrl` in `quartz.config.js` if necessary. +2. Go to the [Domains - Dashboard](https://vercel.com/dashboard/domains) page in Vercel. +3. Connect the domain to Vercel +4. Press "Add" to connect a custom domain to Vercel. +5. Select your Quartz repository and press Continue. +6. Enter the domain you want to connect it to. +7. Follow the instructions to update your DNS records until you see "Valid Configuration" + +### Use a Subdomain + +Using `docs.example.com` is an example of a subdomain. They're a simple way of connecting multiple deployments to one domain. + +1. Update the `baseUrl` in `quartz.config.js` if necessary. +2. Ensure your domain has been added to the [Domains - Dashboard](https://vercel.com/dashboard/domains) page in Vercel. +3. Go to the [Vercel Dashboard](https://vercel.com/dashboard) and select your Quartz project. +4. Go to the Settings tab and then click Domains in the sidebar +5. Enter your subdomain into the field and press Add diff --git a/docs/images/dns records.png b/docs/images/dns records.png new file mode 100644 index 0000000000000000000000000000000000000000..bf9f854bdd4b10ffcf43f1d5193559fbea70e147 GIT binary patch literal 77842 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 a3O{<1f78MIRqYe-OH4>wu!R5pm;VPzscn`3 literal 0 HcmV?d00001 diff --git a/docs/images/quartz layout.png b/docs/images/quartz layout.png new file mode 100644 index 0000000000000000000000000000000000000000..03435f7d57874b5c7c438dd0866318fe3c8d61dd GIT binary patch literal 56729 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≧itPg6L@|j0))h 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 [!info] +> Coming from Quartz 3? See the [[migrating from Quartz 3|migration guide]] for the differences between Quartz 3 and Quartz 4 and how to migrate. + +## 🔧 Features + +- [[Obsidian compatibility]], [[full-text search]], [[graph view]], note transclusion, [[wikilinks]], [[backlinks]], [[Latex]], [[syntax highlighting]], [[popover previews]], [[Docker Support]], and [many more](./features) right out of the box +- Hot-reload for both configuration and content +- Simple JSX layouts and [[creating components|page components]] +- [[SPA Routing|Ridiculously fast page loads]] and tiny bundle sizes +- Fully-customizable parsing, filtering, and page generation through [[making plugins|plugins]] + +For a comprehensive list of features, visit the [features page](/features). You can read more about the _why_ behind these features on the [[philosophy]] page and a technical overview on the [[architecture]] page. + +### 🚧 Troubleshooting + Updating + +Having trouble with Quartz? Try searching for your issue using the search feature. If you haven't already, [[upgrading|upgrade]] to the newest version of Quartz to see if this fixes your issue. + +If you're still having trouble, feel free to [submit an issue](https://github.com/jackyzha0/quartz/issues) if you feel you found a bug or ask for help in our [Discord Community](https://discord.gg/cRFFHYye7t). diff --git a/docs/layout.md b/docs/layout.md new file mode 100644 index 0000000..3fabeb7 --- /dev/null +++ b/docs/layout.md @@ -0,0 +1,42 @@ +--- +title: Layout +--- + +Certain emitters may also output [HTML](https://developer.mozilla.org/en-US/docs/Web/HTML) files. To enable easy customization, these emitters allow you to fully rearrange the layout of the page. The default page layouts can be found in `quartz.layout.ts`. + +Each page is composed of multiple different sections which contain `QuartzComponents`. The following code snippet lists all of the valid sections that you can add components to: + +```typescript title="quartz/cfg.ts" +export interface FullPageLayout { + head: QuartzComponent // single component + header: QuartzComponent[] // laid out horizontally + beforeBody: QuartzComponent[] // laid out vertically + pageBody: QuartzComponent // single component + left: QuartzComponent[] // vertical on desktop, horizontal on mobile + right: QuartzComponent[] // vertical on desktop, horizontal on mobile + footer: QuartzComponent // single component +} +``` + +These correspond to following parts of the page: + +![[quartz layout.png|800]] + +> [!note] +> There are two additional layout fields that are _not_ shown in the above diagram. +> +> 1. `head` is a single component that renders the `` [tag](https://developer.mozilla.org/en-US/docs/Web/HTML/Element/head) in the HTML. This doesn't appear visually on the page and is only is responsible for metadata about the document like the tab title, scripts, and styles. +> 2. `header` is a set of components that are laid out horizontally and appears _before_ the `beforeBody` section. This enables you to replicate the old Quartz 3 header bar where the title, search bar, and dark mode toggle. By default, Quartz 4 doesn't place any components in the `header`. + +Quartz **components**, like plugins, can take in additional properties as configuration options. If you're familiar with React terminology, you can think of them as Higher-order Components. + +See [a list of all the components](component.md) for all available components along with their configuration options. You can also checkout the guide on [[creating components]] if you're interested in further customizing the behaviour of Quartz. + +### Style + +Most meaningful style changes like colour scheme and font can be done simply through the [[configuration#General Configuration|general configuration]] options. However, if you'd like to make more involved style changes, you can do this by writing your own styles. Quartz 4, like Quartz 3, uses [Sass](https://sass-lang.com/guide/) for styling. + +You can see the base style sheet in `quartz/styles/base.scss` and write your own in `quartz/styles/custom.scss`. + +> [!note] +> Some components may provide their own styling as well! For example, `quartz/components/Darkmode.tsx` imports styles from `quartz/components/styles/darkmode.scss`. If you'd like to customize styling for a specific component, double check the component definition to see how its styles are defined. diff --git a/docs/migrating from Quartz 3.md b/docs/migrating from Quartz 3.md new file mode 100644 index 0000000..2fdc731 --- /dev/null +++ b/docs/migrating from Quartz 3.md @@ -0,0 +1,41 @@ +--- +title: "Migrating from Quartz 3" +--- + +As you already have Quartz locally, you don't need to fork or clone it again. Simply just checkout the alpha branch, install the dependencies, and import your old vault. + +```bash +git fetch +git checkout v4 +git pull upstream v4 +npm i +npx quartz create +``` + +If you get an error like `fatal: 'upstream' does not appear to be a git repository`, make sure you add `upstream` as a remote origin: + +```shell +git remote add upstream https://github.com/jackyzha0/quartz.git +``` + +When running `npx quartz create`, you will be prompted as to how to initialize your content folder. Here, you can choose to import or link your previous content folder and Quartz should work just as you expect it to. + +> [!note] +> If the existing content folder you'd like to use is at the _same_ path on a different branch, clone the repo again somewhere at a _different_ path in order to use it. + +## Key changes + +1. **Removing Hugo and `hugo-obsidian`**: Hugo worked well for earlier versions of Quartz but it also made it hard for people outside of the Golang and Hugo communities to fully understand what Quartz was doing under the hood and be able to properly customize it to their needs. Quartz 4 now uses a Node-based static-site generation process which should lead to a much more helpful error messages and an overall smoother user experience. +2. **Full-hot reload**: The many rough edges of how `hugo-obsidian` integrated with Hugo meant that watch mode didn't re-trigger `hugo-obsidian` to update the content index. This lead to a lot of weird cases where the watch mode output wasn't accurate. Quartz 4 now uses a cohesive parse, filter, and emit pipeline which gets run on every change so hot-reloads are always accurate. +3. **Replacing Go template syntax with JSX**: Quartz 3 used [Go templates](https://pkg.go.dev/text/template) to create layouts for pages. However, the syntax isn't great for doing any sort of complex rendering (like [text processing](https://github.com/jackyzha0/quartz/blob/hugo/layouts/partials/textprocessing.html)) and it got very difficult to make any meaningful layout changes to Quartz 3. Quartz 4 uses an extension of JavaScript syntax called JSX which allows you to write layout code that looks like HTML in JavaScript which is significantly easier to understand and maintain. +4. **A new extensible [[configuration]] and [[configuration#Plugins|plugin]] system**: Quartz 3 was hard to configure without technical knowledge of how Hugo's partials worked. Extensions were even hard to make. Quartz 4's configuration and plugin system is designed to be extended by users while making updating to new versions of Quartz easy. + +## Things to update + +- You will need to update your deploy scripts. See the [[hosting]] guide for more details. +- Ensure that your default branch on GitHub is updated from `hugo` to `v4`. +- [[folder and tag listings|Folder and tag listings]] have also changed. + - Folder descriptions should go under `content//index.md` where `` is the name of the folder. + - Tag descriptions should go under `content/tags/.md` where `` is the name of the tag. +- Some HTML layout may not be the same between Quartz 3 and Quartz 4. If you depended on a particular HTML hierarchy or class names, you may need to update your custom CSS to reflect these changes. +- If you customized the layout of Quartz 3, you may need to translate these changes from Go templates back to JSX as Quartz 4 no longer uses Hugo. For components, check out the guide on [[creating components]] for more details on this. diff --git a/docs/philosophy.md b/docs/philosophy.md new file mode 100644 index 0000000..b8af845 --- /dev/null +++ b/docs/philosophy.md @@ -0,0 +1,28 @@ +--- +title: Philosophy of Quartz +--- + +## A garden should be a true hypertext + +> The garden is the web as topology. Every walk through the garden creates new paths, new meanings, and when we add things to the garden we add them in a way that allows many future, unpredicted relationships. +> +> _(The Garden and the Stream)_ + +The problem with the file cabinet is that it focuses on efficiency of access and interoperability rather than generativity and creativity. Thinking is not linear, nor is it hierarchical. In fact, not many things are linear or hierarchical at all. Then why is it that most tools and thinking strategies assume a nice chronological or hierarchical order for my thought processes? The ideal tool for thought for me would embrace the messiness of my mind, and organically help insights emerge from chaos instead of forcing an artificial order. A rhizomatic, not arboresecent, form of note taking. + +My goal with a digital garden is not purely as an organizing system and information store (though it works nicely for that). I want my digital garden to be a playground for new ways ideas can connect together. As a result, existing formal organizing systems like Zettelkasten or the hierarchical folder structures of Notion don’t work well for me. There is way too much upfront friction that by the time I’ve thought about how to organize my thought into folders categories, I’ve lost it. + +Quartz embraces the inherent rhizomatic and web-like nature of our thinking and tries to encourage note-taking in a similar form. + +--- + +## A garden should be shared + +The goal of digital gardening should be to tap into your network’s collective intelligence to create constructive feedback loops. If done well, I have a shareable representation of my thoughts that I can send out into the world and people can respond. Even for my most half-baked thoughts, this helps me create a feedback cycle to strengthen and fully flesh out that idea. + +Quartz is designed first and foremost as a tool for publishing [digital gardens](https://jzhao.xyz/posts/networked-thought) to the web. To me, digital gardening is not just passive knowledge collection. It’s a form of expression and sharing. + +> “[One] who works with the door open gets all kinds of interruptions, but [they] also occasionally gets clues as to what the world is and what might be important.” +> — Richard Hamming + +**The goal of Quartz is to make sharing your digital garden free and simple.** At its core, Quartz is designed to be easy to use enough for non-technical people to get going but also powerful enough that senior developers can tweak it to work how they'd like it to work. diff --git a/docs/showcase.md b/docs/showcase.md new file mode 100644 index 0000000..6814dd0 --- /dev/null +++ b/docs/showcase.md @@ -0,0 +1,24 @@ +--- +title: "Quartz Showcase" +--- + +Want to see what Quartz can do? Here are some cool community gardens: + +- [Quartz Documentation (this site!)](https://quartz.jzhao.xyz/) +- [Jacky Zhao's Garden](https://jzhao.xyz/) +- [Brandon Boswell's Garden](https://brandonkboswell.com) +- [Scaling Synthesis - A hypertext research notebook](https://scalingsynthesis.com/) +- [AWAGMI Intern Notes](https://notes.awagmi.xyz/) +- [Course notes for Information Technology Advanced Theory](https://a2itnotes.github.io/quartz/) +- [Data Dictionary 🧠](https://glossary.airbyte.com/) +- [sspaeti.com's Second Brain](https://brain.sspaeti.com/) +- [oldwinter の数字花园](https://garden.oldwinter.top/) +- [Abhijeet's Math Wiki](https://abhmul.github.io/quartz/Math-Wiki/) +- [Mike's AI Garden 🤖🪴](https://mwalton.me/) +- [Matt Dunn's Second Brain](https://mattdunn.info/) +- [Pelayo Arbues' Notes](https://pelayoarbues.github.io/) +- [Vince Imbat's Talahardin](https://vinceimbat.com/) +- [🧠🌳 Chad's Mind Garden](https://www.chadly.net/) +- [Pedro MC Fernandes's Topo da Mente](https://www.pmcf.xyz/topo-da-mente/) + +If you want to see your own on here, submit a [Pull Request adding yourself to this file](https://github.com/jackyzha0/quartz/blob/v4/docs/showcase.md)! diff --git a/docs/tags/component.md b/docs/tags/component.md new file mode 100644 index 0000000..57592e8 --- /dev/null +++ b/docs/tags/component.md @@ -0,0 +1,5 @@ +--- +title: Components +--- + +Want to create your own custom component? Check out the advanced guide on [[creating components]] for more information. diff --git a/docs/upgrading.md b/docs/upgrading.md new file mode 100644 index 0000000..a3a8275 --- /dev/null +++ b/docs/upgrading.md @@ -0,0 +1,19 @@ +--- +title: "Upgrading Quartz" +--- + +> [!note] +> This is specifically a guide for upgrading Quartz 4 version to a more recent update. If you are coming from Quartz 3, check out the [[migrating from Quartz 3|migration guide]] for more info. + +To fetch the latest Quartz updates, simply run + +```bash +npx quartz update +``` + +As Quartz uses [git](https://git-scm.com/) under the hood for versioning, updating effectively 'pulls' in the updates from the official Quartz GitHub repository. If you have local changes that might conflict with the updates, you may need to resolve these manually yourself (or, pull manually using `git pull origin upstream`). + +> [!hint] +> Quartz will try to cache your content before updating to try and prevent merge conflicts. If you get a conflict mid-merge, you can stop the merge and then run `npx quartz restore` to restore your content from the cache. + +If you have the [GitHub desktop app](https://desktop.github.com/), this will automatically open to help you resolve the conflicts. Otherwise, you will need to resolve this in a text editor like VSCode. For more help on resolving conflicts manually, check out the [GitHub guide on resolving merge conflicts](https://docs.github.com/en/pull-requests/collaborating-with-pull-requests/addressing-merge-conflicts/resolving-a-merge-conflict-using-the-command-line#competing-line-change-merge-conflicts). diff --git a/globals.d.ts b/globals.d.ts new file mode 100644 index 0000000..4473d59 --- /dev/null +++ b/globals.d.ts @@ -0,0 +1,12 @@ +export declare global { + interface Document { + addEventListener( + type: K, + listener: (this: Document, ev: CustomEventMap[K]) => void, + ): void + dispatchEvent(ev: CustomEventMap[K]): void + } + interface Window { + spaNavigate(url: URL, isBack: boolean = false) + } +} diff --git a/index.d.ts b/index.d.ts new file mode 100644 index 0000000..aec536d --- /dev/null +++ b/index.d.ts @@ -0,0 +1,11 @@ +declare module "*.scss" { + const content: string + export = content +} + +// dom custom event +interface CustomEventMap { + nav: CustomEvent<{ url: FullSlug }> +} + +declare const fetchData: Promise diff --git a/logseq/bak/pages/July 2021 Meeting/2022-06-29T23_03_23.581Z.md b/logseq/bak/pages/July 2021 Meeting/2022-06-29T23_03_23.581Z.md deleted file mode 100755 index 3249ec9..0000000 --- a/logseq/bak/pages/July 2021 Meeting/2022-06-29T23_03_23.581Z.md +++ /dev/null @@ -1,6 +0,0 @@ -date:: [[Jul 27th, 2021]] - -- speakers:: [[Linus Lee]], [[Kevin Lin]], [[Rafael Nepo]] -- video:: https://youtu.be/u-ib5IZnZu0 -- [[July Meeting Recap]] -- [[July Meeting Chat Log]] \ No newline at end of file diff --git a/logseq/config.edn b/logseq/config.edn deleted file mode 100644 index 8f7f9ee..0000000 --- a/logseq/config.edn +++ /dev/null @@ -1,154 +0,0 @@ -{;; Currently, we support either "Markdown" or "Org". - ;; This can overwrite your global preference so that - ;; maybe your personal preferred format is Org but you'd - ;; need to use Markdown for some projects. - ;; :preferred-format "" - - ;; Preferred workflow style. - ;; Value is either ":now" for NOW/LATER style, - ;; or ":todo" for TODO/DOING style. - :preferred-workflow :now - - ;; Git is only available in web app - ;; Git settings - :git-pull-secs 60 - :git-push-secs 10 - ;; Whether to enable git auto push - :git-auto-push true - - ;; The app will ignore those directories or files. - ;; E.g. "/archived" "/test.md" - :hidden [] - - ;; When creating the new journal page, the app will use your template content here. - ;; Example for Markdown users: "[[Work]]\n -\n- [[Family]]\n -\n" - ;; Example for Org mode users: "** [[Work]]\n***\n** [[Family]]\n***\n" - :default-templates - {:journals ""} - - ;; Whether to enable hover on tooltip preview feature - ;; Default is true, you can also toggle this via setting page - :ui/enable-tooltip? true - - ;; Specify a custom CSS import - ;; This option take precedence over your local `logseq/custom.css` file - ;; You may find a list of awesome logseq themes here: - ;; https://github.com/logseq/awesome-logseq#css-themes - ;; Example: - ;; :custom-css-url "@import url('https://cdn.jsdelivr.net/gh/dracula/logseq@master/custom.css');" - - ;; When :all-pages-public? true, export repo would export all pages within that repo. - ;; Regardless of whether you've set any page to public or not. - ;; Example: - ;; :all-pages-public? true - - ;; Specify default home page and sidebar status for Logseq - ;; If not specified, Logseq default opens journals page on startup - ;; value for `:page` is name of page - ;; Possible options for `:sidebar` are - ;; 1. `"Contents"` to open up `Favorites` in sidebar by default - ;; 2. `page name` to open up some page in sidebar - ;; 3. Or multiple pages in an array ["Contents" "Page A" "Page B"] - ;; If `:sidebar` is not set, sidebar will be hidden - ;; Example: - ;; 1. Setup page "Changelog" as home page and "Favorites" in sidebar - ;; :default-home {:page "Changelog", :sidebar "Contents"} - ;; 2. Setup page "Jun 3rd, 2021" as home page without sidebar - ;; :default-home {:page "Jun 3rd, 2021"} - ;; 3. Setup page "home" as home page with multiple pages in sidebar - ;; :default-home {:page "home" :sidebar ["page a" "page b"]} -:default-home {:page "Contents"} - - ;; Tell logseq to use a specific folder in the repo as a default location for notes - ;; if not specified, notes are stored in `pages` directory - ;; :pages-directory "your-directory" - - ;; Tell logseq to use a specific folder in the repo as a default location for journals - ;; if not specified, journals are stored in `journals` directory - ;; :journals-directory "your-directory" - - ;; Set this to true will convert - ;; `[[Grant Ideas]]` to `[[file:./grant_ideas.org][Grant Ideas]]` for org-mode - ;; For more, see https://github.com/logseq/logseq/issues/672 - ;; :org-mode/insert-file-link? true - - ;; If you prefer to use the file name as the page title - ;; instead of the first heading's title - ;; the only option for now is `file` - ;; :page-name-order "file" - - ;; Setup custom shortcuts under `:shortcuts` key - ;; Syntax: - ;; 1. `+` means keys pressing simultaneously. eg: `ctrl+shift+a` - ;; 2. ` ` empty space between keys represents key chords. eg: `t s` means press `s` follow by `t` - ;; 3. `mod` means `Ctrl` for Windows/Linux and `Command` for Mac - ;; 4. use `false` to disable particular shortcut - ;; 4. you can define multiple bindings for one action, eg `["ctrl+j" "down"]` - ;; full list of configurable shortcuts are available below: - ;; https://github.com/logseq/logseq/blob/master/src/main/frontend/modules/shortcut/config.cljs - ;; Example: - ;; :shortcuts - ;; {:editor/new-block "enter" - ;; :editor/new-line "shift+enter" - ;; :editor/insert-link "mod+shift+k" - ;; :editor/hightlight false - ;; :ui/toggle-settings "t s" - ;; :editor/up ["ctrl+k" "up"] - ;; :editor/down ["ctrl+j" "down"] - ;; :editor/left ["ctrl+h" "left"] - ;; :editor/right ["ctrl+l" "right"]} - - ;; The app will show those queries in today's journal page, - ;; the "NOW" query asks the tasks which need to be finished "now", - ;; the "NEXT" query asks the future tasks. - :default-queries - {:journals - [{:title "🔨 NOW" - :query [:find (pull ?h [*]) - :in $ ?start ?today - :where - [?h :block/marker ?marker] - [(contains? #{"NOW" "DOING"} ?marker)] - [?h :block/page ?p] - [?p :block/journal? true] - [?p :block/journal-day ?d] - [(>= ?d ?start)] - [(<= ?d ?today)]] - :inputs [:14d :today] - :result-transform (fn [result] - (sort-by (fn [h] - (get h :block/priority "Z")) result)) - :collapsed? false} - {:title "📅 NEXT" - :query [:find (pull ?h [*]) - :in $ ?start ?next - :where - [?h :block/marker ?marker] - [(contains? #{"NOW" "LATER" "TODO"} ?marker)] - [?h :block/ref-pages ?p] - [?p :block/journal? true] - [?p :block/journal-day ?d] - [(> ?d ?start)] - [(< ?d ?next)]] - :inputs [:today :7d-after] - :collapsed? false}]} - - ;; Add your own commands to speedup. - ;; E.g. [["js" "Javascript"]] - :commands - [] - - ;; Macros replace texts and will make you more productive. - ;; For example: - ;; Add this to the macros below: - ;; {"poem" "Rose is $1, violet's $2. Life's ordered: Org assists you."} - ;; input "{{{poem red,blue}}}" - ;; becomes - ;; Rose is red, violet's blue. Life's ordered: Org assists you. - :macros {} - - :markdown/version 2 - :publishing/all-pages-public? true - :feature/enable-timetracking? true - :feature/enable-flashcards? false - :favorites ["Early Tools for Thought" "Tools for Thought App List" "Monthly Meetings" "Mastodon Resources"]} diff --git a/logseq/metadata.edn b/logseq/metadata.edn deleted file mode 100755 index 9e26dfe..0000000 --- a/logseq/metadata.edn +++ /dev/null @@ -1 +0,0 @@ -{} \ No newline at end of file diff --git a/logseq/pages-metadata.edn b/logseq/pages-metadata.edn deleted file mode 100755 index d5ccf64..0000000 --- a/logseq/pages-metadata.edn +++ /dev/null @@ -1,648 +0,0 @@ -[{:block/name "62bcd24d-7ae0-42f9-bc4e-650d0b35559f", - :block/created-at 1656545609213, - :block/updated-at 1656545609213} - {:block/name "a", - :block/created-at 1646357158470, - :block/updated-at 1646357158470} - {:block/name "a big list of mastodon resources", - :block/created-at 1668387176566, - :block/updated-at 1668387813711} - {:block/name "about", - :block/created-at 1668908457562, - :block/updated-at 1668908457562} - {:block/name "activitypub", - :block/created-at 1668387835838, - :block/updated-at 1668387835838} - {:block/name "alexander obernauer", - :block/created-at 1668908458161, - :block/updated-at 1668908458161} - {:block/name "amazon s3", - :block/created-at 1646357158471, - :block/updated-at 1646357158471} - {:block/name "andy matuschak", - :block/created-at 1646357158473, - :block/updated-at 1646357158473} - {:block/name "app", - :block/created-at 1646357158472, - :block/updated-at 1646357158472} - {:block/name "apr 28th, 2022", - :block/created-at 1668908457421, - :block/updated-at 1668908457421} - {:block/name "april 2022 meeting", - :block/created-at 1668908457429, - :block/updated-at 1668908457429} - {:block/name "aug 16th, 2021", - :block/created-at 1629072000000, - :block/updated-at 1656545762548} - {:block/name "aug 16th, 2022", - :block/created-at 1660672021971, - :block/updated-at 1660672047966} - {:block/name "aug 25th, 2021", - :block/created-at 1629849600000, - :block/updated-at 1629849600000} - {:block/name "aug 9th, 2022", - :block/created-at 1668908458291, - :block/updated-at 1668908458291} - {:block/name "augmented text author", - :block/created-at 1646357158470, - :block/updated-at 1646357158470} - {:block/name "augmented text company", - :block/created-at 1646357158471, - :block/updated-at 1646357158471} - {:block/name "augmented text reader", - :block/created-at 1646357158471, - :block/updated-at 1646357158471} - {:block/name "august 2021 meeting", - :block/created-at 1646357158471, - :block/updated-at 1656543159343} - {:block/name "australia", - :block/created-at 1668908457824, - :block/updated-at 1668908457824} - {:block/name "author", - :block/created-at 1660672020753, - :block/updated-at 1660672020753} - {:block/name "avi bryant", - :block/created-at 1646357158471, - :block/updated-at 1646357158471} - {:block/name "awesome list", - :block/created-at 1646357158473, - :block/updated-at 1646357158473} - {:block/name "b", - :block/created-at 1646357158471, - :block/updated-at 1646357158471} - {:block/name "bear theme", - :block/created-at 1668908457349, - :block/updated-at 1668908457349} - {:block/name "bill seitz", - :block/created-at 1646357158471, - :block/updated-at 1646357158471} - {:block/name "blockchain dialogues", - :block/created-at 1668454359470, - :block/updated-at 1668454359470} - {:block/name "boris", - :block/created-at 1646357158473, - :block/updated-at 1646357158473} - {:block/name "boris mann", - :block/created-at 1646357158471, - :block/updated-at 1668670319749} - {:block/name "brooklyn zelenka", - :block/created-at 1646357158471, - :block/updated-at 1646357158471} - {:block/name "butter", - :block/created-at 1668474726269, - :block/updated-at 1668474726269} - {:block/name "c", - :block/created-at 1646357158471, - :block/updated-at 1646357158471} - {:block/name "canceled", - :block/created-at 1660672712784, - :block/updated-at 1660672712784} - {:block/name "cancelled", - :block/created-at 1660672712784, - :block/updated-at 1660672712784} - {:block/name "card", - :block/created-at 1646357158471, - :block/updated-at 1646357158471} - {:block/name "charles chamberlain", - :block/created-at 1668908458208, - :block/updated-at 1668908458208} - {:block/name "chatlog", - :block/created-at 1646357158472, - :block/updated-at 1646357158472} - {:block/name "christina bowen", - :block/created-at 1660672498203, - :block/updated-at 1668391685267} - {:block/name "circle", - :block/created-at 1646357158471, - :block/updated-at 1646357158471} - {:block/name "cloudflare", - :block/created-at 1646357158471, - :block/updated-at 1646357158472} - {:block/name "cloudflare r2", - :block/created-at 1646357158471, - :block/updated-at 1646357158471} - {:block/name "colophon", - :block/created-at 1668669381914, - :block/updated-at 1668669778173} - {:block/name "contents", - :block/created-at 1668474726930, - :block/updated-at 1668669825904} - {:block/name "cory doctorow", - :block/created-at 1646357158471, - :block/updated-at 1646357158471} - {:block/name "dabbledb", - :block/created-at 1646357158471, - :block/updated-at 1646357158471} - {:block/name "dalton banks", - :block/created-at 1668908458162, - :block/updated-at 1668908458162} - {:block/name "danielle navarro", - :block/created-at 1668386381758, - :block/updated-at 1668386381758} - {:block/name "darlene dang", - :block/created-at 1668908457746, - :block/updated-at 1668908457746} - {:block/name "date", - :block/created-at 1660672020621, - :block/updated-at 1660672020621} - {:block/name "david brooks", - :block/created-at 1646357158471, - :block/updated-at 1646357158471} - {:block/name "david perell", - :block/created-at 1646357158471, - :block/updated-at 1646357158471} - {:block/name "dec 5th, 2021", - :block/created-at 1638662400000, - :block/updated-at 1638662400000} - {:block/name "discord", - :block/created-at 1646357158471, - :block/updated-at 1646357158471} - {:block/name "doing", - :block/created-at 1646357158471, - :block/updated-at 1646357158471} - {:block/name "done", - :block/created-at 1646357158471, - :block/updated-at 1646357158471} - {:block/name "donor", - :block/created-at 1668908457347, - :block/updated-at 1668908457347} - {:block/name "elixirconf 2021", - :block/created-at 1646357158473, - :block/updated-at 1646357158473} - {:block/name "estonia", - :block/created-at 1668388266947, - :block/updated-at 1668388266947} - {:block/name "event", - :block/created-at 1646357158471, - :block/updated-at 1646357158471} - {:block/name "event listing", - :block/created-at 1646357158471, - :block/updated-at 1668712117877} - {:block/name "events", - :block/created-at 1668908457801, - :block/updated-at 1668908457801} - {:block/name "everything i know about mastodon", - :block/created-at 1668386360969, - :block/updated-at 1668386517092} - {:block/name "export considered harmful", - :block/created-at 1668474726149, - :block/updated-at 1668474726149} - {:block/name "favorites", - :block/created-at 1668474726906, - :block/updated-at 1668474726906} - {:block/name "feb 23rd, 2020", - :block/created-at 1582416000000, - :block/updated-at 1582416000000} - {:block/name "feb 23rd, 2022", - :block/created-at 1668908458161, - :block/updated-at 1668908458161} - {:block/name "february 2022 meeting", - :block/created-at 1668908458169, - :block/updated-at 1668908458169} - {:block/name "federated hosting by spacebear", - :block/created-at 1668388122374, - :block/updated-at 1668388216405} - {:block/name "fedi tips", - :block/created-at 1668387813711, - :block/updated-at 1668387860529} - {:block/name "fedifinder", - :block/created-at 1668387485567, - :block/updated-at 1668387506777} - {:block/name "fediverse", - :block/created-at 1668386009653, - :block/updated-at 1668386009653} - {:block/name "fediverse party - mastodon", - :block/created-at 1668387740916, - :block/updated-at 1668387746666} - {:block/name "fission", - :block/created-at 1646357158471, - :block/updated-at 1646357158471} - {:block/name "fission discord", - :block/created-at 1646357158471, - :block/updated-at 1646357158471} - {:block/name "flancian", - :block/created-at 1668387651287, - :block/updated-at 1668387651287} - {:block/name "for thought", - :block/created-at 1668474725978, - :block/updated-at 1668474725978} - {:block/name "frode hegland", - :block/created-at 1646357158471, - :block/updated-at 1646357158471} - {:block/name "future of text", - :block/created-at 1646357158471, - :block/updated-at 1646357158471} - {:block/name "geoffrey litt", - :block/created-at 1646357158471, - :block/updated-at 1668670276454} - {:block/name "gordon brander", - :block/created-at 1668908458448, - :block/updated-at 1668908458448} - {:block/name "gsheet mastodon open servers english 5k+ active users", - :block/created-at 1668474726275, - :block/updated-at 1668474726275} - {:block/name "gumroad", - :block/created-at 1668388214632, - :block/updated-at 1668388214632} - {:block/name "hometown", - :block/created-at 1668388148807, - :block/updated-at 1668388148807} - {:block/name "how to make the fediverse your own", - :block/created-at 1668474726326, - :block/updated-at 1668474726326} - {:block/name "howard rheingold", - :block/created-at 1646357158471, - :block/updated-at 1646357158471} - {:block/name "hugo", - :block/created-at 1668389310486, - :block/updated-at 1668389310486} - {:block/name "hunter clarke", - :block/created-at 1668908457422, - :block/updated-at 1668908457422} - {:block/name "hyperdraft", - :block/created-at 1646357158473, - :block/updated-at 1646357158473} - {:block/name "in-progress", - :block/created-at 1646357158473, - :block/updated-at 1646357158473} - {:block/name "instances.social", - :block/created-at 1668387375588, - :block/updated-at 1668387894676} - {:block/name "ipfs", - :block/created-at 1646357158471, - :block/updated-at 1646357158471} - {:block/name "isomorphic git", - :block/created-at 1646357158471, - :block/updated-at 1646357158471} - {:block/name "james walker", - :block/created-at 1668391227179, - :block/updated-at 1668391236897} - {:block/name "jan 27th, 2022", - :block/created-at 1668908457048, - :block/updated-at 1668908457048} - {:block/name "january 2022 meeting", - :block/created-at 1668908457055, - :block/updated-at 1668908457055} - {:block/name "jared pereira", - :block/created-at 1646357158472, - :block/updated-at 1656541914523} - {:block/name "jess martin", - :block/created-at 1660672487742, - :block/updated-at 1660672487742} - {:block/name "joybox", - :block/created-at 1668474726392, - :block/updated-at 1668474726392} - {:block/name "jul 27th, 2021", - :block/created-at 1627344000000, - :block/updated-at 1627344000000} - {:block/name "july 2021 meeting", - :block/created-at 1668474726245, - :block/updated-at 1668474726245} - {:block/name "july meeting chat log", - :block/created-at 1668474726370, - :block/updated-at 1668474726370} - {:block/name "july meeting recap", - :block/created-at 1646357158471, - :block/updated-at 1646357158471} - {:block/name "jun 10th, 2020", - :block/created-at 1591747200000, - :block/updated-at 1591747200000} - {:block/name "jun 15th, 2021", - :block/created-at 1623715200000, - :block/updated-at 1623715200000} - {:block/name "jun 29th, 2022", - :block/created-at 1656541775347, - :block/updated-at 1656541775347} - {:block/name "justin johnson", - :block/created-at 1668391227179, - :block/updated-at 1668391227179} - {:block/name "kevin lin", - :block/created-at 1646357158471, - :block/updated-at 1656542041921} - {:block/name "later", - :block/created-at 1646357158471, - :block/updated-at 1646357158471} - {:block/name "link", - :block/created-at 1660672020618, - :block/updated-at 1660672020618} - {:block/name "linus lee", - :block/created-at 1646357158471, - :block/updated-at 1656542027342} - {:block/name "logseq", - :block/created-at 1646357158470, - :block/updated-at 1646357158470} - {:block/name "logseq plugin", - :block/created-at 1668389310484, - :block/updated-at 1668389310484} - {:block/name "logseq publish", - :block/created-at 1668389162320, - :block/updated-at 1668391173998} - {:block/name "logseq schrödinger", - :block/created-at 1668389146939, - :block/updated-at 1668389310486} - {:block/name "lucerne", - :block/created-at 1668474726586, - :block/updated-at 1668474726586} - {:block/name "luma", - :block/created-at 1646357158471, - :block/updated-at 1668670010747} - {:block/name "lumalink", - :block/created-at 1660672021394, - :block/updated-at 1660672021394} - {:block/name "maggie appleton", - :block/created-at 1668908457421, - :block/updated-at 1668908457421} - {:block/name "mar 30th, 2022", - :block/created-at 1668908458448, - :block/updated-at 1668908458448} - {:block/name "marc-antoine parent", - :block/created-at 1646357158471, - :block/updated-at 1646357158471} - {:block/name "march 2022 meeting", - :block/created-at 1668908458454, - :block/updated-at 1668908458454} - {:block/name "mark upton", - :block/created-at 1668908457342, - :block/updated-at 1668908457342} - {:block/name "mastodon", - :block/created-at 1668385921645, - :block/updated-at 1668385921645} - {:block/name "mastodon resources", - :block/created-at 1668386204605, - :block/updated-at 1668387176567} - {:block/name "may 2022 meeting", - :block/created-at 1668908458230, - :block/updated-at 1668908458230} - {:block/name "may 26th, 2022", - :block/created-at 1668908458223, - :block/updated-at 1668908458223} - {:block/name "meeting notes", - :block/created-at 1646357158470, - :block/updated-at 1646357158470} - {:block/name "michael gartner", - :block/created-at 1668391294844, - :block/updated-at 1668391335730} - {:block/name "moa party", - :block/created-at 1668387477941, - :block/updated-at 1668387651288} - {:block/name "monthly meetings", - :block/created-at 1668908457782, - :block/updated-at 1668908457782} - {:block/name "monthly-meeting", - :block/created-at 1668908457050, - :block/updated-at 1668908457050} - {:block/name "monthlydonor", - :block/created-at 1668908458246, - :block/updated-at 1668908458246} - {:block/name "nov 13th, 2022", - :block/created-at 1668382703663, - :block/updated-at 1668391582548} - {:block/name "nov 14th, 2022", - :block/created-at 1668466686678, - :block/updated-at 1668466686678} - {:block/name "nov 15th, 2022", - :block/created-at 1668531662019, - :block/updated-at 1668531662019} - {:block/name "nov 16th, 2022", - :block/created-at 1668908457350, - :block/updated-at 1668908457350} - {:block/name "nov 17th, 2022", - :block/created-at 1668711695049, - :block/updated-at 1668711695049} - {:block/name "nov 30th, 2021", - :block/created-at 1638230400000, - :block/updated-at 1638230400000} - {:block/name "nov 3rd, 2022", - :block/created-at 1668386517047, - :block/updated-at 1668386517047} - {:block/name "november 2021 chat log", - :block/created-at 1668474726692, - :block/updated-at 1668474726692} - {:block/name "november 2021 meeting", - :block/created-at 1646357158471, - :block/updated-at 1656543116708} - {:block/name "now", - :block/created-at 1646357158473, - :block/updated-at 1646357158473} - {:block/name "oct 20th, 2022", - :block/created-at 1668908458187, - :block/updated-at 1668908458187} - {:block/name "oct 25th, 2021", - :block/created-at 1635120000000, - :block/updated-at 1635120000000} - {:block/name "oct 29th, 2021", - :block/created-at 1635465600000, - :block/updated-at 1635465600000} - {:block/name "oct 4th, 2022", - :block/created-at 1668908457950, - :block/updated-at 1668908457950} - {:block/name "october 2021 chat log", - :block/created-at 1668474726782, - :block/updated-at 1668474726782} - {:block/name "october 2021 meeting", - :block/created-at 1646357158471, - :block/updated-at 1668729049791} - {:block/name "on", - :block/created-at 1668466656716, - :block/updated-at 1668466656716} - {:block/name "open collective", - :block/created-at 1668391493078, - :block/updated-at 1668391493078} - {:block/name "opencollective", - :block/created-at 1668387966217, - :block/updated-at 1668387966217} - {:block/name "opensource", - :block/created-at 1668387634874, - :block/updated-at 1668387634874} - {:block/name "organizer", - :block/created-at 1668391642268, - :block/updated-at 1668391642268} - {:block/name "paul rony", - :block/created-at 1668908458207, - :block/updated-at 1668908458207} - {:block/name "paul shen", - :block/created-at 1668908458188, - :block/updated-at 1668908458188} - {:block/name "peertube", - :block/created-at 1668388145307, - :block/updated-at 1668388145307} - {:block/name "person", - :block/created-at 1646357158471, - :block/updated-at 1646357158471} - {:block/name "pesos", - :block/created-at 1646357158471, - :block/updated-at 1646357158471} - {:block/name "pirijan", - :block/created-at 1668908457746, - :block/updated-at 1668908457746} - {:block/name "pixelfed", - :block/created-at 1668388136128, - :block/updated-at 1668388136128} - {:block/name "pleroma", - :block/created-at 1668388154178, - :block/updated-at 1668388154178} - {:block/name "pol baladas", - :block/created-at 1668908458187, - :block/updated-at 1668908458187} - {:block/name "posse", - :block/created-at 1646357158471, - :block/updated-at 1646357158471} - {:block/name "python", - :block/created-at 1668387634873, - :block/updated-at 1668387634873} - {:block/name "rafael nepo", - :block/created-at 1646357158471, - :block/updated-at 1656542043807} - {:block/name "readme", - :block/created-at 1646357158472, - :block/updated-at 1646357158472} - {:block/name "researchbuzz", - :block/created-at 1668387176565, - :block/updated-at 1668387176565} - {:block/name "roam research", - :block/created-at 1646357158471, - :block/updated-at 1646357158471} - {:block/name "rob mcroberson", - :block/created-at 1668993645420, - :block/updated-at 1668993645420} - {:block/name "robbl", - :block/created-at 1668992696050, - :block/updated-at 1668992696050} - {:block/name "robert haisfield", - :block/created-at 1668728562402, - :block/updated-at 1668728562402} - {:block/name "rosano", - :block/created-at 1646357158471, - :block/updated-at 1668670207296} - {:block/name "ryan atkinson", - :block/created-at 1668908457345, - :block/updated-at 1668908457345} - {:block/name "scaling synthesis - extending tools for thought", - :block/created-at 1668908457957, - :block/updated-at 1668908457957} - {:block/name "scaling synthesis - multiplayer tft", - :block/created-at 1668908458300, - :block/updated-at 1668908458300} - {:block/name "scalingsynthesis", - :block/created-at 1668908457947, - :block/updated-at 1668908457947} - {:block/name "sep 29th, 2022", - :block/created-at 1668908457745, - :block/updated-at 1668908457745} - {:block/name "sindre sorhus", - :block/created-at 1646357158470, - :block/updated-at 1646357158470} - {:block/name "social coop", - :block/created-at 1668386009654, - :block/updated-at 1668386009654} - {:block/name "spacebear", - :block/created-at 1668388044051, - :block/updated-at 1668388266947} - {:block/name "spatial canvases - part 1", - :block/created-at 1668908457755, - :block/updated-at 1668908457755} - {:block/name "spatial canvases - part 2", - :block/created-at 1668908458194, - :block/updated-at 1668908458194} - {:block/name "spatial canvases - part 3", - :block/created-at 1668908458213, - :block/updated-at 1668908458213} - {:block/name "speaker", - :block/created-at 1646357158471, - :block/updated-at 1646357158471} - {:block/name "speaker listing", - :block/created-at 1646357158470, - :block/updated-at 1656543978978} - {:block/name "sprout", - :block/created-at 1668474726783, - :block/updated-at 1668474726783} - {:block/name "start here", - :block/created-at 1668908457787, - :block/updated-at 1668908457787} - {:block/name "tara calishain", - :block/created-at 1668387306860, - :block/updated-at 1668387306860} - {:block/name "taylor hadden", - :block/created-at 1668727460765, - :block/updated-at 1668727460765} - {:block/name "the jump to hyperspace", - :block/created-at 1646357158471, - :block/updated-at 1646357158471} - {:block/name "the memex method", - :block/created-at 1646357158471, - :block/updated-at 1646357158471} - {:block/name "todo", - :block/created-at 1646357158471, - :block/updated-at 1646357158471} - {:block/name "tony confrey", - :block/created-at 1668908458224, - :block/updated-at 1668908458224} - {:block/name "tool", - :block/created-at 1668474725978, - :block/updated-at 1668474725978} - {:block/name "tool for thought", - :block/created-at 1646357158471, - :block/updated-at 1646357158471} - {:block/name "tools for thought app list", - :block/created-at 1660672047966, - :block/updated-at 1668391724917} - {:block/name "tools for thought mastodon", - :block/created-at 1668385876683, - :block/updated-at 1668388284805} - {:block/name "tools for thought rocks", - :block/created-at 1668474726920, - :block/updated-at 1668669340900} - {:block/name "tools for thought rocks logseq", - :block/created-at 1646357158471, - :block/updated-at 1646357158471} - {:block/name "tools for thought rocks open collective", - :block/created-at 1668391470219, - :block/updated-at 1668391496857} - {:block/name "toolsforthoughtrocks", - :block/created-at 1668474726916, - :block/updated-at 1668474726916} - {:block/name "twitter", - :block/created-at 1660672021350, - :block/updated-at 1660672021350} - {:block/name "utilities", - :block/created-at 1668988496810, - :block/updated-at 1668988496810} - {:block/name "vancouver", - :block/created-at 1668391335730, - :block/updated-at 1668391335730} - {:block/name "video", - :block/created-at 1660672020917, - :block/updated-at 1660672020917} - {:block/name "vimeo", - :block/created-at 1668463378644, - :block/updated-at 1668463378644} - {:block/name "vint cerf", - :block/created-at 1646357158471, - :block/updated-at 1646357158471} - {:block/name "visual meta", - :block/created-at 1646357158471, - :block/updated-at 1646357158471} - {:block/name "volunteer organizer kick off meeting", - :block/created-at 1646357158471, - :block/updated-at 1646357158471} - {:block/name "wait", - :block/created-at 1646357158471, - :block/updated-at 1646357158471} - {:block/name "waiting", - :block/created-at 1646357158473, - :block/updated-at 1646357158473} - {:block/name "walkah", - :block/created-at 1668391236897, - :block/updated-at 1668391236897} - {:block/name "welcome to mastodon", - :block/created-at 1668387247603, - :block/updated-at 1668387247603} - {:block/name "wetware of writing and doing", - :block/created-at 1668474726605, - :block/updated-at 1668474726605} - {:block/name "write of passage", - :block/created-at 1668474726789, - :block/updated-at 1668474726789} - {:block/name "zerodataapp", - :block/created-at 1646357158471, - :block/updated-at 1646357158471}] diff --git a/package-lock.json b/package-lock.json new file mode 100644 index 0000000..e4208c4 --- /dev/null +++ b/package-lock.json @@ -0,0 +1,6213 @@ +{ + "name": "@jackyzha0/quartz", + "version": "4.1.0", + "lockfileVersion": 3, + "requires": true, + "packages": { + "": { + "name": "@jackyzha0/quartz", + "version": "4.1.0", + "license": "MIT", + "dependencies": { + "@clack/prompts": "^0.6.3", + "@floating-ui/dom": "^1.4.0", + "@napi-rs/simple-git": "0.1.9", + "async-mutex": "^0.4.0", + "chalk": "^4.1.2", + "chokidar": "^3.5.3", + "cli-spinner": "^0.2.10", + "d3": "^7.8.5", + "esbuild-sass-plugin": "^2.12.0", + "flexsearch": "0.7.21", + "github-slugger": "^2.0.0", + "globby": "^13.1.4", + "gray-matter": "^4.0.3", + "hast-util-to-html": "^8.0.4", + "hast-util-to-jsx-runtime": "^1.2.0", + "hast-util-to-string": "^2.0.0", + "is-absolute-url": "^4.0.1", + "js-yaml": "^4.1.0", + "lightningcss": "1.21.7", + "mdast-util-find-and-replace": "^2.2.2", + "mdast-util-to-hast": "^12.3.0", + "mdast-util-to-string": "^3.2.0", + "micromorph": "^0.4.5", + "plausible-tracker": "^0.3.8", + "preact": "^10.14.1", + "preact-render-to-string": "^6.0.3", + "pretty-bytes": "^6.1.0", + "pretty-time": "^1.1.0", + "reading-time": "^1.5.0", + "rehype-autolink-headings": "^6.1.1", + "rehype-katex": "^6.0.3", + "rehype-mathjax": "^4.0.3", + "rehype-pretty-code": "^0.10.0", + "rehype-raw": "^6.1.1", + "rehype-slug": "^5.1.0", + "remark": "^14.0.2", + "remark-breaks": "^3.0.3", + "remark-frontmatter": "^4.0.1", + "remark-gfm": "^3.0.1", + "remark-math": "^5.1.1", + "remark-parse": "^10.0.1", + "remark-rehype": "^10.1.0", + "remark-smartypants": "^2.0.0", + "rimraf": "^5.0.1", + "serve-handler": "^6.1.5", + "source-map-support": "^0.5.21", + "to-vfile": "^7.2.4", + "toml": "^3.0.0", + "unified": "^10.1.2", + "unist-util-visit": "^4.1.2", + "vfile": "^5.3.7", + "workerpool": "^6.4.0", + "ws": "^8.13.0", + "yargs": "^17.7.2" + }, + "bin": { + "quartz": "quartz/bootstrap-cli.mjs" + }, + "devDependencies": { + "@types/cli-spinner": "^0.2.1", + "@types/d3": "^7.4.0", + "@types/flexsearch": "^0.7.3", + "@types/hast": "^2.3.4", + "@types/js-yaml": "^4.0.5", + "@types/node": "^20.1.2", + "@types/pretty-time": "^1.1.2", + "@types/source-map-support": "^0.5.6", + "@types/workerpool": "^6.4.0", + "@types/ws": "^8.5.5", + "@types/yargs": "^17.0.24", + "esbuild": "0.19.2", + "prettier": "^3.0.0", + "tsx": "^3.12.7", + "typescript": "^5.0.4" + }, + "engines": { + "node": ">=18.14", + "npm": ">=9.3.1" + } + }, + "node_modules/@clack/core": { + "version": "0.3.2", + "resolved": "https://registry.npmjs.org/@clack/core/-/core-0.3.2.tgz", + "integrity": "sha512-FZnsNynwGDIDktx6PEZK1EuCkFpY4ldEX6VYvfl0dqeoLPb9Jpw1xoUXaVcGR8ExmYNm1w2vdGdJkEUYD/2pqg==", + "dependencies": { + "picocolors": "^1.0.0", + "sisteransi": "^1.0.5" + } + }, + "node_modules/@clack/prompts": { + "version": "0.6.3", + "resolved": "https://registry.npmjs.org/@clack/prompts/-/prompts-0.6.3.tgz", + "integrity": "sha512-AM+kFmAHawpUQv2q9+mcB6jLKxXGjgu/r2EQjEwujgpCdzrST6BJqYw00GRn56/L/Izw5U7ImoLmy00X/r80Pw==", + "bundleDependencies": [ + "is-unicode-supported" + ], + "dependencies": { + "@clack/core": "^0.3.2", + "is-unicode-supported": "*", + "picocolors": "^1.0.0", + "sisteransi": "^1.0.5" + } + }, + "node_modules/@clack/prompts/node_modules/is-unicode-supported": { + "version": "1.3.0", + "inBundle": true, + "license": "MIT", + "engines": { + "node": ">=12" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/@esbuild-kit/cjs-loader": { + "version": "2.4.2", + "resolved": "https://registry.npmjs.org/@esbuild-kit/cjs-loader/-/cjs-loader-2.4.2.tgz", + "integrity": "sha512-BDXFbYOJzT/NBEtp71cvsrGPwGAMGRB/349rwKuoxNSiKjPraNNnlK6MIIabViCjqZugu6j+xeMDlEkWdHHJSg==", + "dev": true, + "dependencies": { + "@esbuild-kit/core-utils": "^3.0.0", + "get-tsconfig": "^4.4.0" + } + }, + "node_modules/@esbuild-kit/core-utils": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/@esbuild-kit/core-utils/-/core-utils-3.1.0.tgz", + "integrity": "sha512-Uuk8RpCg/7fdHSceR1M6XbSZFSuMrxcePFuGgyvsBn+u339dk5OeL4jv2EojwTN2st/unJGsVm4qHWjWNmJ/tw==", + "dev": true, + "dependencies": { + "esbuild": "~0.17.6", + "source-map-support": "^0.5.21" + } + }, + "node_modules/@esbuild-kit/core-utils/node_modules/@esbuild/android-arm": { + "version": "0.17.19", + "resolved": "https://registry.npmjs.org/@esbuild/android-arm/-/android-arm-0.17.19.tgz", + "integrity": "sha512-rIKddzqhmav7MSmoFCmDIb6e2W57geRsM94gV2l38fzhXMwq7hZoClug9USI2pFRGL06f4IOPHHpFNOkWieR8A==", + "cpu": [ + "arm" + ], + "dev": true, + "optional": true, + "os": [ + "android" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild-kit/core-utils/node_modules/@esbuild/android-arm64": { + "version": "0.17.19", + "resolved": "https://registry.npmjs.org/@esbuild/android-arm64/-/android-arm64-0.17.19.tgz", + "integrity": "sha512-KBMWvEZooR7+kzY0BtbTQn0OAYY7CsiydT63pVEaPtVYF0hXbUaOyZog37DKxK7NF3XacBJOpYT4adIJh+avxA==", + "cpu": [ + "arm64" + ], + "dev": true, + "optional": true, + "os": [ + "android" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild-kit/core-utils/node_modules/@esbuild/android-x64": { + "version": "0.17.19", + "resolved": "https://registry.npmjs.org/@esbuild/android-x64/-/android-x64-0.17.19.tgz", + "integrity": "sha512-uUTTc4xGNDT7YSArp/zbtmbhO0uEEK9/ETW29Wk1thYUJBz3IVnvgEiEwEa9IeLyvnpKrWK64Utw2bgUmDveww==", + "cpu": [ + "x64" + ], + "dev": true, + "optional": true, + "os": [ + "android" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild-kit/core-utils/node_modules/@esbuild/darwin-arm64": { + "version": "0.17.19", + "resolved": "https://registry.npmjs.org/@esbuild/darwin-arm64/-/darwin-arm64-0.17.19.tgz", + "integrity": "sha512-80wEoCfF/hFKM6WE1FyBHc9SfUblloAWx6FJkFWTWiCoht9Mc0ARGEM47e67W9rI09YoUxJL68WHfDRYEAvOhg==", + "cpu": [ + "arm64" + ], + "dev": true, + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild-kit/core-utils/node_modules/@esbuild/darwin-x64": { + "version": "0.17.19", + "resolved": "https://registry.npmjs.org/@esbuild/darwin-x64/-/darwin-x64-0.17.19.tgz", + "integrity": "sha512-IJM4JJsLhRYr9xdtLytPLSH9k/oxR3boaUIYiHkAawtwNOXKE8KoU8tMvryogdcT8AU+Bflmh81Xn6Q0vTZbQw==", + "cpu": [ + "x64" + ], + "dev": true, + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild-kit/core-utils/node_modules/@esbuild/freebsd-arm64": { + "version": "0.17.19", + "resolved": "https://registry.npmjs.org/@esbuild/freebsd-arm64/-/freebsd-arm64-0.17.19.tgz", + "integrity": "sha512-pBwbc7DufluUeGdjSU5Si+P3SoMF5DQ/F/UmTSb8HXO80ZEAJmrykPyzo1IfNbAoaqw48YRpv8shwd1NoI0jcQ==", + "cpu": [ + "arm64" + ], + "dev": true, + "optional": true, + "os": [ + "freebsd" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild-kit/core-utils/node_modules/@esbuild/freebsd-x64": { + "version": "0.17.19", + "resolved": "https://registry.npmjs.org/@esbuild/freebsd-x64/-/freebsd-x64-0.17.19.tgz", + "integrity": "sha512-4lu+n8Wk0XlajEhbEffdy2xy53dpR06SlzvhGByyg36qJw6Kpfk7cp45DR/62aPH9mtJRmIyrXAS5UWBrJT6TQ==", + "cpu": [ + "x64" + ], + "dev": true, + "optional": true, + "os": [ + "freebsd" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild-kit/core-utils/node_modules/@esbuild/linux-arm": { + "version": "0.17.19", + "resolved": "https://registry.npmjs.org/@esbuild/linux-arm/-/linux-arm-0.17.19.tgz", + "integrity": "sha512-cdmT3KxjlOQ/gZ2cjfrQOtmhG4HJs6hhvm3mWSRDPtZ/lP5oe8FWceS10JaSJC13GBd4eH/haHnqf7hhGNLerA==", + "cpu": [ + "arm" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild-kit/core-utils/node_modules/@esbuild/linux-arm64": { + "version": "0.17.19", + "resolved": "https://registry.npmjs.org/@esbuild/linux-arm64/-/linux-arm64-0.17.19.tgz", + "integrity": "sha512-ct1Tg3WGwd3P+oZYqic+YZF4snNl2bsnMKRkb3ozHmnM0dGWuxcPTTntAF6bOP0Sp4x0PjSF+4uHQ1xvxfRKqg==", + "cpu": [ + "arm64" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild-kit/core-utils/node_modules/@esbuild/linux-ia32": { + "version": "0.17.19", + "resolved": "https://registry.npmjs.org/@esbuild/linux-ia32/-/linux-ia32-0.17.19.tgz", + "integrity": "sha512-w4IRhSy1VbsNxHRQpeGCHEmibqdTUx61Vc38APcsRbuVgK0OPEnQ0YD39Brymn96mOx48Y2laBQGqgZ0j9w6SQ==", + "cpu": [ + "ia32" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild-kit/core-utils/node_modules/@esbuild/linux-loong64": { + "version": "0.17.19", + "resolved": "https://registry.npmjs.org/@esbuild/linux-loong64/-/linux-loong64-0.17.19.tgz", + "integrity": "sha512-2iAngUbBPMq439a+z//gE+9WBldoMp1s5GWsUSgqHLzLJ9WoZLZhpwWuym0u0u/4XmZ3gpHmzV84PonE+9IIdQ==", + "cpu": [ + "loong64" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild-kit/core-utils/node_modules/@esbuild/linux-mips64el": { + "version": "0.17.19", + "resolved": "https://registry.npmjs.org/@esbuild/linux-mips64el/-/linux-mips64el-0.17.19.tgz", + "integrity": "sha512-LKJltc4LVdMKHsrFe4MGNPp0hqDFA1Wpt3jE1gEyM3nKUvOiO//9PheZZHfYRfYl6AwdTH4aTcXSqBerX0ml4A==", + "cpu": [ + "mips64el" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild-kit/core-utils/node_modules/@esbuild/linux-ppc64": { + "version": "0.17.19", + "resolved": "https://registry.npmjs.org/@esbuild/linux-ppc64/-/linux-ppc64-0.17.19.tgz", + "integrity": "sha512-/c/DGybs95WXNS8y3Ti/ytqETiW7EU44MEKuCAcpPto3YjQbyK3IQVKfF6nbghD7EcLUGl0NbiL5Rt5DMhn5tg==", + "cpu": [ + "ppc64" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild-kit/core-utils/node_modules/@esbuild/linux-riscv64": { + "version": "0.17.19", + "resolved": "https://registry.npmjs.org/@esbuild/linux-riscv64/-/linux-riscv64-0.17.19.tgz", + "integrity": "sha512-FC3nUAWhvFoutlhAkgHf8f5HwFWUL6bYdvLc/TTuxKlvLi3+pPzdZiFKSWz/PF30TB1K19SuCxDTI5KcqASJqA==", + "cpu": [ + "riscv64" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild-kit/core-utils/node_modules/@esbuild/linux-s390x": { + "version": "0.17.19", + "resolved": "https://registry.npmjs.org/@esbuild/linux-s390x/-/linux-s390x-0.17.19.tgz", + "integrity": "sha512-IbFsFbxMWLuKEbH+7sTkKzL6NJmG2vRyy6K7JJo55w+8xDk7RElYn6xvXtDW8HCfoKBFK69f3pgBJSUSQPr+4Q==", + "cpu": [ + "s390x" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild-kit/core-utils/node_modules/@esbuild/linux-x64": { + "version": "0.17.19", + "resolved": "https://registry.npmjs.org/@esbuild/linux-x64/-/linux-x64-0.17.19.tgz", + "integrity": "sha512-68ngA9lg2H6zkZcyp22tsVt38mlhWde8l3eJLWkyLrp4HwMUr3c1s/M2t7+kHIhvMjglIBrFpncX1SzMckomGw==", + "cpu": [ + "x64" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild-kit/core-utils/node_modules/@esbuild/netbsd-x64": { + "version": "0.17.19", + "resolved": "https://registry.npmjs.org/@esbuild/netbsd-x64/-/netbsd-x64-0.17.19.tgz", + "integrity": "sha512-CwFq42rXCR8TYIjIfpXCbRX0rp1jo6cPIUPSaWwzbVI4aOfX96OXY8M6KNmtPcg7QjYeDmN+DD0Wp3LaBOLf4Q==", + "cpu": [ + "x64" + ], + "dev": true, + "optional": true, + "os": [ + "netbsd" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild-kit/core-utils/node_modules/@esbuild/openbsd-x64": { + "version": "0.17.19", + "resolved": "https://registry.npmjs.org/@esbuild/openbsd-x64/-/openbsd-x64-0.17.19.tgz", + "integrity": "sha512-cnq5brJYrSZ2CF6c35eCmviIN3k3RczmHz8eYaVlNasVqsNY+JKohZU5MKmaOI+KkllCdzOKKdPs762VCPC20g==", + "cpu": [ + "x64" + ], + "dev": true, + "optional": true, + "os": [ + "openbsd" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild-kit/core-utils/node_modules/@esbuild/sunos-x64": { + "version": "0.17.19", + "resolved": "https://registry.npmjs.org/@esbuild/sunos-x64/-/sunos-x64-0.17.19.tgz", + "integrity": "sha512-vCRT7yP3zX+bKWFeP/zdS6SqdWB8OIpaRq/mbXQxTGHnIxspRtigpkUcDMlSCOejlHowLqII7K2JKevwyRP2rg==", + "cpu": [ + "x64" + ], + "dev": true, + "optional": true, + "os": [ + "sunos" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild-kit/core-utils/node_modules/@esbuild/win32-arm64": { + "version": "0.17.19", + "resolved": "https://registry.npmjs.org/@esbuild/win32-arm64/-/win32-arm64-0.17.19.tgz", + "integrity": "sha512-yYx+8jwowUstVdorcMdNlzklLYhPxjniHWFKgRqH7IFlUEa0Umu3KuYplf1HUZZ422e3NU9F4LGb+4O0Kdcaag==", + "cpu": [ + "arm64" + ], + "dev": true, + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild-kit/core-utils/node_modules/@esbuild/win32-ia32": { + "version": "0.17.19", + "resolved": "https://registry.npmjs.org/@esbuild/win32-ia32/-/win32-ia32-0.17.19.tgz", + "integrity": "sha512-eggDKanJszUtCdlVs0RB+h35wNlb5v4TWEkq4vZcmVt5u/HiDZrTXe2bWFQUez3RgNHwx/x4sk5++4NSSicKkw==", + "cpu": [ + "ia32" + ], + "dev": true, + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild-kit/core-utils/node_modules/@esbuild/win32-x64": { + "version": "0.17.19", + "resolved": "https://registry.npmjs.org/@esbuild/win32-x64/-/win32-x64-0.17.19.tgz", + "integrity": "sha512-lAhycmKnVOuRYNtRtatQR1LPQf2oYCkRGkSFnseDAKPl8lu5SOsK/e1sXe5a0Pc5kHIHe6P2I/ilntNv2xf3cA==", + "cpu": [ + "x64" + ], + "dev": true, + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild-kit/core-utils/node_modules/esbuild": { + "version": "0.17.19", + "resolved": "https://registry.npmjs.org/esbuild/-/esbuild-0.17.19.tgz", + "integrity": "sha512-XQ0jAPFkK/u3LcVRcvVHQcTIqD6E2H1fvZMA5dQPSOWb3suUbWbfbRf94pjc0bNzRYLfIrDRQXr7X+LHIm5oHw==", + "dev": true, + "hasInstallScript": true, + "bin": { + "esbuild": "bin/esbuild" + }, + "engines": { + "node": ">=12" + }, + "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" + } + }, + "node_modules/@esbuild-kit/esm-loader": { + "version": "2.5.5", + "resolved": "https://registry.npmjs.org/@esbuild-kit/esm-loader/-/esm-loader-2.5.5.tgz", + "integrity": "sha512-Qwfvj/qoPbClxCRNuac1Du01r9gvNOT+pMYtJDapfB1eoGN1YlJ1BixLyL9WVENRx5RXgNLdfYdx/CuswlGhMw==", + "dev": true, + "dependencies": { + "@esbuild-kit/core-utils": "^3.0.0", + "get-tsconfig": "^4.4.0" + } + }, + "node_modules/@esbuild/android-arm": { + "version": "0.19.2", + "resolved": "https://registry.npmjs.org/@esbuild/android-arm/-/android-arm-0.19.2.tgz", + "integrity": "sha512-tM8yLeYVe7pRyAu9VMi/Q7aunpLwD139EY1S99xbQkT4/q2qa6eA4ige/WJQYdJ8GBL1K33pPFhPfPdJ/WzT8Q==", + "cpu": [ + "arm" + ], + "optional": true, + "os": [ + "android" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/android-arm64": { + "version": "0.19.2", + "resolved": "https://registry.npmjs.org/@esbuild/android-arm64/-/android-arm64-0.19.2.tgz", + "integrity": "sha512-lsB65vAbe90I/Qe10OjkmrdxSX4UJDjosDgb8sZUKcg3oefEuW2OT2Vozz8ef7wrJbMcmhvCC+hciF8jY/uAkw==", + "cpu": [ + "arm64" + ], + "optional": true, + "os": [ + "android" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/android-x64": { + "version": "0.19.2", + "resolved": "https://registry.npmjs.org/@esbuild/android-x64/-/android-x64-0.19.2.tgz", + "integrity": "sha512-qK/TpmHt2M/Hg82WXHRc/W/2SGo/l1thtDHZWqFq7oi24AjZ4O/CpPSu6ZuYKFkEgmZlFoa7CooAyYmuvnaG8w==", + "cpu": [ + "x64" + ], + "optional": true, + "os": [ + "android" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/darwin-arm64": { + "version": "0.19.2", + "resolved": "https://registry.npmjs.org/@esbuild/darwin-arm64/-/darwin-arm64-0.19.2.tgz", + "integrity": "sha512-Ora8JokrvrzEPEpZO18ZYXkH4asCdc1DLdcVy8TGf5eWtPO1Ie4WroEJzwI52ZGtpODy3+m0a2yEX9l+KUn0tA==", + "cpu": [ + "arm64" + ], + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/darwin-x64": { + "version": "0.19.2", + "resolved": "https://registry.npmjs.org/@esbuild/darwin-x64/-/darwin-x64-0.19.2.tgz", + "integrity": "sha512-tP+B5UuIbbFMj2hQaUr6EALlHOIOmlLM2FK7jeFBobPy2ERdohI4Ka6ZFjZ1ZYsrHE/hZimGuU90jusRE0pwDw==", + "cpu": [ + "x64" + ], + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/freebsd-arm64": { + "version": "0.19.2", + "resolved": "https://registry.npmjs.org/@esbuild/freebsd-arm64/-/freebsd-arm64-0.19.2.tgz", + "integrity": "sha512-YbPY2kc0acfzL1VPVK6EnAlig4f+l8xmq36OZkU0jzBVHcOTyQDhnKQaLzZudNJQyymd9OqQezeaBgkTGdTGeQ==", + "cpu": [ + "arm64" + ], + "optional": true, + "os": [ + "freebsd" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/freebsd-x64": { + "version": "0.19.2", + "resolved": "https://registry.npmjs.org/@esbuild/freebsd-x64/-/freebsd-x64-0.19.2.tgz", + "integrity": "sha512-nSO5uZT2clM6hosjWHAsS15hLrwCvIWx+b2e3lZ3MwbYSaXwvfO528OF+dLjas1g3bZonciivI8qKR/Hm7IWGw==", + "cpu": [ + "x64" + ], + "optional": true, + "os": [ + "freebsd" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-arm": { + "version": "0.19.2", + "resolved": "https://registry.npmjs.org/@esbuild/linux-arm/-/linux-arm-0.19.2.tgz", + "integrity": "sha512-Odalh8hICg7SOD7XCj0YLpYCEc+6mkoq63UnExDCiRA2wXEmGlK5JVrW50vZR9Qz4qkvqnHcpH+OFEggO3PgTg==", + "cpu": [ + "arm" + ], + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-arm64": { + "version": "0.19.2", + "resolved": "https://registry.npmjs.org/@esbuild/linux-arm64/-/linux-arm64-0.19.2.tgz", + "integrity": "sha512-ig2P7GeG//zWlU0AggA3pV1h5gdix0MA3wgB+NsnBXViwiGgY77fuN9Wr5uoCrs2YzaYfogXgsWZbm+HGr09xg==", + "cpu": [ + "arm64" + ], + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-ia32": { + "version": "0.19.2", + "resolved": "https://registry.npmjs.org/@esbuild/linux-ia32/-/linux-ia32-0.19.2.tgz", + "integrity": "sha512-mLfp0ziRPOLSTek0Gd9T5B8AtzKAkoZE70fneiiyPlSnUKKI4lp+mGEnQXcQEHLJAcIYDPSyBvsUbKUG2ri/XQ==", + "cpu": [ + "ia32" + ], + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-loong64": { + "version": "0.19.2", + "resolved": "https://registry.npmjs.org/@esbuild/linux-loong64/-/linux-loong64-0.19.2.tgz", + "integrity": "sha512-hn28+JNDTxxCpnYjdDYVMNTR3SKavyLlCHHkufHV91fkewpIyQchS1d8wSbmXhs1fiYDpNww8KTFlJ1dHsxeSw==", + "cpu": [ + "loong64" + ], + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-mips64el": { + "version": "0.19.2", + "resolved": "https://registry.npmjs.org/@esbuild/linux-mips64el/-/linux-mips64el-0.19.2.tgz", + "integrity": "sha512-KbXaC0Sejt7vD2fEgPoIKb6nxkfYW9OmFUK9XQE4//PvGIxNIfPk1NmlHmMg6f25x57rpmEFrn1OotASYIAaTg==", + "cpu": [ + "mips64el" + ], + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-ppc64": { + "version": "0.19.2", + "resolved": "https://registry.npmjs.org/@esbuild/linux-ppc64/-/linux-ppc64-0.19.2.tgz", + "integrity": "sha512-dJ0kE8KTqbiHtA3Fc/zn7lCd7pqVr4JcT0JqOnbj4LLzYnp+7h8Qi4yjfq42ZlHfhOCM42rBh0EwHYLL6LEzcw==", + "cpu": [ + "ppc64" + ], + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-riscv64": { + "version": "0.19.2", + "resolved": "https://registry.npmjs.org/@esbuild/linux-riscv64/-/linux-riscv64-0.19.2.tgz", + "integrity": "sha512-7Z/jKNFufZ/bbu4INqqCN6DDlrmOTmdw6D0gH+6Y7auok2r02Ur661qPuXidPOJ+FSgbEeQnnAGgsVynfLuOEw==", + "cpu": [ + "riscv64" + ], + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-s390x": { + "version": "0.19.2", + "resolved": "https://registry.npmjs.org/@esbuild/linux-s390x/-/linux-s390x-0.19.2.tgz", + "integrity": "sha512-U+RinR6aXXABFCcAY4gSlv4CL1oOVvSSCdseQmGO66H+XyuQGZIUdhG56SZaDJQcLmrSfRmx5XZOWyCJPRqS7g==", + "cpu": [ + "s390x" + ], + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-x64": { + "version": "0.19.2", + "resolved": "https://registry.npmjs.org/@esbuild/linux-x64/-/linux-x64-0.19.2.tgz", + "integrity": "sha512-oxzHTEv6VPm3XXNaHPyUTTte+3wGv7qVQtqaZCrgstI16gCuhNOtBXLEBkBREP57YTd68P0VgDgG73jSD8bwXQ==", + "cpu": [ + "x64" + ], + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/netbsd-x64": { + "version": "0.19.2", + "resolved": "https://registry.npmjs.org/@esbuild/netbsd-x64/-/netbsd-x64-0.19.2.tgz", + "integrity": "sha512-WNa5zZk1XpTTwMDompZmvQLHszDDDN7lYjEHCUmAGB83Bgs20EMs7ICD+oKeT6xt4phV4NDdSi/8OfjPbSbZfQ==", + "cpu": [ + "x64" + ], + "optional": true, + "os": [ + "netbsd" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/openbsd-x64": { + "version": "0.19.2", + "resolved": "https://registry.npmjs.org/@esbuild/openbsd-x64/-/openbsd-x64-0.19.2.tgz", + "integrity": "sha512-S6kI1aT3S++Dedb7vxIuUOb3oAxqxk2Rh5rOXOTYnzN8JzW1VzBd+IqPiSpgitu45042SYD3HCoEyhLKQcDFDw==", + "cpu": [ + "x64" + ], + "optional": true, + "os": [ + "openbsd" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/sunos-x64": { + "version": "0.19.2", + "resolved": "https://registry.npmjs.org/@esbuild/sunos-x64/-/sunos-x64-0.19.2.tgz", + "integrity": "sha512-VXSSMsmb+Z8LbsQGcBMiM+fYObDNRm8p7tkUDMPG/g4fhFX5DEFmjxIEa3N8Zr96SjsJ1woAhF0DUnS3MF3ARw==", + "cpu": [ + "x64" + ], + "optional": true, + "os": [ + "sunos" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/win32-arm64": { + "version": "0.19.2", + "resolved": "https://registry.npmjs.org/@esbuild/win32-arm64/-/win32-arm64-0.19.2.tgz", + "integrity": "sha512-5NayUlSAyb5PQYFAU9x3bHdsqB88RC3aM9lKDAz4X1mo/EchMIT1Q+pSeBXNgkfNmRecLXA0O8xP+x8V+g/LKg==", + "cpu": [ + "arm64" + ], + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/win32-ia32": { + "version": "0.19.2", + "resolved": "https://registry.npmjs.org/@esbuild/win32-ia32/-/win32-ia32-0.19.2.tgz", + "integrity": "sha512-47gL/ek1v36iN0wL9L4Q2MFdujR0poLZMJwhO2/N3gA89jgHp4MR8DKCmwYtGNksbfJb9JoTtbkoe6sDhg2QTA==", + "cpu": [ + "ia32" + ], + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/win32-x64": { + "version": "0.19.2", + "resolved": "https://registry.npmjs.org/@esbuild/win32-x64/-/win32-x64-0.19.2.tgz", + "integrity": "sha512-tcuhV7ncXBqbt/Ybf0IyrMcwVOAPDckMK9rXNHtF17UTK18OKLpg08glminN06pt2WCoALhXdLfSPbVvK/6fxw==", + "cpu": [ + "x64" + ], + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@floating-ui/core": { + "version": "1.3.1", + "resolved": "https://registry.npmjs.org/@floating-ui/core/-/core-1.3.1.tgz", + "integrity": "sha512-Bu+AMaXNjrpjh41znzHqaz3r2Nr8hHuHZT6V2LBKMhyMl0FgKA62PNYbqnfgmzOhoWZj70Zecisbo4H1rotP5g==" + }, + "node_modules/@floating-ui/dom": { + "version": "1.4.3", + "resolved": "https://registry.npmjs.org/@floating-ui/dom/-/dom-1.4.3.tgz", + "integrity": "sha512-nB/68NyaQlcdY22L+Fgd1HERQ7UGv7XFN+tPxwrEfQL4nKtAP/jIZnZtpUlXbtV+VEGHh6W/63Gy2C5biWI3sA==", + "dependencies": { + "@floating-ui/core": "^1.3.1" + } + }, + "node_modules/@isaacs/cliui": { + "version": "8.0.2", + "resolved": "https://registry.npmjs.org/@isaacs/cliui/-/cliui-8.0.2.tgz", + "integrity": "sha512-O8jcjabXaleOG9DQ0+ARXWZBTfnP4WNAqzuiJK7ll44AmxGKv/J2M4TPjxjY3znBCfvBXFzucm1twdyFybFqEA==", + "dependencies": { + "string-width": "^5.1.2", + "string-width-cjs": "npm:string-width@^4.2.0", + "strip-ansi": "^7.0.1", + "strip-ansi-cjs": "npm:strip-ansi@^6.0.1", + "wrap-ansi": "^8.1.0", + "wrap-ansi-cjs": "npm:wrap-ansi@^7.0.0" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/@napi-rs/simple-git": { + "version": "0.1.9", + "resolved": "https://registry.npmjs.org/@napi-rs/simple-git/-/simple-git-0.1.9.tgz", + "integrity": "sha512-qKzDS0+VjMvVyU28px+C6zlD1HKy83NIdYzfMQWa/g/V1iG/Ic8uwrS2ihHfm7mp7X0PPrmINLiTTi6ieUIKfw==", + "engines": { + "node": ">= 10" + }, + "optionalDependencies": { + "@napi-rs/simple-git-android-arm-eabi": "0.1.9", + "@napi-rs/simple-git-android-arm64": "0.1.9", + "@napi-rs/simple-git-darwin-arm64": "0.1.9", + "@napi-rs/simple-git-darwin-x64": "0.1.9", + "@napi-rs/simple-git-linux-arm-gnueabihf": "0.1.9", + "@napi-rs/simple-git-linux-arm64-gnu": "0.1.9", + "@napi-rs/simple-git-linux-arm64-musl": "0.1.9", + "@napi-rs/simple-git-linux-x64-gnu": "0.1.9", + "@napi-rs/simple-git-linux-x64-musl": "0.1.9", + "@napi-rs/simple-git-win32-arm64-msvc": "0.1.9", + "@napi-rs/simple-git-win32-x64-msvc": "0.1.9" + } + }, + "node_modules/@napi-rs/simple-git-android-arm-eabi": { + "version": "0.1.9", + "resolved": "https://registry.npmjs.org/@napi-rs/simple-git-android-arm-eabi/-/simple-git-android-arm-eabi-0.1.9.tgz", + "integrity": "sha512-9D4JnfePMpgL4pg9aMUX7/TIWEUQ+Tgx8n3Pf8TNCMGjUbImJyYsDSLJzbcv9wH7srgn4GRjSizXFJHAPjzEug==", + "cpu": [ + "arm" + ], + "optional": true, + "os": [ + "android" + ], + "engines": { + "node": ">= 10" + } + }, + "node_modules/@napi-rs/simple-git-android-arm64": { + "version": "0.1.9", + "resolved": "https://registry.npmjs.org/@napi-rs/simple-git-android-arm64/-/simple-git-android-arm64-0.1.9.tgz", + "integrity": "sha512-Krilsw0gPrrASZzudNEl9pdLuNbhoTK0j7pUbfB8FRifpPdFB/zouwuEm0aSnsDXN4ftGrmGG82kuiR/2MeoPg==", + "cpu": [ + "arm64" + ], + "optional": true, + "os": [ + "android" + ], + "engines": { + "node": ">= 10" + } + }, + "node_modules/@napi-rs/simple-git-darwin-arm64": { + "version": "0.1.9", + "resolved": "https://registry.npmjs.org/@napi-rs/simple-git-darwin-arm64/-/simple-git-darwin-arm64-0.1.9.tgz", + "integrity": "sha512-H/F09nDgYjv4gcFrZBgdTKkZEepqt0KLYcCJuUADuxkKupmjLdecMhypXLk13AzvLW4UQI7NlLTLDXUFLyr2BA==", + "cpu": [ + "arm64" + ], + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": ">= 10" + } + }, + "node_modules/@napi-rs/simple-git-darwin-x64": { + "version": "0.1.9", + "resolved": "https://registry.npmjs.org/@napi-rs/simple-git-darwin-x64/-/simple-git-darwin-x64-0.1.9.tgz", + "integrity": "sha512-jBR2xS9nVPqmHv0TWz874W0m/d453MGrMeLjB+boK5IPPLhg3AWIZj0aN9jy2Je1BGVAa0w3INIQJtBBeB6kFA==", + "cpu": [ + "x64" + ], + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": ">= 10" + } + }, + "node_modules/@napi-rs/simple-git-linux-arm-gnueabihf": { + "version": "0.1.9", + "resolved": "https://registry.npmjs.org/@napi-rs/simple-git-linux-arm-gnueabihf/-/simple-git-linux-arm-gnueabihf-0.1.9.tgz", + "integrity": "sha512-3n0+VpO4YfZxndZ0sCvsHIvsazd+JmbSjrlTRBCnJeAU1/sfos3skNZtKGZksZhjvd+3o+/GFM8L7Xnv01yggA==", + "cpu": [ + "arm" + ], + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">= 10" + } + }, + "node_modules/@napi-rs/simple-git-linux-arm64-gnu": { + "version": "0.1.9", + "resolved": "https://registry.npmjs.org/@napi-rs/simple-git-linux-arm64-gnu/-/simple-git-linux-arm64-gnu-0.1.9.tgz", + "integrity": "sha512-lIzf0KHU2SKC12vMrWwCtysG2Sdt31VHRPMUiz9lD9t3xwVn8qhFSTn5yDkTeG3rgX6o0p5EKalfQN5BXsJq2w==", + "cpu": [ + "arm64" + ], + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">= 10" + } + }, + "node_modules/@napi-rs/simple-git-linux-arm64-musl": { + "version": "0.1.9", + "resolved": "https://registry.npmjs.org/@napi-rs/simple-git-linux-arm64-musl/-/simple-git-linux-arm64-musl-0.1.9.tgz", + "integrity": "sha512-KQozUoNXrxrB8k741ncWXSiMbjl1AGBGfZV21PANzUM8wH4Yem2bg3kfglYS/QIx3udspsT35I9abu49n7D1/w==", + "cpu": [ + "arm64" + ], + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">= 10" + } + }, + "node_modules/@napi-rs/simple-git-linux-x64-gnu": { + "version": "0.1.9", + "resolved": "https://registry.npmjs.org/@napi-rs/simple-git-linux-x64-gnu/-/simple-git-linux-x64-gnu-0.1.9.tgz", + "integrity": "sha512-O/Niui5mnHPcK3iYC3ui8wgERtJWsQ3Y74W/09t0bL/3dgzGMl4oQt0qTj9dWCsnoGsIEYHPzwCBp/2vqYp/pw==", + "cpu": [ + "x64" + ], + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">= 10" + } + }, + "node_modules/@napi-rs/simple-git-linux-x64-musl": { + "version": "0.1.9", + "resolved": "https://registry.npmjs.org/@napi-rs/simple-git-linux-x64-musl/-/simple-git-linux-x64-musl-0.1.9.tgz", + "integrity": "sha512-L9n+e8Wn3hKr3RsIdY8GaB+ry4xZ4BaGwyKExgoB8nDGQuRUY9oP6p0WA4hWfJvJnU1H6hvo36a5UFPReyBO7A==", + "cpu": [ + "x64" + ], + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">= 10" + } + }, + "node_modules/@napi-rs/simple-git-win32-arm64-msvc": { + "version": "0.1.9", + "resolved": "https://registry.npmjs.org/@napi-rs/simple-git-win32-arm64-msvc/-/simple-git-win32-arm64-msvc-0.1.9.tgz", + "integrity": "sha512-Z6Ja/SZK+lMvRWaxj7wjnvSbAsGrH006sqZo8P8nxKUdZfkVvoCaAWr1r0cfkk2Z3aijLLtD+vKeXGlUPH6gGQ==", + "cpu": [ + "arm64" + ], + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">= 10" + } + }, + "node_modules/@napi-rs/simple-git-win32-x64-msvc": { + "version": "0.1.9", + "resolved": "https://registry.npmjs.org/@napi-rs/simple-git-win32-x64-msvc/-/simple-git-win32-x64-msvc-0.1.9.tgz", + "integrity": "sha512-VAZj1UvC+R2MjKOD3I/Y7dmQlHWAYy4omhReQJRpbCf+oGCBi9CWiIduGqeYEq723nLIKdxP7XjaO0wl1NnUww==", + "cpu": [ + "x64" + ], + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">= 10" + } + }, + "node_modules/@nodelib/fs.scandir": { + "version": "2.1.5", + "resolved": "https://registry.npmjs.org/@nodelib/fs.scandir/-/fs.scandir-2.1.5.tgz", + "integrity": "sha512-vq24Bq3ym5HEQm2NKCr3yXDwjc7vTsEThRDnkp2DK9p1uqLR+DHurm/NOTo0KG7HYHU7eppKZj3MyqYuMBf62g==", + "dependencies": { + "@nodelib/fs.stat": "2.0.5", + "run-parallel": "^1.1.9" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/@nodelib/fs.stat": { + "version": "2.0.5", + "resolved": "https://registry.npmjs.org/@nodelib/fs.stat/-/fs.stat-2.0.5.tgz", + "integrity": "sha512-RkhPPp2zrqDAQA/2jNhnztcPAlv64XdhIp7a7454A5ovI7Bukxgt7MX7udwAu3zg1DcpPU0rz3VV1SeaqvY4+A==", + "engines": { + "node": ">= 8" + } + }, + "node_modules/@nodelib/fs.walk": { + "version": "1.2.8", + "resolved": "https://registry.npmjs.org/@nodelib/fs.walk/-/fs.walk-1.2.8.tgz", + "integrity": "sha512-oGB+UxlgWcgQkgwo8GcEGwemoTFt3FIO9ababBmaGwXIoBKZ+GTy0pP185beGg7Llih/NSHSV2XAs1lnznocSg==", + "dependencies": { + "@nodelib/fs.scandir": "2.1.5", + "fastq": "^1.6.0" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/@pkgjs/parseargs": { + "version": "0.11.0", + "resolved": "https://registry.npmjs.org/@pkgjs/parseargs/-/parseargs-0.11.0.tgz", + "integrity": "sha512-+1VkjdD0QBLPodGrJUeqarH8VAIvQODIbwh9XpP5Syisf7YoQgsJKPNFoqqLQlu+VQ/tVSshMR6loPMn8U+dPg==", + "optional": true, + "engines": { + "node": ">=14" + } + }, + "node_modules/@tootallnate/once": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/@tootallnate/once/-/once-2.0.0.tgz", + "integrity": "sha512-XCuKFP5PS55gnMVu3dty8KPatLqUoy/ZYzDzAGCQ8JNFCkLXzmI7vNHCR+XpbZaMWQK/vQubr7PkYq8g470J/A==", + "engines": { + "node": ">= 10" + } + }, + "node_modules/@types/cli-spinner": { + "version": "0.2.1", + "resolved": "https://registry.npmjs.org/@types/cli-spinner/-/cli-spinner-0.2.1.tgz", + "integrity": "sha512-bsdlZy3LThi9QbsK0GXm5s/e3F6HAJi1tMsIanm9trtoStSlV3gzir9JpfOK40gERMNIVevDTpG5NzSGnYs3QA==", + "dev": true, + "dependencies": { + "@types/node": "*" + } + }, + "node_modules/@types/d3": { + "version": "7.4.0", + "resolved": "https://registry.npmjs.org/@types/d3/-/d3-7.4.0.tgz", + "integrity": "sha512-jIfNVK0ZlxcuRDKtRS/SypEyOQ6UHaFQBKv032X45VvxSJ6Yi5G9behy9h6tNTHTDGh5Vq+KbmBjUWLgY4meCA==", + "dev": true, + "dependencies": { + "@types/d3-array": "*", + "@types/d3-axis": "*", + "@types/d3-brush": "*", + "@types/d3-chord": "*", + "@types/d3-color": "*", + "@types/d3-contour": "*", + "@types/d3-delaunay": "*", + "@types/d3-dispatch": "*", + "@types/d3-drag": "*", + "@types/d3-dsv": "*", + "@types/d3-ease": "*", + "@types/d3-fetch": "*", + "@types/d3-force": "*", + "@types/d3-format": "*", + "@types/d3-geo": "*", + "@types/d3-hierarchy": "*", + "@types/d3-interpolate": "*", + "@types/d3-path": "*", + "@types/d3-polygon": "*", + "@types/d3-quadtree": "*", + "@types/d3-random": "*", + "@types/d3-scale": "*", + "@types/d3-scale-chromatic": "*", + "@types/d3-selection": "*", + "@types/d3-shape": "*", + "@types/d3-time": "*", + "@types/d3-time-format": "*", + "@types/d3-timer": "*", + "@types/d3-transition": "*", + "@types/d3-zoom": "*" + } + }, + "node_modules/@types/d3-array": { + "version": "3.0.5", + "resolved": "https://registry.npmjs.org/@types/d3-array/-/d3-array-3.0.5.tgz", + "integrity": "sha512-Qk7fpJ6qFp+26VeQ47WY0mkwXaiq8+76RJcncDEfMc2ocRzXLO67bLFRNI4OX1aGBoPzsM5Y2T+/m1pldOgD+A==", + "dev": true + }, + "node_modules/@types/d3-axis": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/@types/d3-axis/-/d3-axis-3.0.2.tgz", + "integrity": "sha512-uGC7DBh0TZrU/LY43Fd8Qr+2ja1FKmH07q2FoZFHo1eYl8aj87GhfVoY1saJVJiq24rp1+wpI6BvQJMKgQm8oA==", + "dev": true, + "dependencies": { + "@types/d3-selection": "*" + } + }, + "node_modules/@types/d3-brush": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/@types/d3-brush/-/d3-brush-3.0.2.tgz", + "integrity": "sha512-2TEm8KzUG3N7z0TrSKPmbxByBx54M+S9lHoP2J55QuLU0VSQ9mE96EJSAOVNEqd1bbynMjeTS9VHmz8/bSw8rA==", + "dev": true, + "dependencies": { + "@types/d3-selection": "*" + } + }, + "node_modules/@types/d3-chord": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/@types/d3-chord/-/d3-chord-3.0.2.tgz", + "integrity": "sha512-abT/iLHD3sGZwqMTX1TYCMEulr+wBd0SzyOQnjYNLp7sngdOHYtNkMRI5v3w5thoN+BWtlHVDx2Osvq6fxhZWw==", + "dev": true + }, + "node_modules/@types/d3-color": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/@types/d3-color/-/d3-color-3.1.0.tgz", + "integrity": "sha512-HKuicPHJuvPgCD+np6Se9MQvS6OCbJmOjGvylzMJRlDwUXjKTTXs6Pwgk79O09Vj/ho3u1ofXnhFOaEWWPrlwA==", + "dev": true + }, + "node_modules/@types/d3-contour": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/@types/d3-contour/-/d3-contour-3.0.2.tgz", + "integrity": "sha512-k6/bGDoAGJZnZWaKzeB+9glgXCYGvh6YlluxzBREiVo8f/X2vpTEdgPy9DN7Z2i42PZOZ4JDhVdlTSTSkLDPlQ==", + "dev": true, + "dependencies": { + "@types/d3-array": "*", + "@types/geojson": "*" + } + }, + "node_modules/@types/d3-delaunay": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/@types/d3-delaunay/-/d3-delaunay-6.0.1.tgz", + "integrity": "sha512-tLxQ2sfT0p6sxdG75c6f/ekqxjyYR0+LwPrsO1mbC9YDBzPJhs2HbJJRrn8Ez1DBoHRo2yx7YEATI+8V1nGMnQ==", + "dev": true + }, + "node_modules/@types/d3-dispatch": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/@types/d3-dispatch/-/d3-dispatch-3.0.2.tgz", + "integrity": "sha512-rxN6sHUXEZYCKV05MEh4z4WpPSqIw+aP7n9ZN6WYAAvZoEAghEK1WeVZMZcHRBwyaKflU43PCUAJNjFxCzPDjg==", + "dev": true + }, + "node_modules/@types/d3-drag": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/@types/d3-drag/-/d3-drag-3.0.2.tgz", + "integrity": "sha512-qmODKEDvyKWVHcWWCOVcuVcOwikLVsyc4q4EBJMREsoQnR2Qoc2cZQUyFUPgO9q4S3qdSqJKBsuefv+h0Qy+tw==", + "dev": true, + "dependencies": { + "@types/d3-selection": "*" + } + }, + "node_modules/@types/d3-dsv": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/@types/d3-dsv/-/d3-dsv-3.0.1.tgz", + "integrity": "sha512-76pBHCMTvPLt44wFOieouXcGXWOF0AJCceUvaFkxSZEu4VDUdv93JfpMa6VGNFs01FHfuP4a5Ou68eRG1KBfTw==", + "dev": true + }, + "node_modules/@types/d3-ease": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/@types/d3-ease/-/d3-ease-3.0.0.tgz", + "integrity": "sha512-aMo4eaAOijJjA6uU+GIeW018dvy9+oH5Y2VPPzjjfxevvGQ/oRDs+tfYC9b50Q4BygRR8yE2QCLsrT0WtAVseA==", + "dev": true + }, + "node_modules/@types/d3-fetch": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/@types/d3-fetch/-/d3-fetch-3.0.2.tgz", + "integrity": "sha512-gllwYWozWfbep16N9fByNBDTkJW/SyhH6SGRlXloR7WdtAaBui4plTP+gbUgiEot7vGw/ZZop1yDZlgXXSuzjA==", + "dev": true, + "dependencies": { + "@types/d3-dsv": "*" + } + }, + "node_modules/@types/d3-force": { + "version": "3.0.4", + "resolved": "https://registry.npmjs.org/@types/d3-force/-/d3-force-3.0.4.tgz", + "integrity": "sha512-q7xbVLrWcXvSBBEoadowIUJ7sRpS1yvgMWnzHJggFy5cUZBq2HZL5k/pBSm0GdYWS1vs5/EDwMjSKF55PDY4Aw==", + "dev": true + }, + "node_modules/@types/d3-format": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/@types/d3-format/-/d3-format-3.0.1.tgz", + "integrity": "sha512-5KY70ifCCzorkLuIkDe0Z9YTf9RR2CjBX1iaJG+rgM/cPP+sO+q9YdQ9WdhQcgPj1EQiJ2/0+yUkkziTG6Lubg==", + "dev": true + }, + "node_modules/@types/d3-geo": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/@types/d3-geo/-/d3-geo-3.0.3.tgz", + "integrity": "sha512-bK9uZJS3vuDCNeeXQ4z3u0E7OeJZXjUgzFdSOtNtMCJCLvDtWDwfpRVWlyt3y8EvRzI0ccOu9xlMVirawolSCw==", + "dev": true, + "dependencies": { + "@types/geojson": "*" + } + }, + "node_modules/@types/d3-hierarchy": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/@types/d3-hierarchy/-/d3-hierarchy-3.1.2.tgz", + "integrity": "sha512-9hjRTVoZjRFR6xo8igAJyNXQyPX6Aq++Nhb5ebrUF414dv4jr2MitM2fWiOY475wa3Za7TOS2Gh9fmqEhLTt0A==", + "dev": true + }, + "node_modules/@types/d3-interpolate": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/@types/d3-interpolate/-/d3-interpolate-3.0.1.tgz", + "integrity": "sha512-jx5leotSeac3jr0RePOH1KdR9rISG91QIE4Q2PYTu4OymLTZfA3SrnURSLzKH48HmXVUru50b8nje4E79oQSQw==", + "dev": true, + "dependencies": { + "@types/d3-color": "*" + } + }, + "node_modules/@types/d3-path": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/@types/d3-path/-/d3-path-3.0.0.tgz", + "integrity": "sha512-0g/A+mZXgFkQxN3HniRDbXMN79K3CdTpLsevj+PXiTcb2hVyvkZUBg37StmgCQkaD84cUJ4uaDAWq7UJOQy2Tg==", + "dev": true + }, + "node_modules/@types/d3-polygon": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/@types/d3-polygon/-/d3-polygon-3.0.0.tgz", + "integrity": "sha512-D49z4DyzTKXM0sGKVqiTDTYr+DHg/uxsiWDAkNrwXYuiZVd9o9wXZIo+YsHkifOiyBkmSWlEngHCQme54/hnHw==", + "dev": true + }, + "node_modules/@types/d3-quadtree": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/@types/d3-quadtree/-/d3-quadtree-3.0.2.tgz", + "integrity": "sha512-QNcK8Jguvc8lU+4OfeNx+qnVy7c0VrDJ+CCVFS9srBo2GL9Y18CnIxBdTF3v38flrGy5s1YggcoAiu6s4fLQIw==", + "dev": true + }, + "node_modules/@types/d3-random": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/@types/d3-random/-/d3-random-3.0.1.tgz", + "integrity": "sha512-IIE6YTekGczpLYo/HehAy3JGF1ty7+usI97LqraNa8IiDur+L44d0VOjAvFQWJVdZOJHukUJw+ZdZBlgeUsHOQ==", + "dev": true + }, + "node_modules/@types/d3-scale": { + "version": "4.0.3", + "resolved": "https://registry.npmjs.org/@types/d3-scale/-/d3-scale-4.0.3.tgz", + "integrity": "sha512-PATBiMCpvHJSMtZAMEhc2WyL+hnzarKzI6wAHYjhsonjWJYGq5BXTzQjv4l8m2jO183/4wZ90rKvSeT7o72xNQ==", + "dev": true, + "dependencies": { + "@types/d3-time": "*" + } + }, + "node_modules/@types/d3-scale-chromatic": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/@types/d3-scale-chromatic/-/d3-scale-chromatic-3.0.0.tgz", + "integrity": "sha512-dsoJGEIShosKVRBZB0Vo3C8nqSDqVGujJU6tPznsBJxNJNwMF8utmS83nvCBKQYPpjCzaaHcrf66iTRpZosLPw==", + "dev": true + }, + "node_modules/@types/d3-selection": { + "version": "3.0.5", + "resolved": "https://registry.npmjs.org/@types/d3-selection/-/d3-selection-3.0.5.tgz", + "integrity": "sha512-xCB0z3Hi8eFIqyja3vW8iV01+OHGYR2di/+e+AiOcXIOrY82lcvWW8Ke1DYE/EUVMsBl4Db9RppSBS3X1U6J0w==", + "dev": true + }, + "node_modules/@types/d3-shape": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/@types/d3-shape/-/d3-shape-3.1.1.tgz", + "integrity": "sha512-6Uh86YFF7LGg4PQkuO2oG6EMBRLuW9cbavUW46zkIO5kuS2PfTqo2o9SkgtQzguBHbLgNnU90UNsITpsX1My+A==", + "dev": true, + "dependencies": { + "@types/d3-path": "*" + } + }, + "node_modules/@types/d3-time": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/@types/d3-time/-/d3-time-3.0.0.tgz", + "integrity": "sha512-sZLCdHvBUcNby1cB6Fd3ZBrABbjz3v1Vm90nysCQ6Vt7vd6e/h9Lt7SiJUoEX0l4Dzc7P5llKyhqSi1ycSf1Hg==", + "dev": true + }, + "node_modules/@types/d3-time-format": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/@types/d3-time-format/-/d3-time-format-4.0.0.tgz", + "integrity": "sha512-yjfBUe6DJBsDin2BMIulhSHmr5qNR5Pxs17+oW4DoVPyVIXZ+m6bs7j1UVKP08Emv6jRmYrYqxYzO63mQxy1rw==", + "dev": true + }, + "node_modules/@types/d3-timer": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/@types/d3-timer/-/d3-timer-3.0.0.tgz", + "integrity": "sha512-HNB/9GHqu7Fo8AQiugyJbv6ZxYz58wef0esl4Mv828w1ZKpAshw/uFWVDUcIB9KKFeFKoxS3cHY07FFgtTRZ1g==", + "dev": true + }, + "node_modules/@types/d3-transition": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/@types/d3-transition/-/d3-transition-3.0.3.tgz", + "integrity": "sha512-/S90Od8Id1wgQNvIA8iFv9jRhCiZcGhPd2qX0bKF/PS+y0W5CrXKgIiELd2CvG1mlQrWK/qlYh3VxicqG1ZvgA==", + "dev": true, + "dependencies": { + "@types/d3-selection": "*" + } + }, + "node_modules/@types/d3-zoom": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/@types/d3-zoom/-/d3-zoom-3.0.3.tgz", + "integrity": "sha512-OWk1yYIIWcZ07+igN6BeoG6rqhnJ/pYe+R1qWFM2DtW49zsoSjgb9G5xB0ZXA8hh2jAzey1XuRmMSoXdKw8MDA==", + "dev": true, + "dependencies": { + "@types/d3-interpolate": "*", + "@types/d3-selection": "*" + } + }, + "node_modules/@types/debug": { + "version": "4.1.8", + "resolved": "https://registry.npmjs.org/@types/debug/-/debug-4.1.8.tgz", + "integrity": "sha512-/vPO1EPOs306Cvhwv7KfVfYvOJqA/S/AXjaHQiJboCZzcNDb+TIJFN9/2C9DZ//ijSKWioNyUxD792QmDJ+HKQ==", + "dependencies": { + "@types/ms": "*" + } + }, + "node_modules/@types/flexsearch": { + "version": "0.7.3", + "resolved": "https://registry.npmjs.org/@types/flexsearch/-/flexsearch-0.7.3.tgz", + "integrity": "sha512-HXwADeHEP4exXkCIwy2n1+i0f1ilP1ETQOH5KDOugjkTFZPntWo0Gr8stZOaebkxsdx+k0X/K6obU/+it07ocg==", + "dev": true + }, + "node_modules/@types/geojson": { + "version": "7946.0.10", + "resolved": "https://registry.npmjs.org/@types/geojson/-/geojson-7946.0.10.tgz", + "integrity": "sha512-Nmh0K3iWQJzniTuPRcJn5hxXkfB1T1pgB89SBig5PlJQU5yocazeu4jATJlaA0GYFKWMqDdvYemoSnF2pXgLVA==", + "dev": true + }, + "node_modules/@types/hast": { + "version": "2.3.5", + "resolved": "https://registry.npmjs.org/@types/hast/-/hast-2.3.5.tgz", + "integrity": "sha512-SvQi0L/lNpThgPoleH53cdjB3y9zpLlVjRbqB3rH8hx1jiRSBGAhyjV3H+URFjNVRqt2EdYNrbZE5IsGlNfpRg==", + "dependencies": { + "@types/unist": "^2" + } + }, + "node_modules/@types/js-yaml": { + "version": "4.0.5", + "resolved": "https://registry.npmjs.org/@types/js-yaml/-/js-yaml-4.0.5.tgz", + "integrity": "sha512-FhpRzf927MNQdRZP0J5DLIdTXhjLYzeUTmLAu69mnVksLH9CJY3IuSeEgbKUki7GQZm0WqDkGzyxju2EZGD2wA==", + "dev": true + }, + "node_modules/@types/katex": { + "version": "0.14.0", + "resolved": "https://registry.npmjs.org/@types/katex/-/katex-0.14.0.tgz", + "integrity": "sha512-+2FW2CcT0K3P+JMR8YG846bmDwplKUTsWgT2ENwdQ1UdVfRk3GQrh6Mi4sTopy30gI8Uau5CEqHTDZ6YvWIUPA==" + }, + "node_modules/@types/mathjax": { + "version": "0.0.37", + "resolved": "https://registry.npmjs.org/@types/mathjax/-/mathjax-0.0.37.tgz", + "integrity": "sha512-y0WSZBtBNQwcYipTU/BhgeFu1EZNlFvUNCmkMXV9kBQZq7/o5z82dNVyH3yy2Xv5zzeNeQoHSL4Xm06+EQiH+g==" + }, + "node_modules/@types/mdast": { + "version": "3.0.11", + "resolved": "https://registry.npmjs.org/@types/mdast/-/mdast-3.0.11.tgz", + "integrity": "sha512-Y/uImid8aAwrEA24/1tcRZwpxX3pIFTSilcNDKSPn+Y2iDywSEachzRuvgAYYLR3wpGXAsMbv5lvKLDZLeYPAw==", + "dependencies": { + "@types/unist": "*" + } + }, + "node_modules/@types/ms": { + "version": "0.7.31", + "resolved": "https://registry.npmjs.org/@types/ms/-/ms-0.7.31.tgz", + "integrity": "sha512-iiUgKzV9AuaEkZqkOLDIvlQiL6ltuZd9tGcW3gwpnX8JbuiuhFlEGmmFXEXkN50Cvq7Os88IY2v0dkDqXYWVgA==" + }, + "node_modules/@types/nlcst": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/@types/nlcst/-/nlcst-1.0.0.tgz", + "integrity": "sha512-3TGCfOcy8R8mMQ4CNSNOe3PG66HttvjcLzCoOpvXvDtfWOTi+uT/rxeOKm/qEwbM4SNe1O/PjdiBK2YcTjU4OQ==", + "dependencies": { + "@types/unist": "*" + } + }, + "node_modules/@types/node": { + "version": "20.3.3", + "resolved": "https://registry.npmjs.org/@types/node/-/node-20.3.3.tgz", + "integrity": "sha512-wheIYdr4NYML61AjC8MKj/2jrR/kDQri/CIpVoZwldwhnIrD/j9jIU5bJ8yBKuB2VhpFV7Ab6G2XkBjv9r9Zzw==", + "dev": true + }, + "node_modules/@types/parse5": { + "version": "6.0.3", + "resolved": "https://registry.npmjs.org/@types/parse5/-/parse5-6.0.3.tgz", + "integrity": "sha512-SuT16Q1K51EAVPz1K29DJ/sXjhSQ0zjvsypYJ6tlwVsRV9jwW5Adq2ch8Dq8kDBCkYnELS7N7VNCSB5nC56t/g==" + }, + "node_modules/@types/pretty-time": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/@types/pretty-time/-/pretty-time-1.1.2.tgz", + "integrity": "sha512-4i+Y+O5H80Rh01lY/3Z0hB/UWc4R64ReE83joEpVsIG3iQWpYx66k6pQh1amJNZquKtJQyu/RcfkTtvL0KwssA==", + "dev": true + }, + "node_modules/@types/source-map-support": { + "version": "0.5.6", + "resolved": "https://registry.npmjs.org/@types/source-map-support/-/source-map-support-0.5.6.tgz", + "integrity": "sha512-b2nJ9YyXmkhGaa2b8VLM0kJ04xxwNyijcq12/kDoomCt43qbHBeK2SLNJ9iJmETaAj+bKUT05PQUu3Q66GvLhQ==", + "dev": true, + "dependencies": { + "source-map": "^0.6.0" + } + }, + "node_modules/@types/unist": { + "version": "2.0.6", + "resolved": "https://registry.npmjs.org/@types/unist/-/unist-2.0.6.tgz", + "integrity": "sha512-PBjIUxZHOuj0R15/xuwJYjFi+KZdNFrehocChv4g5hu6aFroHue8m0lBP0POdK2nKzbw0cgV1mws8+V/JAcEkQ==" + }, + "node_modules/@types/workerpool": { + "version": "6.4.0", + "resolved": "https://registry.npmjs.org/@types/workerpool/-/workerpool-6.4.0.tgz", + "integrity": "sha512-SIF2/169pDsLKeM8GQGHkOFifGalDbZgiBSaLUnnlVSRsAOenkAvQ6h4uhV2W+PZZczS+8LQxACwNkSykdT91A==", + "dev": true, + "dependencies": { + "@types/node": "*" + } + }, + "node_modules/@types/ws": { + "version": "8.5.5", + "resolved": "https://registry.npmjs.org/@types/ws/-/ws-8.5.5.tgz", + "integrity": "sha512-lwhs8hktwxSjf9UaZ9tG5M03PGogvFaH8gUgLNbN9HKIg0dvv6q+gkSuJ8HN4/VbyxkuLzCjlN7GquQ0gUJfIg==", + "dev": true, + "dependencies": { + "@types/node": "*" + } + }, + "node_modules/@types/yargs": { + "version": "17.0.24", + "resolved": "https://registry.npmjs.org/@types/yargs/-/yargs-17.0.24.tgz", + "integrity": "sha512-6i0aC7jV6QzQB8ne1joVZ0eSFIstHsCrobmOtghM11yGlH0j43FKL2UhWdELkyps0zuf7qVTUVCCR+tgSlyLLw==", + "dev": true, + "dependencies": { + "@types/yargs-parser": "*" + } + }, + "node_modules/@types/yargs-parser": { + "version": "21.0.0", + "resolved": "https://registry.npmjs.org/@types/yargs-parser/-/yargs-parser-21.0.0.tgz", + "integrity": "sha512-iO9ZQHkZxHn4mSakYV0vFHAVDyEOIJQrV2uZ06HxEPcx+mt8swXoZHIbaaJ2crJYFfErySgktuTZ3BeLz+XmFA==", + "dev": true + }, + "node_modules/abab": { + "version": "2.0.6", + "resolved": "https://registry.npmjs.org/abab/-/abab-2.0.6.tgz", + "integrity": "sha512-j2afSsaIENvHZN2B8GOpF566vZ5WVk5opAiMTvWgaQT8DkbOqsTfvNAvHoRGU2zzP8cPoqys+xHTRDWW8L+/BA==" + }, + "node_modules/acorn": { + "version": "8.10.0", + "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.10.0.tgz", + "integrity": "sha512-F0SAmZ8iUtS//m8DmCTA0jlh6TDKkHQyK6xc6V4KDTyZKA9dnvX9/3sRTVQrWm79glUAZbnmmNcdYwUIHWVybw==", + "bin": { + "acorn": "bin/acorn" + }, + "engines": { + "node": ">=0.4.0" + } + }, + "node_modules/acorn-globals": { + "version": "7.0.1", + "resolved": "https://registry.npmjs.org/acorn-globals/-/acorn-globals-7.0.1.tgz", + "integrity": "sha512-umOSDSDrfHbTNPuNpC2NSnnA3LUrqpevPb4T9jRx4MagXNS0rs+gwiTcAvqCRmsD6utzsrzNt+ebm00SNWiC3Q==", + "dependencies": { + "acorn": "^8.1.0", + "acorn-walk": "^8.0.2" + } + }, + "node_modules/acorn-walk": { + "version": "8.2.0", + "resolved": "https://registry.npmjs.org/acorn-walk/-/acorn-walk-8.2.0.tgz", + "integrity": "sha512-k+iyHEuPgSw6SbuDpGQM+06HQUa04DZ3o+F6CSzXMvvI5KMvnaEqXe+YVe555R9nn6GPt404fos4wcgpw12SDA==", + "engines": { + "node": ">=0.4.0" + } + }, + "node_modules/agent-base": { + "version": "6.0.2", + "resolved": "https://registry.npmjs.org/agent-base/-/agent-base-6.0.2.tgz", + "integrity": "sha512-RZNwNclF7+MS/8bDg70amg32dyeZGZxiDuQmZxKLAlQjr3jGyLx+4Kkk58UO7D2QdgFIQCovuSuZESne6RG6XQ==", + "dependencies": { + "debug": "4" + }, + "engines": { + "node": ">= 6.0.0" + } + }, + "node_modules/ansi-regex": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-6.0.1.tgz", + "integrity": "sha512-n5M855fKb2SsfMIiFFoVrABHJC8QtHwVx+mHWP3QcEqBHYienj5dHSgjbxtC0WEZXYt4wcD6zrQElDPhFuZgfA==", + "engines": { + "node": ">=12" + }, + "funding": { + "url": "https://github.com/chalk/ansi-regex?sponsor=1" + } + }, + "node_modules/ansi-sequence-parser": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/ansi-sequence-parser/-/ansi-sequence-parser-1.1.0.tgz", + "integrity": "sha512-lEm8mt52to2fT8GhciPCGeCXACSz2UwIN4X2e2LJSnZ5uAbn2/dsYdOmUXq0AtWS5cpAupysIneExOgH0Vd2TQ==", + "peer": true + }, + "node_modules/ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dependencies": { + "color-convert": "^2.0.1" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/anymatch": { + "version": "3.1.3", + "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-3.1.3.tgz", + "integrity": "sha512-KMReFUr0B4t+D+OBkjR3KYqvocp2XaSzO55UcB6mgQMd3KbcE+mWTyvVV7D/zsdEbNnV6acZUutkiHQXvTr1Rw==", + "dependencies": { + "normalize-path": "^3.0.0", + "picomatch": "^2.0.4" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/argparse": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/argparse/-/argparse-2.0.1.tgz", + "integrity": "sha512-8+9WqebbFzpX9OR+Wa6O29asIogeRMzcGtAINdpMHHyAg10f05aSFVBbcEqGf/PXw1EjAZ+q2/bEBg3DvurK3Q==" + }, + "node_modules/array-iterate": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/array-iterate/-/array-iterate-2.0.1.tgz", + "integrity": "sha512-I1jXZMjAgCMmxT4qxXfPXa6SthSoE8h6gkSI9BGGNv8mP8G/v0blc+qFnZu6K42vTOiuME596QaLO0TP3Lk0xg==", + "funding": { + "type": "github", + "url": "https://github.com/sponsors/wooorm" + } + }, + "node_modules/async-mutex": { + "version": "0.4.0", + "resolved": "https://registry.npmjs.org/async-mutex/-/async-mutex-0.4.0.tgz", + "integrity": "sha512-eJFZ1YhRR8UN8eBLoNzcDPcy/jqjsg6I1AP+KvWQX80BqOSW1oJPJXDylPUEeMr2ZQvHgnQ//Lp6f3RQ1zI7HA==", + "dependencies": { + "tslib": "^2.4.0" + } + }, + "node_modules/asynckit": { + "version": "0.4.0", + "resolved": "https://registry.npmjs.org/asynckit/-/asynckit-0.4.0.tgz", + "integrity": "sha512-Oei9OH4tRh0YqU3GxhX79dM/mwVgvbZJaSNaRk+bshkj0S5cfHcgYakreBjrHwatXKbz+IoIdYLxrKim2MjW0Q==" + }, + "node_modules/bail": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/bail/-/bail-2.0.2.tgz", + "integrity": "sha512-0xO6mYd7JB2YesxDKplafRpsiOzPt9V02ddPCLbY1xYGPOX24NTyN50qnUxgCPcSoYMhKpAuBTjQoRZCAkUDRw==", + "funding": { + "type": "github", + "url": "https://github.com/sponsors/wooorm" + } + }, + "node_modules/balanced-match": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz", + "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==" + }, + "node_modules/binary-extensions": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-2.2.0.tgz", + "integrity": "sha512-jDctJ/IVQbZoJykoeHbhXpOlNBqGNcwXJKJog42E5HDPUwQTSdjCHdihjj0DlnheQ7blbT6dHOafNAiS8ooQKA==", + "engines": { + "node": ">=8" + } + }, + "node_modules/brace-expansion": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-2.0.1.tgz", + "integrity": "sha512-XnAIvQ8eM+kC6aULx6wuQiwVsnzsi9d3WxzV3FpWTGA19F621kwdbsAcFKXgKUHZWsy+mY6iL1sHTxWEFCytDA==", + "dependencies": { + "balanced-match": "^1.0.0" + } + }, + "node_modules/braces": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.2.tgz", + "integrity": "sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==", + "dependencies": { + "fill-range": "^7.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/buffer-from": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/buffer-from/-/buffer-from-1.1.2.tgz", + "integrity": "sha512-E+XQCRwSbaaiChtv6k6Dwgc+bx+Bs6vuKJHHl5kox/BaKbhiXzqQOwK4cO22yElGp2OCmjwVhT3HmxgyPGnJfQ==" + }, + "node_modules/bytes": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.0.0.tgz", + "integrity": "sha512-pMhOfFDPiv9t5jjIXkHosWmkSyQbvsgEVNkz0ERHbuLh2T/7j4Mqqpz523Fe8MVY89KC6Sh/QfS2sM+SjgFDcw==", + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/ccount": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/ccount/-/ccount-2.0.1.tgz", + "integrity": "sha512-eyrF0jiFpY+3drT6383f1qhkbGsLSifNAjA61IUjZjmLCWjItY6LB9ft9YhoDgwfmclB2zhu51Lc7+95b8NRAg==", + "funding": { + "type": "github", + "url": "https://github.com/sponsors/wooorm" + } + }, + "node_modules/chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dependencies": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/chalk?sponsor=1" + } + }, + "node_modules/character-entities": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/character-entities/-/character-entities-2.0.2.tgz", + "integrity": "sha512-shx7oQ0Awen/BRIdkjkvz54PnEEI/EjwXDSIZp86/KKdbafHh1Df/RYGBhn4hbe2+uKC9FnT5UCEdyPz3ai9hQ==", + "funding": { + "type": "github", + "url": "https://github.com/sponsors/wooorm" + } + }, + "node_modules/character-entities-html4": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/character-entities-html4/-/character-entities-html4-2.1.0.tgz", + "integrity": "sha512-1v7fgQRj6hnSwFpq1Eu0ynr/CDEw0rXo2B61qXrLNdHZmPKgb7fqS1a2JwF0rISo9q77jDI8VMEHoApn8qDoZA==", + "funding": { + "type": "github", + "url": "https://github.com/sponsors/wooorm" + } + }, + "node_modules/character-entities-legacy": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/character-entities-legacy/-/character-entities-legacy-3.0.0.tgz", + "integrity": "sha512-RpPp0asT/6ufRm//AJVwpViZbGM/MkjQFxJccQRHmISF/22NBtsHqAWmL+/pmkPWoIUJdWyeVleTl1wydHATVQ==", + "funding": { + "type": "github", + "url": "https://github.com/sponsors/wooorm" + } + }, + "node_modules/chokidar": { + "version": "3.5.3", + "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-3.5.3.tgz", + "integrity": "sha512-Dr3sfKRP6oTcjf2JmUmFJfeVMvXBdegxB0iVQ5eb2V10uFJUCAS8OByZdVAyVb8xXNz3GjjTgj9kLWsZTqE6kw==", + "funding": [ + { + "type": "individual", + "url": "https://paulmillr.com/funding/" + } + ], + "dependencies": { + "anymatch": "~3.1.2", + "braces": "~3.0.2", + "glob-parent": "~5.1.2", + "is-binary-path": "~2.1.0", + "is-glob": "~4.0.1", + "normalize-path": "~3.0.0", + "readdirp": "~3.6.0" + }, + "engines": { + "node": ">= 8.10.0" + }, + "optionalDependencies": { + "fsevents": "~2.3.2" + } + }, + "node_modules/cli-spinner": { + "version": "0.2.10", + "resolved": "https://registry.npmjs.org/cli-spinner/-/cli-spinner-0.2.10.tgz", + "integrity": "sha512-U0sSQ+JJvSLi1pAYuJykwiA8Dsr15uHEy85iCJ6A+0DjVxivr3d+N2Wjvodeg89uP5K6TswFkKBfAD7B3YSn/Q==", + "engines": { + "node": ">=0.10" + } + }, + "node_modules/cliui": { + "version": "8.0.1", + "resolved": "https://registry.npmjs.org/cliui/-/cliui-8.0.1.tgz", + "integrity": "sha512-BSeNnyus75C4//NQ9gQt1/csTXyo/8Sb+afLAkzAptFuMsod9HFokGNudZpi/oQV73hnVK+sR+5PVRMd+Dr7YQ==", + "dependencies": { + "string-width": "^4.2.0", + "strip-ansi": "^6.0.1", + "wrap-ansi": "^7.0.0" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/cliui/node_modules/ansi-regex": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz", + "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==", + "engines": { + "node": ">=8" + } + }, + "node_modules/cliui/node_modules/emoji-regex": { + "version": "8.0.0", + "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz", + "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==" + }, + "node_modules/cliui/node_modules/string-width": { + "version": "4.2.3", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.3.tgz", + "integrity": "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==", + "dependencies": { + "emoji-regex": "^8.0.0", + "is-fullwidth-code-point": "^3.0.0", + "strip-ansi": "^6.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/cliui/node_modules/strip-ansi": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", + "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", + "dependencies": { + "ansi-regex": "^5.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/cliui/node_modules/wrap-ansi": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-7.0.0.tgz", + "integrity": "sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q==", + "dependencies": { + "ansi-styles": "^4.0.0", + "string-width": "^4.1.0", + "strip-ansi": "^6.0.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/wrap-ansi?sponsor=1" + } + }, + "node_modules/color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dependencies": { + "color-name": "~1.1.4" + }, + "engines": { + "node": ">=7.0.0" + } + }, + "node_modules/color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==" + }, + "node_modules/combined-stream": { + "version": "1.0.8", + "resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.8.tgz", + "integrity": "sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==", + "dependencies": { + "delayed-stream": "~1.0.0" + }, + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/comma-separated-tokens": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/comma-separated-tokens/-/comma-separated-tokens-2.0.3.tgz", + "integrity": "sha512-Fu4hJdvzeylCfQPp9SGWidpzrMs7tTrlu6Vb8XGaRGck8QSNZJJp538Wrb60Lax4fPwR64ViY468OIUTbRlGZg==", + "funding": { + "type": "github", + "url": "https://github.com/sponsors/wooorm" + } + }, + "node_modules/commander": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/commander/-/commander-7.2.0.tgz", + "integrity": "sha512-QrWXB+ZQSVPmIWIhtEO9H+gwHaMGYiF5ChvoJ+K9ZGHG/sVsa6yiesAD1GC/x46sET00Xlwo1u49RVVVzvcSkw==", + "engines": { + "node": ">= 10" + } + }, + "node_modules/concat-map": { + "version": "0.0.1", + "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", + "integrity": "sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg==" + }, + "node_modules/content-disposition": { + "version": "0.5.2", + "resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-0.5.2.tgz", + "integrity": "sha512-kRGRZw3bLlFISDBgwTSA1TMBFN6J6GWDeubmDE3AF+3+yXL8hTWv8r5rkLbqYXY4RjPk/EzHnClI3zQf1cFmHA==", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/cross-spawn": { + "version": "7.0.3", + "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.3.tgz", + "integrity": "sha512-iRDPJKUPVEND7dHPO8rkbOnPpyDygcDFtWjpeWNCgy8WP2rXcxXL8TskReQl6OrB2G7+UJrags1q15Fudc7G6w==", + "dependencies": { + "path-key": "^3.1.0", + "shebang-command": "^2.0.0", + "which": "^2.0.1" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/cssom": { + "version": "0.5.0", + "resolved": "https://registry.npmjs.org/cssom/-/cssom-0.5.0.tgz", + "integrity": "sha512-iKuQcq+NdHqlAcwUY0o/HL69XQrUaQdMjmStJ8JFmUaiiQErlhrmuigkg/CU4E2J0IyUKUrMAgl36TvN67MqTw==" + }, + "node_modules/cssstyle": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/cssstyle/-/cssstyle-2.3.0.tgz", + "integrity": "sha512-AZL67abkUzIuvcHqk7c09cezpGNcxUxU4Ioi/05xHk4DQeTkWmGYftIE6ctU6AEt+Gn4n1lDStOtj7FKycP71A==", + "dependencies": { + "cssom": "~0.3.6" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/cssstyle/node_modules/cssom": { + "version": "0.3.8", + "resolved": "https://registry.npmjs.org/cssom/-/cssom-0.3.8.tgz", + "integrity": "sha512-b0tGHbfegbhPJpxpiBPU2sCkigAqtM9O121le6bbOlgyV+NyGyCmVfJ6QW9eRjz8CpNfWEOYBIMIGRYkLwsIYg==" + }, + "node_modules/d3": { + "version": "7.8.5", + "resolved": "https://registry.npmjs.org/d3/-/d3-7.8.5.tgz", + "integrity": "sha512-JgoahDG51ncUfJu6wX/1vWQEqOflgXyl4MaHqlcSruTez7yhaRKR9i8VjjcQGeS2en/jnFivXuaIMnseMMt0XA==", + "dependencies": { + "d3-array": "3", + "d3-axis": "3", + "d3-brush": "3", + "d3-chord": "3", + "d3-color": "3", + "d3-contour": "4", + "d3-delaunay": "6", + "d3-dispatch": "3", + "d3-drag": "3", + "d3-dsv": "3", + "d3-ease": "3", + "d3-fetch": "3", + "d3-force": "3", + "d3-format": "3", + "d3-geo": "3", + "d3-hierarchy": "3", + "d3-interpolate": "3", + "d3-path": "3", + "d3-polygon": "3", + "d3-quadtree": "3", + "d3-random": "3", + "d3-scale": "4", + "d3-scale-chromatic": "3", + "d3-selection": "3", + "d3-shape": "3", + "d3-time": "3", + "d3-time-format": "4", + "d3-timer": "3", + "d3-transition": "3", + "d3-zoom": "3" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/d3-array": { + "version": "3.2.4", + "resolved": "https://registry.npmjs.org/d3-array/-/d3-array-3.2.4.tgz", + "integrity": "sha512-tdQAmyA18i4J7wprpYq8ClcxZy3SC31QMeByyCFyRt7BVHdREQZ5lpzoe5mFEYZUWe+oq8HBvk9JjpibyEV4Jg==", + "dependencies": { + "internmap": "1 - 2" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/d3-axis": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/d3-axis/-/d3-axis-3.0.0.tgz", + "integrity": "sha512-IH5tgjV4jE/GhHkRV0HiVYPDtvfjHQlQfJHs0usq7M30XcSBvOotpmH1IgkcXsO/5gEQZD43B//fc7SRT5S+xw==", + "engines": { + "node": ">=12" + } + }, + "node_modules/d3-brush": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/d3-brush/-/d3-brush-3.0.0.tgz", + "integrity": "sha512-ALnjWlVYkXsVIGlOsuWH1+3udkYFI48Ljihfnh8FZPF2QS9o+PzGLBslO0PjzVoHLZ2KCVgAM8NVkXPJB2aNnQ==", + "dependencies": { + "d3-dispatch": "1 - 3", + "d3-drag": "2 - 3", + "d3-interpolate": "1 - 3", + "d3-selection": "3", + "d3-transition": "3" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/d3-chord": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/d3-chord/-/d3-chord-3.0.1.tgz", + "integrity": "sha512-VE5S6TNa+j8msksl7HwjxMHDM2yNK3XCkusIlpX5kwauBfXuyLAtNg9jCp/iHH61tgI4sb6R/EIMWCqEIdjT/g==", + "dependencies": { + "d3-path": "1 - 3" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/d3-color": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/d3-color/-/d3-color-3.1.0.tgz", + "integrity": "sha512-zg/chbXyeBtMQ1LbD/WSoW2DpC3I0mpmPdW+ynRTj/x2DAWYrIY7qeZIHidozwV24m4iavr15lNwIwLxRmOxhA==", + "engines": { + "node": ">=12" + } + }, + "node_modules/d3-contour": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/d3-contour/-/d3-contour-4.0.2.tgz", + "integrity": "sha512-4EzFTRIikzs47RGmdxbeUvLWtGedDUNkTcmzoeyg4sP/dvCexO47AaQL7VKy/gul85TOxw+IBgA8US2xwbToNA==", + "dependencies": { + "d3-array": "^3.2.0" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/d3-delaunay": { + "version": "6.0.4", + "resolved": "https://registry.npmjs.org/d3-delaunay/-/d3-delaunay-6.0.4.tgz", + "integrity": "sha512-mdjtIZ1XLAM8bm/hx3WwjfHt6Sggek7qH043O8KEjDXN40xi3vx/6pYSVTwLjEgiXQTbvaouWKynLBiUZ6SK6A==", + "dependencies": { + "delaunator": "5" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/d3-dispatch": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/d3-dispatch/-/d3-dispatch-3.0.1.tgz", + "integrity": "sha512-rzUyPU/S7rwUflMyLc1ETDeBj0NRuHKKAcvukozwhshr6g6c5d8zh4c2gQjY2bZ0dXeGLWc1PF174P2tVvKhfg==", + "engines": { + "node": ">=12" + } + }, + "node_modules/d3-drag": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/d3-drag/-/d3-drag-3.0.0.tgz", + "integrity": "sha512-pWbUJLdETVA8lQNJecMxoXfH6x+mO2UQo8rSmZ+QqxcbyA3hfeprFgIT//HW2nlHChWeIIMwS2Fq+gEARkhTkg==", + "dependencies": { + "d3-dispatch": "1 - 3", + "d3-selection": "3" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/d3-dsv": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/d3-dsv/-/d3-dsv-3.0.1.tgz", + "integrity": "sha512-UG6OvdI5afDIFP9w4G0mNq50dSOsXHJaRE8arAS5o9ApWnIElp8GZw1Dun8vP8OyHOZ/QJUKUJwxiiCCnUwm+Q==", + "dependencies": { + "commander": "7", + "iconv-lite": "0.6", + "rw": "1" + }, + "bin": { + "csv2json": "bin/dsv2json.js", + "csv2tsv": "bin/dsv2dsv.js", + "dsv2dsv": "bin/dsv2dsv.js", + "dsv2json": "bin/dsv2json.js", + "json2csv": "bin/json2dsv.js", + "json2dsv": "bin/json2dsv.js", + "json2tsv": "bin/json2dsv.js", + "tsv2csv": "bin/dsv2dsv.js", + "tsv2json": "bin/dsv2json.js" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/d3-ease": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/d3-ease/-/d3-ease-3.0.1.tgz", + "integrity": "sha512-wR/XK3D3XcLIZwpbvQwQ5fK+8Ykds1ip7A2Txe0yxncXSdq1L9skcG7blcedkOX+ZcgxGAmLX1FrRGbADwzi0w==", + "engines": { + "node": ">=12" + } + }, + "node_modules/d3-fetch": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/d3-fetch/-/d3-fetch-3.0.1.tgz", + "integrity": "sha512-kpkQIM20n3oLVBKGg6oHrUchHM3xODkTzjMoj7aWQFq5QEM+R6E4WkzT5+tojDY7yjez8KgCBRoj4aEr99Fdqw==", + "dependencies": { + "d3-dsv": "1 - 3" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/d3-force": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/d3-force/-/d3-force-3.0.0.tgz", + "integrity": "sha512-zxV/SsA+U4yte8051P4ECydjD/S+qeYtnaIyAs9tgHCqfguma/aAQDjo85A9Z6EKhBirHRJHXIgJUlffT4wdLg==", + "dependencies": { + "d3-dispatch": "1 - 3", + "d3-quadtree": "1 - 3", + "d3-timer": "1 - 3" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/d3-format": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/d3-format/-/d3-format-3.1.0.tgz", + "integrity": "sha512-YyUI6AEuY/Wpt8KWLgZHsIU86atmikuoOmCfommt0LYHiQSPjvX2AcFc38PX0CBpr2RCyZhjex+NS/LPOv6YqA==", + "engines": { + "node": ">=12" + } + }, + "node_modules/d3-geo": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/d3-geo/-/d3-geo-3.1.0.tgz", + "integrity": "sha512-JEo5HxXDdDYXCaWdwLRt79y7giK8SbhZJbFWXqbRTolCHFI5jRqteLzCsq51NKbUoX0PjBVSohxrx+NoOUujYA==", + "dependencies": { + "d3-array": "2.5.0 - 3" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/d3-hierarchy": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/d3-hierarchy/-/d3-hierarchy-3.1.2.tgz", + "integrity": "sha512-FX/9frcub54beBdugHjDCdikxThEqjnR93Qt7PvQTOHxyiNCAlvMrHhclk3cD5VeAaq9fxmfRp+CnWw9rEMBuA==", + "engines": { + "node": ">=12" + } + }, + "node_modules/d3-interpolate": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/d3-interpolate/-/d3-interpolate-3.0.1.tgz", + "integrity": "sha512-3bYs1rOD33uo8aqJfKP3JWPAibgw8Zm2+L9vBKEHJ2Rg+viTR7o5Mmv5mZcieN+FRYaAOWX5SJATX6k1PWz72g==", + "dependencies": { + "d3-color": "1 - 3" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/d3-path": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/d3-path/-/d3-path-3.1.0.tgz", + "integrity": "sha512-p3KP5HCf/bvjBSSKuXid6Zqijx7wIfNW+J/maPs+iwR35at5JCbLUT0LzF1cnjbCHWhqzQTIN2Jpe8pRebIEFQ==", + "engines": { + "node": ">=12" + } + }, + "node_modules/d3-polygon": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/d3-polygon/-/d3-polygon-3.0.1.tgz", + "integrity": "sha512-3vbA7vXYwfe1SYhED++fPUQlWSYTTGmFmQiany/gdbiWgU/iEyQzyymwL9SkJjFFuCS4902BSzewVGsHHmHtXg==", + "engines": { + "node": ">=12" + } + }, + "node_modules/d3-quadtree": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/d3-quadtree/-/d3-quadtree-3.0.1.tgz", + "integrity": "sha512-04xDrxQTDTCFwP5H6hRhsRcb9xxv2RzkcsygFzmkSIOJy3PeRJP7sNk3VRIbKXcog561P9oU0/rVH6vDROAgUw==", + "engines": { + "node": ">=12" + } + }, + "node_modules/d3-random": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/d3-random/-/d3-random-3.0.1.tgz", + "integrity": "sha512-FXMe9GfxTxqd5D6jFsQ+DJ8BJS4E/fT5mqqdjovykEB2oFbTMDVdg1MGFxfQW+FBOGoB++k8swBrgwSHT1cUXQ==", + "engines": { + "node": ">=12" + } + }, + "node_modules/d3-scale": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/d3-scale/-/d3-scale-4.0.2.tgz", + "integrity": "sha512-GZW464g1SH7ag3Y7hXjf8RoUuAFIqklOAq3MRl4OaWabTFJY9PN/E1YklhXLh+OQ3fM9yS2nOkCoS+WLZ6kvxQ==", + "dependencies": { + "d3-array": "2.10.0 - 3", + "d3-format": "1 - 3", + "d3-interpolate": "1.2.0 - 3", + "d3-time": "2.1.1 - 3", + "d3-time-format": "2 - 4" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/d3-scale-chromatic": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/d3-scale-chromatic/-/d3-scale-chromatic-3.0.0.tgz", + "integrity": "sha512-Lx9thtxAKrO2Pq6OO2Ua474opeziKr279P/TKZsMAhYyNDD3EnCffdbgeSYN5O7m2ByQsxtuP2CSDczNUIZ22g==", + "dependencies": { + "d3-color": "1 - 3", + "d3-interpolate": "1 - 3" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/d3-selection": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/d3-selection/-/d3-selection-3.0.0.tgz", + "integrity": "sha512-fmTRWbNMmsmWq6xJV8D19U/gw/bwrHfNXxrIN+HfZgnzqTHp9jOmKMhsTUjXOJnZOdZY9Q28y4yebKzqDKlxlQ==", + "engines": { + "node": ">=12" + } + }, + "node_modules/d3-shape": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/d3-shape/-/d3-shape-3.2.0.tgz", + "integrity": "sha512-SaLBuwGm3MOViRq2ABk3eLoxwZELpH6zhl3FbAoJ7Vm1gofKx6El1Ib5z23NUEhF9AsGl7y+dzLe5Cw2AArGTA==", + "dependencies": { + "d3-path": "^3.1.0" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/d3-time": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/d3-time/-/d3-time-3.1.0.tgz", + "integrity": "sha512-VqKjzBLejbSMT4IgbmVgDjpkYrNWUYJnbCGo874u7MMKIWsILRX+OpX/gTk8MqjpT1A/c6HY2dCA77ZN0lkQ2Q==", + "dependencies": { + "d3-array": "2 - 3" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/d3-time-format": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/d3-time-format/-/d3-time-format-4.1.0.tgz", + "integrity": "sha512-dJxPBlzC7NugB2PDLwo9Q8JiTR3M3e4/XANkreKSUxF8vvXKqm1Yfq4Q5dl8budlunRVlUUaDUgFt7eA8D6NLg==", + "dependencies": { + "d3-time": "1 - 3" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/d3-timer": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/d3-timer/-/d3-timer-3.0.1.tgz", + "integrity": "sha512-ndfJ/JxxMd3nw31uyKoY2naivF+r29V+Lc0svZxe1JvvIRmi8hUsrMvdOwgS1o6uBHmiz91geQ0ylPP0aj1VUA==", + "engines": { + "node": ">=12" + } + }, + "node_modules/d3-transition": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/d3-transition/-/d3-transition-3.0.1.tgz", + "integrity": "sha512-ApKvfjsSR6tg06xrL434C0WydLr7JewBB3V+/39RMHsaXTOG0zmt/OAXeng5M5LBm0ojmxJrpomQVZ1aPvBL4w==", + "dependencies": { + "d3-color": "1 - 3", + "d3-dispatch": "1 - 3", + "d3-ease": "1 - 3", + "d3-interpolate": "1 - 3", + "d3-timer": "1 - 3" + }, + "engines": { + "node": ">=12" + }, + "peerDependencies": { + "d3-selection": "2 - 3" + } + }, + "node_modules/d3-zoom": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/d3-zoom/-/d3-zoom-3.0.0.tgz", + "integrity": "sha512-b8AmV3kfQaqWAuacbPuNbL6vahnOJflOhexLzMMNLga62+/nh0JzvJ0aO/5a5MVgUFGS7Hu1P9P03o3fJkDCyw==", + "dependencies": { + "d3-dispatch": "1 - 3", + "d3-drag": "2 - 3", + "d3-interpolate": "1 - 3", + "d3-selection": "2 - 3", + "d3-transition": "2 - 3" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/data-urls": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/data-urls/-/data-urls-3.0.2.tgz", + "integrity": "sha512-Jy/tj3ldjZJo63sVAvg6LHt2mHvl4V6AgRAmNDtLdm7faqtsx+aJG42rsyCo9JCoRVKwPFzKlIPx3DIibwSIaQ==", + "dependencies": { + "abab": "^2.0.6", + "whatwg-mimetype": "^3.0.0", + "whatwg-url": "^11.0.0" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/debug": { + "version": "4.3.4", + "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.4.tgz", + "integrity": "sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ==", + "dependencies": { + "ms": "2.1.2" + }, + "engines": { + "node": ">=6.0" + }, + "peerDependenciesMeta": { + "supports-color": { + "optional": true + } + } + }, + "node_modules/decimal.js": { + "version": "10.4.3", + "resolved": "https://registry.npmjs.org/decimal.js/-/decimal.js-10.4.3.tgz", + "integrity": "sha512-VBBaLc1MgL5XpzgIP7ny5Z6Nx3UrRkIViUkPUdtl9aya5amy3De1gsUUSB1g3+3sExYNjCAsAznmukyxCb1GRA==" + }, + "node_modules/decode-named-character-reference": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/decode-named-character-reference/-/decode-named-character-reference-1.0.2.tgz", + "integrity": "sha512-O8x12RzrUF8xyVcY0KJowWsmaJxQbmy0/EtnNtHRpsOcT7dFk5W598coHqBVpmWo1oQQfsCqfCmkZN5DJrZVdg==", + "dependencies": { + "character-entities": "^2.0.0" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/wooorm" + } + }, + "node_modules/delaunator": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/delaunator/-/delaunator-5.0.0.tgz", + "integrity": "sha512-AyLvtyJdbv/U1GkiS6gUUzclRoAY4Gs75qkMygJJhU75LW4DNuSF2RMzpxs9jw9Oz1BobHjTdkG3zdP55VxAqw==", + "dependencies": { + "robust-predicates": "^3.0.0" + } + }, + "node_modules/delayed-stream": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz", + "integrity": "sha512-ZySD7Nf91aLB0RxL4KGrKHBXl7Eds1DAmEdcoVawXnLD7SDhpNgtuII2aAkg7a7QS41jxPSZ17p4VdGnMHk3MQ==", + "engines": { + "node": ">=0.4.0" + } + }, + "node_modules/dequal": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/dequal/-/dequal-2.0.3.tgz", + "integrity": "sha512-0je+qPKHEMohvfRTCEo3CrPG6cAzAYgmzKyxRiYSSDkS6eGJdyVJm7WaYA5ECaAD9wLB2T4EEeymA5aFVcYXCA==", + "engines": { + "node": ">=6" + } + }, + "node_modules/detect-libc": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/detect-libc/-/detect-libc-1.0.3.tgz", + "integrity": "sha512-pGjwhsmsp4kL2RTz08wcOlGN83otlqHeD/Z5T8GXZB+/YcpQ/dgo+lbU8ZsGxV0HIvqqxo9l7mqYwyYMD9bKDg==", + "bin": { + "detect-libc": "bin/detect-libc.js" + }, + "engines": { + "node": ">=0.10" + } + }, + "node_modules/diff": { + "version": "5.1.0", + "resolved": "https://registry.npmjs.org/diff/-/diff-5.1.0.tgz", + "integrity": "sha512-D+mk+qE8VC/PAUrlAU34N+VfXev0ghe5ywmpqrawphmVZc1bEfn56uo9qpyGp1p4xpzOHkSW4ztBd6L7Xx4ACw==", + "engines": { + "node": ">=0.3.1" + } + }, + "node_modules/dir-glob": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/dir-glob/-/dir-glob-3.0.1.tgz", + "integrity": "sha512-WkrWp9GR4KXfKGYzOLmTuGVi1UWFfws377n9cc55/tb6DuqyF6pcQ5AbiHEshaDpY9v6oaSr2XCDidGmMwdzIA==", + "dependencies": { + "path-type": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/domexception": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/domexception/-/domexception-4.0.0.tgz", + "integrity": "sha512-A2is4PLG+eeSfoTMA95/s4pvAoSo2mKtiM5jlHkAVewmiO8ISFTFKZjH7UAM1Atli/OT/7JHOrJRJiMKUZKYBw==", + "dependencies": { + "webidl-conversions": "^7.0.0" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/eastasianwidth": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/eastasianwidth/-/eastasianwidth-0.2.0.tgz", + "integrity": "sha512-I88TYZWc9XiYHRQ4/3c5rjjfgkjhLyW2luGIheGERbNQ6OY7yTybanSpDXZa8y7VUP9YmDcYa+eyq4ca7iLqWA==" + }, + "node_modules/emoji-regex": { + "version": "9.2.2", + "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-9.2.2.tgz", + "integrity": "sha512-L18DaJsXSUk2+42pv8mLs5jJT2hqFkFE4j21wOmgbUqsZ2hL72NsUU785g9RXgo3s0ZNgVl42TiHp3ZtOv/Vyg==" + }, + "node_modules/entities": { + "version": "4.5.0", + "resolved": "https://registry.npmjs.org/entities/-/entities-4.5.0.tgz", + "integrity": "sha512-V0hjH4dGPh9Ao5p0MoRY6BVqtwCjhz6vI5LT8AJ55H+4g9/4vbHx1I54fS0XuclLhDHArPQCiMjDxjaL8fPxhw==", + "engines": { + "node": ">=0.12" + }, + "funding": { + "url": "https://github.com/fb55/entities?sponsor=1" + } + }, + "node_modules/esbuild": { + "version": "0.19.2", + "resolved": "https://registry.npmjs.org/esbuild/-/esbuild-0.19.2.tgz", + "integrity": "sha512-G6hPax8UbFakEj3hWO0Vs52LQ8k3lnBhxZWomUJDxfz3rZTLqF5k/FCzuNdLx2RbpBiQQF9H9onlDDH1lZsnjg==", + "hasInstallScript": true, + "bin": { + "esbuild": "bin/esbuild" + }, + "engines": { + "node": ">=12" + }, + "optionalDependencies": { + "@esbuild/android-arm": "0.19.2", + "@esbuild/android-arm64": "0.19.2", + "@esbuild/android-x64": "0.19.2", + "@esbuild/darwin-arm64": "0.19.2", + "@esbuild/darwin-x64": "0.19.2", + "@esbuild/freebsd-arm64": "0.19.2", + "@esbuild/freebsd-x64": "0.19.2", + "@esbuild/linux-arm": "0.19.2", + "@esbuild/linux-arm64": "0.19.2", + "@esbuild/linux-ia32": "0.19.2", + "@esbuild/linux-loong64": "0.19.2", + "@esbuild/linux-mips64el": "0.19.2", + "@esbuild/linux-ppc64": "0.19.2", + "@esbuild/linux-riscv64": "0.19.2", + "@esbuild/linux-s390x": "0.19.2", + "@esbuild/linux-x64": "0.19.2", + "@esbuild/netbsd-x64": "0.19.2", + "@esbuild/openbsd-x64": "0.19.2", + "@esbuild/sunos-x64": "0.19.2", + "@esbuild/win32-arm64": "0.19.2", + "@esbuild/win32-ia32": "0.19.2", + "@esbuild/win32-x64": "0.19.2" + } + }, + "node_modules/esbuild-sass-plugin": { + "version": "2.12.0", + "resolved": "https://registry.npmjs.org/esbuild-sass-plugin/-/esbuild-sass-plugin-2.12.0.tgz", + "integrity": "sha512-+k/5WM/Yf/Ur7ahn6XXxEPwa/lmuacLO7vrCIAJuvQapX1CiIHtlX/nc2eiMoJ6P6RvqZhKpQvIiwgYJonzHtw==", + "dependencies": { + "resolve": "^1.22.2", + "sass": "^1.65.1" + }, + "peerDependencies": { + "esbuild": "^0.19.1" + } + }, + "node_modules/escalade": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/escalade/-/escalade-3.1.1.tgz", + "integrity": "sha512-k0er2gUkLf8O0zKJiAhmkTnJlTvINGv7ygDNPbeIsX/TJjGJZHuh9B2UxbsaEkmlEo9MfhrSzmhIlhRlI2GXnw==", + "engines": { + "node": ">=6" + } + }, + "node_modules/escape-string-regexp": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-5.0.0.tgz", + "integrity": "sha512-/veY75JbMK4j1yjvuUxuVsiS/hr/4iHs9FTT6cgTexxdE0Ly/glccBAkloH/DofkjRbZU3bnoj38mOmhkZ0lHw==", + "engines": { + "node": ">=12" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/escodegen": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/escodegen/-/escodegen-2.1.0.tgz", + "integrity": "sha512-2NlIDTwUWJN0mRPQOdtQBzbUHvdGY2P1VXSyU83Q3xKxM7WHX2Ql8dKq782Q9TgQUNOLEzEYu9bzLNj1q88I5w==", + "dependencies": { + "esprima": "^4.0.1", + "estraverse": "^5.2.0", + "esutils": "^2.0.2" + }, + "bin": { + "escodegen": "bin/escodegen.js", + "esgenerate": "bin/esgenerate.js" + }, + "engines": { + "node": ">=6.0" + }, + "optionalDependencies": { + "source-map": "~0.6.1" + } + }, + "node_modules/esm": { + "version": "3.2.25", + "resolved": "https://registry.npmjs.org/esm/-/esm-3.2.25.tgz", + "integrity": "sha512-U1suiZ2oDVWv4zPO56S0NcR5QriEahGtdN2OR6FiOG4WJvcjBVFB0qI4+eKoWFH483PKGuLuu6V8Z4T5g63UVA==", + "engines": { + "node": ">=6" + } + }, + "node_modules/esprima": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/esprima/-/esprima-4.0.1.tgz", + "integrity": "sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A==", + "bin": { + "esparse": "bin/esparse.js", + "esvalidate": "bin/esvalidate.js" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/estraverse": { + "version": "5.3.0", + "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-5.3.0.tgz", + "integrity": "sha512-MMdARuVEQziNTeJD8DgMqmhwR11BRQ/cBP+pLtYdSTnf3MIO8fFeiINEbX36ZdNlfU/7A9f3gUw49B3oQsvwBA==", + "engines": { + "node": ">=4.0" + } + }, + "node_modules/esutils": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/esutils/-/esutils-2.0.3.tgz", + "integrity": "sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/extend": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/extend/-/extend-3.0.2.tgz", + "integrity": "sha512-fjquC59cD7CyW6urNXK0FBufkZcoiGG80wTuPujX590cB5Ttln20E2UB4S/WARVqhXffZl2LNgS+gQdPIIim/g==" + }, + "node_modules/extend-shallow": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", + "integrity": "sha512-zCnTtlxNoAiDc3gqY2aYAWFx7XWWiasuF2K8Me5WbN8otHKTUKBwjPtNpRs/rbUZm7KxWAaNj7P1a/p52GbVug==", + "dependencies": { + "is-extendable": "^0.1.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/fast-glob": { + "version": "3.3.0", + "resolved": "https://registry.npmjs.org/fast-glob/-/fast-glob-3.3.0.tgz", + "integrity": "sha512-ChDuvbOypPuNjO8yIDf36x7BlZX1smcUMTTcyoIjycexOxd6DFsKsg21qVBzEmr3G7fUKIRy2/psii+CIUt7FA==", + "dependencies": { + "@nodelib/fs.stat": "^2.0.2", + "@nodelib/fs.walk": "^1.2.3", + "glob-parent": "^5.1.2", + "merge2": "^1.3.0", + "micromatch": "^4.0.4" + }, + "engines": { + "node": ">=8.6.0" + } + }, + "node_modules/fast-url-parser": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/fast-url-parser/-/fast-url-parser-1.1.3.tgz", + "integrity": "sha512-5jOCVXADYNuRkKFzNJ0dCCewsZiYo0dz8QNYljkOpFC6r2U4OBmKtvm/Tsuh4w1YYdDqDb31a8TVhBJ2OJKdqQ==", + "dependencies": { + "punycode": "^1.3.2" + } + }, + "node_modules/fastq": { + "version": "1.15.0", + "resolved": "https://registry.npmjs.org/fastq/-/fastq-1.15.0.tgz", + "integrity": "sha512-wBrocU2LCXXa+lWBt8RoIRD89Fi8OdABODa/kEnyeyjS5aZO5/GNvI5sEINADqP/h8M29UHTHUb53sUu5Ihqdw==", + "dependencies": { + "reusify": "^1.0.4" + } + }, + "node_modules/fault": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/fault/-/fault-2.0.1.tgz", + "integrity": "sha512-WtySTkS4OKev5JtpHXnib4Gxiurzh5NCGvWrFaZ34m6JehfTUhKZvn9njTfw48t6JumVQOmrKqpmGcdwxnhqBQ==", + "dependencies": { + "format": "^0.2.0" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/wooorm" + } + }, + "node_modules/fill-range": { + "version": "7.0.1", + "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.0.1.tgz", + "integrity": "sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ==", + "dependencies": { + "to-regex-range": "^5.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/flexsearch": { + "version": "0.7.21", + "resolved": "https://registry.npmjs.org/flexsearch/-/flexsearch-0.7.21.tgz", + "integrity": "sha512-W7cHV7Hrwjid6lWmy0IhsWDFQboWSng25U3VVywpHOTJnnAZNPScog67G+cVpeX9f7yDD21ih0WDrMMT+JoaYg==" + }, + "node_modules/foreground-child": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/foreground-child/-/foreground-child-3.1.1.tgz", + "integrity": "sha512-TMKDUnIte6bfb5nWv7V/caI169OHgvwjb7V4WkeUvbQQdjr5rWKqHFiKWb/fcOwB+CzBT+qbWjvj+DVwRskpIg==", + "dependencies": { + "cross-spawn": "^7.0.0", + "signal-exit": "^4.0.1" + }, + "engines": { + "node": ">=14" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/form-data": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/form-data/-/form-data-4.0.0.tgz", + "integrity": "sha512-ETEklSGi5t0QMZuiXoA/Q6vcnxcLQP5vdugSpuAyi6SVGi2clPPp+xgEhuMaHC+zGgn31Kd235W35f7Hykkaww==", + "dependencies": { + "asynckit": "^0.4.0", + "combined-stream": "^1.0.8", + "mime-types": "^2.1.12" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/format": { + "version": "0.2.2", + "resolved": "https://registry.npmjs.org/format/-/format-0.2.2.tgz", + "integrity": "sha512-wzsgA6WOq+09wrU1tsJ09udeR/YZRaeArL9e1wPbFg3GG2yDnC2ldKpxs4xunpFF9DgqCqOIra3bc1HWrJ37Ww==", + "engines": { + "node": ">=0.4.x" + } + }, + "node_modules/fsevents": { + "version": "2.3.2", + "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.2.tgz", + "integrity": "sha512-xiqMQR4xAeHTuB9uWm+fFRcIOgKBMiOBP+eXiyT7jsgVCq1bkVygt00oASowB7EdtpOHaaPgKt812P9ab+DDKA==", + "hasInstallScript": true, + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": "^8.16.0 || ^10.6.0 || >=11.0.0" + } + }, + "node_modules/function-bind": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.1.tgz", + "integrity": "sha512-yIovAzMX49sF8Yl58fSCWJ5svSLuaibPxXQJFLmBObTuCr0Mf1KiPopGM9NiFjiYBCbfaa2Fh6breQ6ANVTI0A==" + }, + "node_modules/get-caller-file": { + "version": "2.0.5", + "resolved": "https://registry.npmjs.org/get-caller-file/-/get-caller-file-2.0.5.tgz", + "integrity": "sha512-DyFP3BM/3YHTQOCUL/w0OZHR0lpKeGrxotcHWcqNEdnltqFwXVfhEBQ94eIo34AfQpo0rGki4cyIiftY06h2Fg==", + "engines": { + "node": "6.* || 8.* || >= 10.*" + } + }, + "node_modules/get-tsconfig": { + "version": "4.6.2", + "resolved": "https://registry.npmjs.org/get-tsconfig/-/get-tsconfig-4.6.2.tgz", + "integrity": "sha512-E5XrT4CbbXcXWy+1jChlZmrmCwd5KGx502kDCXJJ7y898TtWW9FwoG5HfOLVRKmlmDGkWN2HM9Ho+/Y8F0sJDg==", + "dev": true, + "dependencies": { + "resolve-pkg-maps": "^1.0.0" + }, + "funding": { + "url": "https://github.com/privatenumber/get-tsconfig?sponsor=1" + } + }, + "node_modules/github-slugger": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/github-slugger/-/github-slugger-2.0.0.tgz", + "integrity": "sha512-IaOQ9puYtjrkq7Y0Ygl9KDZnrf/aiUJYUpVf89y8kyaxbRG7Y1SrX/jaumrv81vc61+kiMempujsM3Yw7w5qcw==" + }, + "node_modules/glob": { + "version": "10.3.1", + "resolved": "https://registry.npmjs.org/glob/-/glob-10.3.1.tgz", + "integrity": "sha512-9BKYcEeIs7QwlCYs+Y3GBvqAMISufUS0i2ELd11zpZjxI5V9iyRj0HgzB5/cLf2NY4vcYBTYzJ7GIui7j/4DOw==", + "dependencies": { + "foreground-child": "^3.1.0", + "jackspeak": "^2.0.3", + "minimatch": "^9.0.1", + "minipass": "^5.0.0 || ^6.0.2", + "path-scurry": "^1.10.0" + }, + "bin": { + "glob": "dist/cjs/src/bin.js" + }, + "engines": { + "node": ">=16 || 14 >=14.17" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/glob-parent": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.2.tgz", + "integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==", + "dependencies": { + "is-glob": "^4.0.1" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/globby": { + "version": "13.2.1", + "resolved": "https://registry.npmjs.org/globby/-/globby-13.2.1.tgz", + "integrity": "sha512-DPCBxctI7dN4EeIqjW2KGqgdcUMbrhJ9AzON+PlxCtvppWhubTLD4+a0GFxiym14ZvacUydTPjLPc2DlKz7EIg==", + "dependencies": { + "dir-glob": "^3.0.1", + "fast-glob": "^3.2.11", + "ignore": "^5.2.0", + "merge2": "^1.4.1", + "slash": "^4.0.0" + }, + "engines": { + "node": "^12.20.0 || ^14.13.1 || >=16.0.0" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/gray-matter": { + "version": "4.0.3", + "resolved": "https://registry.npmjs.org/gray-matter/-/gray-matter-4.0.3.tgz", + "integrity": "sha512-5v6yZd4JK3eMI3FqqCouswVqwugaA9r4dNZB1wwcmrD02QkV5H0y7XBQW8QwQqEaZY1pM9aqORSORhJRdNK44Q==", + "dependencies": { + "js-yaml": "^3.13.1", + "kind-of": "^6.0.2", + "section-matter": "^1.0.0", + "strip-bom-string": "^1.0.0" + }, + "engines": { + "node": ">=6.0" + } + }, + "node_modules/gray-matter/node_modules/argparse": { + "version": "1.0.10", + "resolved": "https://registry.npmjs.org/argparse/-/argparse-1.0.10.tgz", + "integrity": "sha512-o5Roy6tNG4SL/FOkCAN6RzjiakZS25RLYFrcMttJqbdd8BWrnA+fGz57iN5Pb06pvBGvl5gQ0B48dJlslXvoTg==", + "dependencies": { + "sprintf-js": "~1.0.2" + } + }, + "node_modules/gray-matter/node_modules/js-yaml": { + "version": "3.14.1", + "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-3.14.1.tgz", + "integrity": "sha512-okMH7OXXJ7YrN9Ok3/SXrnu4iX9yOk+25nqX4imS2npuvTYDmo/QEZoqwZkYaIDk3jVvBOTOIEgEhaLOynBS9g==", + "dependencies": { + "argparse": "^1.0.7", + "esprima": "^4.0.0" + }, + "bin": { + "js-yaml": "bin/js-yaml.js" + } + }, + "node_modules/has": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/has/-/has-1.0.3.tgz", + "integrity": "sha512-f2dvO0VU6Oej7RkWJGrehjbzMAjFp5/VKPp5tTpWIV4JHHZK1/BxbFRtf/siA2SWTe09caDmVtYYzWEIbBS4zw==", + "dependencies": { + "function-bind": "^1.1.1" + }, + "engines": { + "node": ">= 0.4.0" + } + }, + "node_modules/has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "engines": { + "node": ">=8" + } + }, + "node_modules/hash-obj": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/hash-obj/-/hash-obj-4.0.0.tgz", + "integrity": "sha512-FwO1BUVWkyHasWDW4S8o0ssQXjvyghLV2rfVhnN36b2bbcj45eGiuzdn9XOvOpjV3TKQD7Gm2BWNXdE9V4KKYg==", + "dependencies": { + "is-obj": "^3.0.0", + "sort-keys": "^5.0.0", + "type-fest": "^1.0.2" + }, + "engines": { + "node": ">=12" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/hast-util-from-dom": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/hast-util-from-dom/-/hast-util-from-dom-4.2.0.tgz", + "integrity": "sha512-t1RJW/OpJbCAJQeKi3Qrj1cAOLA0+av/iPFori112+0X7R3wng+jxLA+kXec8K4szqPRGI8vPxbbpEYvvpwaeQ==", + "dependencies": { + "hastscript": "^7.0.0", + "web-namespaces": "^2.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/hast-util-from-html": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/hast-util-from-html/-/hast-util-from-html-1.0.2.tgz", + "integrity": "sha512-LhrTA2gfCbLOGJq2u/asp4kwuG0y6NhWTXiPKP+n0qNukKy7hc10whqqCFfyvIA1Q5U5d0sp9HhNim9gglEH4A==", + "dependencies": { + "@types/hast": "^2.0.0", + "hast-util-from-parse5": "^7.0.0", + "parse5": "^7.0.0", + "vfile": "^5.0.0", + "vfile-message": "^3.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/hast-util-from-html-isomorphic": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/hast-util-from-html-isomorphic/-/hast-util-from-html-isomorphic-1.0.0.tgz", + "integrity": "sha512-Yu480AKeOEN/+l5LA674a+7BmIvtDj24GvOt7MtQWuhzUwlaaRWdEPXAh3Qm5vhuthpAipFb2vTetKXWOjmTvw==", + "dependencies": { + "@types/hast": "^2.0.0", + "hast-util-from-dom": "^4.0.0", + "hast-util-from-html": "^1.0.0", + "unist-util-remove-position": "^4.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/hast-util-from-parse5": { + "version": "7.1.2", + "resolved": "https://registry.npmjs.org/hast-util-from-parse5/-/hast-util-from-parse5-7.1.2.tgz", + "integrity": "sha512-Nz7FfPBuljzsN3tCQ4kCBKqdNhQE2l0Tn+X1ubgKBPRoiDIu1mL08Cfw4k7q71+Duyaw7DXDN+VTAp4Vh3oCOw==", + "dependencies": { + "@types/hast": "^2.0.0", + "@types/unist": "^2.0.0", + "hastscript": "^7.0.0", + "property-information": "^6.0.0", + "vfile": "^5.0.0", + "vfile-location": "^4.0.0", + "web-namespaces": "^2.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/hast-util-has-property": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/hast-util-has-property/-/hast-util-has-property-2.0.1.tgz", + "integrity": "sha512-X2+RwZIMTMKpXUzlotatPzWj8bspCymtXH3cfG3iQKV+wPF53Vgaqxi/eLqGck0wKq1kS9nvoB1wchbCPEL8sg==", + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/hast-util-heading-rank": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/hast-util-heading-rank/-/hast-util-heading-rank-2.1.1.tgz", + "integrity": "sha512-iAuRp+ESgJoRFJbSyaqsfvJDY6zzmFoEnL1gtz1+U8gKtGGj1p0CVlysuUAUjq95qlZESHINLThwJzNGmgGZxA==", + "dependencies": { + "@types/hast": "^2.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/hast-util-is-element": { + "version": "2.1.3", + "resolved": "https://registry.npmjs.org/hast-util-is-element/-/hast-util-is-element-2.1.3.tgz", + "integrity": "sha512-O1bKah6mhgEq2WtVMk+Ta5K7pPMqsBBlmzysLdcwKVrqzZQ0CHqUPiIVspNhAG1rvxpvJjtGee17XfauZYKqVA==", + "dependencies": { + "@types/hast": "^2.0.0", + "@types/unist": "^2.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/hast-util-parse-selector": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/hast-util-parse-selector/-/hast-util-parse-selector-3.1.1.tgz", + "integrity": "sha512-jdlwBjEexy1oGz0aJ2f4GKMaVKkA9jwjr4MjAAI22E5fM/TXVZHuS5OpONtdeIkRKqAaryQ2E9xNQxijoThSZA==", + "dependencies": { + "@types/hast": "^2.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/hast-util-raw": { + "version": "7.2.3", + "resolved": "https://registry.npmjs.org/hast-util-raw/-/hast-util-raw-7.2.3.tgz", + "integrity": "sha512-RujVQfVsOrxzPOPSzZFiwofMArbQke6DJjnFfceiEbFh7S05CbPt0cYN+A5YeD3pso0JQk6O1aHBnx9+Pm2uqg==", + "dependencies": { + "@types/hast": "^2.0.0", + "@types/parse5": "^6.0.0", + "hast-util-from-parse5": "^7.0.0", + "hast-util-to-parse5": "^7.0.0", + "html-void-elements": "^2.0.0", + "parse5": "^6.0.0", + "unist-util-position": "^4.0.0", + "unist-util-visit": "^4.0.0", + "vfile": "^5.0.0", + "web-namespaces": "^2.0.0", + "zwitch": "^2.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/hast-util-raw/node_modules/parse5": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/parse5/-/parse5-6.0.1.tgz", + "integrity": "sha512-Ofn/CTFzRGTTxwpNEs9PP93gXShHcTq255nzRYSKe8AkVpZY7e1fpmTfOyoIvjP5HG7Z2ZM7VS9PPhQGW2pOpw==" + }, + "node_modules/hast-util-to-html": { + "version": "8.0.4", + "resolved": "https://registry.npmjs.org/hast-util-to-html/-/hast-util-to-html-8.0.4.tgz", + "integrity": "sha512-4tpQTUOr9BMjtYyNlt0P50mH7xj0Ks2xpo8M943Vykljf99HW6EzulIoJP1N3eKOSScEHzyzi9dm7/cn0RfGwA==", + "dependencies": { + "@types/hast": "^2.0.0", + "@types/unist": "^2.0.0", + "ccount": "^2.0.0", + "comma-separated-tokens": "^2.0.0", + "hast-util-raw": "^7.0.0", + "hast-util-whitespace": "^2.0.0", + "html-void-elements": "^2.0.0", + "property-information": "^6.0.0", + "space-separated-tokens": "^2.0.0", + "stringify-entities": "^4.0.0", + "zwitch": "^2.0.4" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/hast-util-to-jsx-runtime": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/hast-util-to-jsx-runtime/-/hast-util-to-jsx-runtime-1.2.0.tgz", + "integrity": "sha512-Y4FB8Dx2k6zJZrwbexkVm6YVRA8Sho2tTwacjDSr/x5c0wioOpc1VIoLyGUSb8+8xkAnQPAtHbdMvzA6bl0F1w==", + "dependencies": { + "@types/hast": "^2.0.0", + "@types/unist": "^2.0.0", + "comma-separated-tokens": "^2.0.0", + "hast-util-whitespace": "^2.0.0", + "property-information": "^6.0.0", + "space-separated-tokens": "^2.0.0", + "style-to-object": "^0.4.1", + "unist-util-position": "^4.0.0", + "vfile-message": "^3.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/hast-util-to-parse5": { + "version": "7.1.0", + "resolved": "https://registry.npmjs.org/hast-util-to-parse5/-/hast-util-to-parse5-7.1.0.tgz", + "integrity": "sha512-YNRgAJkH2Jky5ySkIqFXTQiaqcAtJyVE+D5lkN6CdtOqrnkLfGYYrEcKuHOJZlp+MwjSwuD3fZuawI+sic/RBw==", + "dependencies": { + "@types/hast": "^2.0.0", + "comma-separated-tokens": "^2.0.0", + "property-information": "^6.0.0", + "space-separated-tokens": "^2.0.0", + "web-namespaces": "^2.0.0", + "zwitch": "^2.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/hast-util-to-string": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/hast-util-to-string/-/hast-util-to-string-2.0.0.tgz", + "integrity": "sha512-02AQ3vLhuH3FisaMM+i/9sm4OXGSq1UhOOCpTLLQtHdL3tZt7qil69r8M8iDkZYyC0HCFylcYoP+8IO7ddta1A==", + "dependencies": { + "@types/hast": "^2.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/hast-util-to-text": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/hast-util-to-text/-/hast-util-to-text-3.1.2.tgz", + "integrity": "sha512-tcllLfp23dJJ+ju5wCCZHVpzsQQ43+moJbqVX3jNWPB7z/KFC4FyZD6R7y94cHL6MQ33YtMZL8Z0aIXXI4XFTw==", + "dependencies": { + "@types/hast": "^2.0.0", + "@types/unist": "^2.0.0", + "hast-util-is-element": "^2.0.0", + "unist-util-find-after": "^4.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/hast-util-whitespace": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/hast-util-whitespace/-/hast-util-whitespace-2.0.1.tgz", + "integrity": "sha512-nAxA0v8+vXSBDt3AnRUNjyRIQ0rD+ntpbAp4LnPkumc5M9yUbSMa4XDU9Q6etY4f1Wp4bNgvc1yjiZtsTTrSng==", + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/hastscript": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/hastscript/-/hastscript-7.2.0.tgz", + "integrity": "sha512-TtYPq24IldU8iKoJQqvZOuhi5CyCQRAbvDOX0x1eW6rsHSxa/1i2CCiptNTotGHJ3VoHRGmqiv6/D3q113ikkw==", + "dependencies": { + "@types/hast": "^2.0.0", + "comma-separated-tokens": "^2.0.0", + "hast-util-parse-selector": "^3.0.0", + "property-information": "^6.0.0", + "space-separated-tokens": "^2.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/html-encoding-sniffer": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/html-encoding-sniffer/-/html-encoding-sniffer-3.0.0.tgz", + "integrity": "sha512-oWv4T4yJ52iKrufjnyZPkrN0CH3QnrUqdB6In1g5Fe1mia8GmF36gnfNySxoZtxD5+NmYw1EElVXiBk93UeskA==", + "dependencies": { + "whatwg-encoding": "^2.0.0" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/html-void-elements": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/html-void-elements/-/html-void-elements-2.0.1.tgz", + "integrity": "sha512-0quDb7s97CfemeJAnW9wC0hw78MtW7NU3hqtCD75g2vFlDLt36llsYD7uB7SUzojLMP24N5IatXf7ylGXiGG9A==", + "funding": { + "type": "github", + "url": "https://github.com/sponsors/wooorm" + } + }, + "node_modules/http-proxy-agent": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/http-proxy-agent/-/http-proxy-agent-5.0.0.tgz", + "integrity": "sha512-n2hY8YdoRE1i7r6M0w9DIw5GgZN0G25P8zLCRQ8rjXtTU3vsNFBI/vWK/UIeE6g5MUUz6avwAPXmL6Fy9D/90w==", + "dependencies": { + "@tootallnate/once": "2", + "agent-base": "6", + "debug": "4" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/https-proxy-agent": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/https-proxy-agent/-/https-proxy-agent-5.0.1.tgz", + "integrity": "sha512-dFcAjpTQFgoLMzC2VwU+C/CbS7uRL0lWmxDITmqm7C+7F0Odmj6s9l6alZc6AELXhrnggM2CeWSXHGOdX2YtwA==", + "dependencies": { + "agent-base": "6", + "debug": "4" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/iconv-lite": { + "version": "0.6.3", + "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.6.3.tgz", + "integrity": "sha512-4fCk79wshMdzMp2rH06qWrJE4iolqLhCUH+OiuIgU++RB0+94NlDL81atO7GX55uUKueo0txHNtvEyI6D7WdMw==", + "dependencies": { + "safer-buffer": ">= 2.1.2 < 3.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/ignore": { + "version": "5.2.4", + "resolved": "https://registry.npmjs.org/ignore/-/ignore-5.2.4.tgz", + "integrity": "sha512-MAb38BcSbH0eHNBxn7ql2NH/kX33OkB3lZ1BNdh7ENeRChHTYsTvWrMubiIAMNS2llXEEgZ1MUOBtXChP3kaFQ==", + "engines": { + "node": ">= 4" + } + }, + "node_modules/immutable": { + "version": "4.3.3", + "resolved": "https://registry.npmjs.org/immutable/-/immutable-4.3.3.tgz", + "integrity": "sha512-808ZFYMsIRAjLAu5xkKo0TsbY9LBy9H5MazTKIEHerNkg0ymgilGfBPMR/3G7d/ihGmuK2Hw8S1izY2d3kd3wA==" + }, + "node_modules/inline-style-parser": { + "version": "0.1.1", + "resolved": "https://registry.npmjs.org/inline-style-parser/-/inline-style-parser-0.1.1.tgz", + "integrity": "sha512-7NXolsK4CAS5+xvdj5OMMbI962hU/wvwoxk+LWR9Ek9bVtyuuYScDN6eS0rUm6TxApFpw7CX1o4uJzcd4AyD3Q==" + }, + "node_modules/internmap": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/internmap/-/internmap-2.0.3.tgz", + "integrity": "sha512-5Hh7Y1wQbvY5ooGgPbDaL5iYLAPzMTUrjMulskHLH6wnv/A+1q5rgEaiuqEjB+oxGXIVZs1FF+R/KPN3ZSQYYg==", + "engines": { + "node": ">=12" + } + }, + "node_modules/is-absolute-url": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/is-absolute-url/-/is-absolute-url-4.0.1.tgz", + "integrity": "sha512-/51/TKE88Lmm7Gc4/8btclNXWS+g50wXhYJq8HWIBAGUBnoAdRu1aXeh364t/O7wXDAcTJDP8PNuNKWUDWie+A==", + "engines": { + "node": "^12.20.0 || ^14.13.1 || >=16.0.0" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/is-binary-path": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/is-binary-path/-/is-binary-path-2.1.0.tgz", + "integrity": "sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==", + "dependencies": { + "binary-extensions": "^2.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/is-buffer": { + "version": "2.0.5", + "resolved": "https://registry.npmjs.org/is-buffer/-/is-buffer-2.0.5.tgz", + "integrity": "sha512-i2R6zNFDwgEHJyQUtJEk0XFi1i0dPFn/oqjK3/vPCcDeJvW5NQ83V8QbicfF1SupOaB0h8ntgBC2YiE7dfyctQ==", + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/feross" + }, + { + "type": "patreon", + "url": "https://www.patreon.com/feross" + }, + { + "type": "consulting", + "url": "https://feross.org/support" + } + ], + "engines": { + "node": ">=4" + } + }, + "node_modules/is-core-module": { + "version": "2.12.1", + "resolved": "https://registry.npmjs.org/is-core-module/-/is-core-module-2.12.1.tgz", + "integrity": "sha512-Q4ZuBAe2FUsKtyQJoQHlvP8OvBERxO3jEmy1I7hcRXcJBGGHFh/aJBswbXuS9sgrDH2QUO8ilkwNPHvHMd8clg==", + "dependencies": { + "has": "^1.0.3" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-extendable": { + "version": "0.1.1", + "resolved": "https://registry.npmjs.org/is-extendable/-/is-extendable-0.1.1.tgz", + "integrity": "sha512-5BMULNob1vgFX6EjQw5izWDxrecWK9AM72rugNr0TFldMOi0fj6Jk+zeKIt0xGj4cEfQIJth4w3OKWOJ4f+AFw==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/is-extglob": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", + "integrity": "sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/is-fullwidth-code-point": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", + "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==", + "engines": { + "node": ">=8" + } + }, + "node_modules/is-glob": { + "version": "4.0.3", + "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.3.tgz", + "integrity": "sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==", + "dependencies": { + "is-extglob": "^2.1.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/is-number": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz", + "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==", + "engines": { + "node": ">=0.12.0" + } + }, + "node_modules/is-obj": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/is-obj/-/is-obj-3.0.0.tgz", + "integrity": "sha512-IlsXEHOjtKhpN8r/tRFj2nDyTmHvcfNeu/nrRIcXE17ROeatXchkojffa1SpdqW4cr/Fj6QkEf/Gn4zf6KKvEQ==", + "engines": { + "node": ">=12" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/is-plain-obj": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/is-plain-obj/-/is-plain-obj-4.1.0.tgz", + "integrity": "sha512-+Pgi+vMuUNkJyExiMBt5IlFoMyKnr5zhJ4Uspz58WOhBF5QoIZkFyNHIbBAtHwzVAgk5RtndVNsDRN61/mmDqg==", + "engines": { + "node": ">=12" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/is-potential-custom-element-name": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/is-potential-custom-element-name/-/is-potential-custom-element-name-1.0.1.tgz", + "integrity": "sha512-bCYeRA2rVibKZd+s2625gGnGF/t7DSqDs4dP7CrLA1m7jKWz6pps0LpYLJN8Q64HtmPKJ1hrN3nzPNKFEKOUiQ==" + }, + "node_modules/isexe": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz", + "integrity": "sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw==" + }, + "node_modules/jackspeak": { + "version": "2.2.1", + "resolved": "https://registry.npmjs.org/jackspeak/-/jackspeak-2.2.1.tgz", + "integrity": "sha512-MXbxovZ/Pm42f6cDIDkl3xpwv1AGwObKwfmjs2nQePiy85tP3fatofl3FC1aBsOtP/6fq5SbtgHwWcMsLP+bDw==", + "dependencies": { + "@isaacs/cliui": "^8.0.2" + }, + "engines": { + "node": ">=14" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + }, + "optionalDependencies": { + "@pkgjs/parseargs": "^0.11.0" + } + }, + "node_modules/js-yaml": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-4.1.0.tgz", + "integrity": "sha512-wpxZs9NoxZaJESJGIZTyDEaYpl0FKSA+FB9aJiyemKhMwkxQg63h4T1KJgUGHpTqPDNRcmmYLugrRjJlBtWvRA==", + "dependencies": { + "argparse": "^2.0.1" + }, + "bin": { + "js-yaml": "bin/js-yaml.js" + } + }, + "node_modules/jsdom": { + "version": "20.0.3", + "resolved": "https://registry.npmjs.org/jsdom/-/jsdom-20.0.3.tgz", + "integrity": "sha512-SYhBvTh89tTfCD/CRdSOm13mOBa42iTaTyfyEWBdKcGdPxPtLFBXuHR8XHb33YNYaP+lLbmSvBTsnoesCNJEsQ==", + "dependencies": { + "abab": "^2.0.6", + "acorn": "^8.8.1", + "acorn-globals": "^7.0.0", + "cssom": "^0.5.0", + "cssstyle": "^2.3.0", + "data-urls": "^3.0.2", + "decimal.js": "^10.4.2", + "domexception": "^4.0.0", + "escodegen": "^2.0.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.2", + "parse5": "^7.1.1", + "saxes": "^6.0.0", + "symbol-tree": "^3.2.4", + "tough-cookie": "^4.1.2", + "w3c-xmlserializer": "^4.0.0", + "webidl-conversions": "^7.0.0", + "whatwg-encoding": "^2.0.0", + "whatwg-mimetype": "^3.0.0", + "whatwg-url": "^11.0.0", + "ws": "^8.11.0", + "xml-name-validator": "^4.0.0" + }, + "engines": { + "node": ">=14" + }, + "peerDependencies": { + "canvas": "^2.5.0" + }, + "peerDependenciesMeta": { + "canvas": { + "optional": true + } + } + }, + "node_modules/jsonc-parser": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/jsonc-parser/-/jsonc-parser-3.2.0.tgz", + "integrity": "sha512-gfFQZrcTc8CnKXp6Y4/CBT3fTc0OVuDofpre4aEeEpSBPV5X5v4+Vmx+8snU7RLPrNHPKSgLxGo9YuQzz20o+w==", + "peer": true + }, + "node_modules/katex": { + "version": "0.16.8", + "resolved": "https://registry.npmjs.org/katex/-/katex-0.16.8.tgz", + "integrity": "sha512-ftuDnJbcbOckGY11OO+zg3OofESlbR5DRl2cmN8HeWeeFIV7wTXvAOx8kEjZjobhA+9wh2fbKeO6cdcA9Mnovg==", + "funding": [ + "https://opencollective.com/katex", + "https://github.com/sponsors/katex" + ], + "dependencies": { + "commander": "^8.3.0" + }, + "bin": { + "katex": "cli.js" + } + }, + "node_modules/katex/node_modules/commander": { + "version": "8.3.0", + "resolved": "https://registry.npmjs.org/commander/-/commander-8.3.0.tgz", + "integrity": "sha512-OkTL9umf+He2DZkUq8f8J9of7yL6RJKI24dVITBmNfZBmri9zYZQrKkuXiKhyfPSu8tUhnVBB1iKXevvnlR4Ww==", + "engines": { + "node": ">= 12" + } + }, + "node_modules/kind-of": { + "version": "6.0.3", + "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-6.0.3.tgz", + "integrity": "sha512-dcS1ul+9tmeD95T+x28/ehLgd9mENa3LsvDTtzm3vyBEO7RPptvAD+t44WVXaUjTBRcrpFeFlC8WCruUR456hw==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/kleur": { + "version": "4.1.5", + "resolved": "https://registry.npmjs.org/kleur/-/kleur-4.1.5.tgz", + "integrity": "sha512-o+NO+8WrRiQEE4/7nwRJhN1HWpVmJm511pBHUxPLtp0BUISzlBplORYSmTclCnJvQq2tKu/sgl3xVpkc7ZWuQQ==", + "engines": { + "node": ">=6" + } + }, + "node_modules/lightningcss": { + "version": "1.21.7", + "resolved": "https://registry.npmjs.org/lightningcss/-/lightningcss-1.21.7.tgz", + "integrity": "sha512-xITZyh5sLFwRPYUSw15T00Rm7gcQ1qOPuQwNOcvHsTm6nLWTQ723w7zl42wrC5t+xtdg6FPmnXHml1nZxxvp1w==", + "dependencies": { + "detect-libc": "^1.0.3" + }, + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + }, + "optionalDependencies": { + "lightningcss-darwin-arm64": "1.21.7", + "lightningcss-darwin-x64": "1.21.7", + "lightningcss-freebsd-x64": "1.21.7", + "lightningcss-linux-arm-gnueabihf": "1.21.7", + "lightningcss-linux-arm64-gnu": "1.21.7", + "lightningcss-linux-arm64-musl": "1.21.7", + "lightningcss-linux-x64-gnu": "1.21.7", + "lightningcss-linux-x64-musl": "1.21.7", + "lightningcss-win32-x64-msvc": "1.21.7" + } + }, + "node_modules/lightningcss-darwin-arm64": { + "version": "1.21.7", + "resolved": "https://registry.npmjs.org/lightningcss-darwin-arm64/-/lightningcss-darwin-arm64-1.21.7.tgz", + "integrity": "sha512-tt7hIsFio9jZofTVHtCACz6rB6c9RyABMXfA9A/VcKOjS3sq+koX/QkRJWY06utwOImbJIXBC5hbg9t3RkPUAQ==", + "cpu": [ + "arm64" + ], + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + } + }, + "node_modules/lightningcss-darwin-x64": { + "version": "1.21.7", + "resolved": "https://registry.npmjs.org/lightningcss-darwin-x64/-/lightningcss-darwin-x64-1.21.7.tgz", + "integrity": "sha512-F4gS4bf7eWekfPT+TxJNm/pF+QRgZiTrTkQH6cw4/UWfdeZISfuhD5El2dm16giFnY0K5ylIwO+ZusgYNkGSXA==", + "cpu": [ + "x64" + ], + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + } + }, + "node_modules/lightningcss-freebsd-x64": { + "version": "1.21.7", + "resolved": "https://registry.npmjs.org/lightningcss-freebsd-x64/-/lightningcss-freebsd-x64-1.21.7.tgz", + "integrity": "sha512-RMfNzJWXCSfPnL55fcLWEAadcY6QUFT0S8NceNKYzp1KiCZtkJIy6RQ5SaVxPzRqd3iMsahUf5sfnG8N1UQSNQ==", + "cpu": [ + "x64" + ], + "optional": true, + "os": [ + "freebsd" + ], + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + } + }, + "node_modules/lightningcss-linux-arm-gnueabihf": { + "version": "1.21.7", + "resolved": "https://registry.npmjs.org/lightningcss-linux-arm-gnueabihf/-/lightningcss-linux-arm-gnueabihf-1.21.7.tgz", + "integrity": "sha512-biSRUDZNx7vubWP1jArw/qqfZKPGpkV/qzunasZzxmqijbZ43sW9faDQYxWNcxPWljJJdF/qs6qcurYFovWtrQ==", + "cpu": [ + "arm" + ], + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + } + }, + "node_modules/lightningcss-linux-arm64-gnu": { + "version": "1.21.7", + "resolved": "https://registry.npmjs.org/lightningcss-linux-arm64-gnu/-/lightningcss-linux-arm64-gnu-1.21.7.tgz", + "integrity": "sha512-PENY8QekqL9TG3AY/A7rkUBb5ymefGxea7Oe7+x7Hbw4Bz4Hpj5cec5OoMypMqFbURPmpi0fTWx4vSWUPzpDcA==", + "cpu": [ + "arm64" + ], + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + } + }, + "node_modules/lightningcss-linux-arm64-musl": { + "version": "1.21.7", + "resolved": "https://registry.npmjs.org/lightningcss-linux-arm64-musl/-/lightningcss-linux-arm64-musl-1.21.7.tgz", + "integrity": "sha512-pfOipKvA/0X1OjRaZt3870vnV9UGBSjayIqHh0fGx/+aRz3O0MVFHE/60P2UWXpM3YGJEw/hMWtNkrFwqOge8A==", + "cpu": [ + "arm64" + ], + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + } + }, + "node_modules/lightningcss-linux-x64-gnu": { + "version": "1.21.7", + "resolved": "https://registry.npmjs.org/lightningcss-linux-x64-gnu/-/lightningcss-linux-x64-gnu-1.21.7.tgz", + "integrity": "sha512-dgcsis4TAA7s0ia4f31QHX+G4PWPwxk+wJaEQLaV0NdJs09O5hHoA8DpLEr8nrvc/tsRTyVNBP1rDtgzySjpXg==", + "cpu": [ + "x64" + ], + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + } + }, + "node_modules/lightningcss-linux-x64-musl": { + "version": "1.21.7", + "resolved": "https://registry.npmjs.org/lightningcss-linux-x64-musl/-/lightningcss-linux-x64-musl-1.21.7.tgz", + "integrity": "sha512-A+9dXpxld3p4Cd6fxev2eqEvaauYtrgNpXV3t7ioCJy30Oj9nYiNGwiGusM+4MJVcEpUPGUGiuAqY4sWilRDwA==", + "cpu": [ + "x64" + ], + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + } + }, + "node_modules/lightningcss-win32-x64-msvc": { + "version": "1.21.7", + "resolved": "https://registry.npmjs.org/lightningcss-win32-x64-msvc/-/lightningcss-win32-x64-msvc-1.21.7.tgz", + "integrity": "sha512-07/8vogEq+C/mF99pdMhh/f19/xreq8N9Ca6AWeVHZIdODyF/pt6KdKSCWDZWIn+3CUxI8gCJWuUWyOc3xymvw==", + "cpu": [ + "x64" + ], + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + } + }, + "node_modules/longest-streak": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/longest-streak/-/longest-streak-3.1.0.tgz", + "integrity": "sha512-9Ri+o0JYgehTaVBBDoMqIl8GXtbWg711O3srftcHhZ0dqnETqLaoIK0x17fUw9rFSlK/0NlsKe0Ahhyl5pXE2g==", + "funding": { + "type": "github", + "url": "https://github.com/sponsors/wooorm" + } + }, + "node_modules/lru-cache": { + "version": "10.0.0", + "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-10.0.0.tgz", + "integrity": "sha512-svTf/fzsKHffP42sujkO/Rjs37BCIsQVRCeNYIm9WN8rgT7ffoUnRtZCqU+6BqcSBdv8gwJeTz8knJpgACeQMw==", + "engines": { + "node": "14 || >=16.14" + } + }, + "node_modules/markdown-table": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/markdown-table/-/markdown-table-3.0.3.tgz", + "integrity": "sha512-Z1NL3Tb1M9wH4XESsCDEksWoKTdlUafKc4pt0GRwjUyXaCFZ+dc3g2erqB6zm3szA2IUSi7VnPI+o/9jnxh9hw==", + "funding": { + "type": "github", + "url": "https://github.com/sponsors/wooorm" + } + }, + "node_modules/mathjax-full": { + "version": "3.2.2", + "resolved": "https://registry.npmjs.org/mathjax-full/-/mathjax-full-3.2.2.tgz", + "integrity": "sha512-+LfG9Fik+OuI8SLwsiR02IVdjcnRCy5MufYLi0C3TdMT56L/pjB0alMVGgoWJF8pN9Rc7FESycZB9BMNWIid5w==", + "dependencies": { + "esm": "^3.2.25", + "mhchemparser": "^4.1.0", + "mj-context-menu": "^0.6.1", + "speech-rule-engine": "^4.0.6" + } + }, + "node_modules/mdast-util-definitions": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/mdast-util-definitions/-/mdast-util-definitions-5.1.2.tgz", + "integrity": "sha512-8SVPMuHqlPME/z3gqVwWY4zVXn8lqKv/pAhC57FuJ40ImXyBpmO5ukh98zB2v7Blql2FiHjHv9LVztSIqjY+MA==", + "dependencies": { + "@types/mdast": "^3.0.0", + "@types/unist": "^2.0.0", + "unist-util-visit": "^4.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/mdast-util-find-and-replace": { + "version": "2.2.2", + "resolved": "https://registry.npmjs.org/mdast-util-find-and-replace/-/mdast-util-find-and-replace-2.2.2.tgz", + "integrity": "sha512-MTtdFRz/eMDHXzeK6W3dO7mXUlF82Gom4y0oOgvHhh/HXZAGvIQDUvQ0SuUx+j2tv44b8xTHOm8K/9OoRFnXKw==", + "dependencies": { + "@types/mdast": "^3.0.0", + "escape-string-regexp": "^5.0.0", + "unist-util-is": "^5.0.0", + "unist-util-visit-parents": "^5.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/mdast-util-from-markdown": { + "version": "1.3.1", + "resolved": "https://registry.npmjs.org/mdast-util-from-markdown/-/mdast-util-from-markdown-1.3.1.tgz", + "integrity": "sha512-4xTO/M8c82qBcnQc1tgpNtubGUW/Y1tBQ1B0i5CtSoelOLKFYlElIr3bvgREYYO5iRqbMY1YuqZng0GVOI8Qww==", + "dependencies": { + "@types/mdast": "^3.0.0", + "@types/unist": "^2.0.0", + "decode-named-character-reference": "^1.0.0", + "mdast-util-to-string": "^3.1.0", + "micromark": "^3.0.0", + "micromark-util-decode-numeric-character-reference": "^1.0.0", + "micromark-util-decode-string": "^1.0.0", + "micromark-util-normalize-identifier": "^1.0.0", + "micromark-util-symbol": "^1.0.0", + "micromark-util-types": "^1.0.0", + "unist-util-stringify-position": "^3.0.0", + "uvu": "^0.5.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/mdast-util-frontmatter": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/mdast-util-frontmatter/-/mdast-util-frontmatter-1.0.1.tgz", + "integrity": "sha512-JjA2OjxRqAa8wEG8hloD0uTU0kdn8kbtOWpPP94NBkfAlbxn4S8gCGf/9DwFtEeGPXrDcNXdiDjVaRdUFqYokw==", + "dependencies": { + "@types/mdast": "^3.0.0", + "mdast-util-to-markdown": "^1.3.0", + "micromark-extension-frontmatter": "^1.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/mdast-util-gfm": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/mdast-util-gfm/-/mdast-util-gfm-2.0.2.tgz", + "integrity": "sha512-qvZ608nBppZ4icQlhQQIAdc6S3Ffj9RGmzwUKUWuEICFnd1LVkN3EktF7ZHAgfcEdvZB5owU9tQgt99e2TlLjg==", + "dependencies": { + "mdast-util-from-markdown": "^1.0.0", + "mdast-util-gfm-autolink-literal": "^1.0.0", + "mdast-util-gfm-footnote": "^1.0.0", + "mdast-util-gfm-strikethrough": "^1.0.0", + "mdast-util-gfm-table": "^1.0.0", + "mdast-util-gfm-task-list-item": "^1.0.0", + "mdast-util-to-markdown": "^1.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/mdast-util-gfm-autolink-literal": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/mdast-util-gfm-autolink-literal/-/mdast-util-gfm-autolink-literal-1.0.3.tgz", + "integrity": "sha512-My8KJ57FYEy2W2LyNom4n3E7hKTuQk/0SES0u16tjA9Z3oFkF4RrC/hPAPgjlSpezsOvI8ObcXcElo92wn5IGA==", + "dependencies": { + "@types/mdast": "^3.0.0", + "ccount": "^2.0.0", + "mdast-util-find-and-replace": "^2.0.0", + "micromark-util-character": "^1.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/mdast-util-gfm-footnote": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/mdast-util-gfm-footnote/-/mdast-util-gfm-footnote-1.0.2.tgz", + "integrity": "sha512-56D19KOGbE00uKVj3sgIykpwKL179QsVFwx/DCW0u/0+URsryacI4MAdNJl0dh+u2PSsD9FtxPFbHCzJ78qJFQ==", + "dependencies": { + "@types/mdast": "^3.0.0", + "mdast-util-to-markdown": "^1.3.0", + "micromark-util-normalize-identifier": "^1.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/mdast-util-gfm-strikethrough": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/mdast-util-gfm-strikethrough/-/mdast-util-gfm-strikethrough-1.0.3.tgz", + "integrity": "sha512-DAPhYzTYrRcXdMjUtUjKvW9z/FNAMTdU0ORyMcbmkwYNbKocDpdk+PX1L1dQgOID/+vVs1uBQ7ElrBQfZ0cuiQ==", + "dependencies": { + "@types/mdast": "^3.0.0", + "mdast-util-to-markdown": "^1.3.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/mdast-util-gfm-table": { + "version": "1.0.7", + "resolved": "https://registry.npmjs.org/mdast-util-gfm-table/-/mdast-util-gfm-table-1.0.7.tgz", + "integrity": "sha512-jjcpmNnQvrmN5Vx7y7lEc2iIOEytYv7rTvu+MeyAsSHTASGCCRA79Igg2uKssgOs1i1po8s3plW0sTu1wkkLGg==", + "dependencies": { + "@types/mdast": "^3.0.0", + "markdown-table": "^3.0.0", + "mdast-util-from-markdown": "^1.0.0", + "mdast-util-to-markdown": "^1.3.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/mdast-util-gfm-task-list-item": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/mdast-util-gfm-task-list-item/-/mdast-util-gfm-task-list-item-1.0.2.tgz", + "integrity": "sha512-PFTA1gzfp1B1UaiJVyhJZA1rm0+Tzn690frc/L8vNX1Jop4STZgOE6bxUhnzdVSB+vm2GU1tIsuQcA9bxTQpMQ==", + "dependencies": { + "@types/mdast": "^3.0.0", + "mdast-util-to-markdown": "^1.3.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/mdast-util-math": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/mdast-util-math/-/mdast-util-math-2.0.2.tgz", + "integrity": "sha512-8gmkKVp9v6+Tgjtq6SYx9kGPpTf6FVYRa53/DLh479aldR9AyP48qeVOgNZ5X7QUK7nOy4yw7vg6mbiGcs9jWQ==", + "dependencies": { + "@types/mdast": "^3.0.0", + "longest-streak": "^3.0.0", + "mdast-util-to-markdown": "^1.3.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/mdast-util-newline-to-break": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/mdast-util-newline-to-break/-/mdast-util-newline-to-break-1.0.0.tgz", + "integrity": "sha512-491LcYv3gbGhhCrLoeALncQmega2xPh+m3gbsIhVsOX4sw85+ShLFPvPyibxc1Swx/6GtzxgVodq+cGa/47ULg==", + "dependencies": { + "@types/mdast": "^3.0.0", + "mdast-util-find-and-replace": "^2.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/mdast-util-phrasing": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/mdast-util-phrasing/-/mdast-util-phrasing-3.0.1.tgz", + "integrity": "sha512-WmI1gTXUBJo4/ZmSk79Wcb2HcjPJBzM1nlI/OUWA8yk2X9ik3ffNbBGsU+09BFmXaL1IBb9fiuvq6/KMiNycSg==", + "dependencies": { + "@types/mdast": "^3.0.0", + "unist-util-is": "^5.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/mdast-util-to-hast": { + "version": "12.3.0", + "resolved": "https://registry.npmjs.org/mdast-util-to-hast/-/mdast-util-to-hast-12.3.0.tgz", + "integrity": "sha512-pits93r8PhnIoU4Vy9bjW39M2jJ6/tdHyja9rrot9uujkN7UTU9SDnE6WNJz/IGyQk3XHX6yNNtrBH6cQzm8Hw==", + "dependencies": { + "@types/hast": "^2.0.0", + "@types/mdast": "^3.0.0", + "mdast-util-definitions": "^5.0.0", + "micromark-util-sanitize-uri": "^1.1.0", + "trim-lines": "^3.0.0", + "unist-util-generated": "^2.0.0", + "unist-util-position": "^4.0.0", + "unist-util-visit": "^4.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/mdast-util-to-markdown": { + "version": "1.5.0", + "resolved": "https://registry.npmjs.org/mdast-util-to-markdown/-/mdast-util-to-markdown-1.5.0.tgz", + "integrity": "sha512-bbv7TPv/WC49thZPg3jXuqzuvI45IL2EVAr/KxF0BSdHsU0ceFHOmwQn6evxAh1GaoK/6GQ1wp4R4oW2+LFL/A==", + "dependencies": { + "@types/mdast": "^3.0.0", + "@types/unist": "^2.0.0", + "longest-streak": "^3.0.0", + "mdast-util-phrasing": "^3.0.0", + "mdast-util-to-string": "^3.0.0", + "micromark-util-decode-string": "^1.0.0", + "unist-util-visit": "^4.0.0", + "zwitch": "^2.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/mdast-util-to-string": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/mdast-util-to-string/-/mdast-util-to-string-3.2.0.tgz", + "integrity": "sha512-V4Zn/ncyN1QNSqSBxTrMOLpjr+IKdHl2v3KVLoWmDPscP4r9GcCi71gjgvUV1SFSKh92AjAG4peFuBl2/YgCJg==", + "dependencies": { + "@types/mdast": "^3.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/merge2": { + "version": "1.4.1", + "resolved": "https://registry.npmjs.org/merge2/-/merge2-1.4.1.tgz", + "integrity": "sha512-8q7VEgMJW4J8tcfVPy8g09NcQwZdbwFEqhe/WZkoIzjn/3TGDwtOCYtXGxA3O8tPzpczCCDgv+P2P5y00ZJOOg==", + "engines": { + "node": ">= 8" + } + }, + "node_modules/mhchemparser": { + "version": "4.2.1", + "resolved": "https://registry.npmjs.org/mhchemparser/-/mhchemparser-4.2.1.tgz", + "integrity": "sha512-kYmyrCirqJf3zZ9t/0wGgRZ4/ZJw//VwaRVGA75C4nhE60vtnIzhl9J9ndkX/h6hxSN7pjg/cE0VxbnNM+bnDQ==" + }, + "node_modules/micromark": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/micromark/-/micromark-3.2.0.tgz", + "integrity": "sha512-uD66tJj54JLYq0De10AhWycZWGQNUvDI55xPgk2sQM5kn1JYlhbCMTtEeT27+vAhW2FBQxLlOmS3pmA7/2z4aA==", + "funding": [ + { + "type": "GitHub Sponsors", + "url": "https://github.com/sponsors/unifiedjs" + }, + { + "type": "OpenCollective", + "url": "https://opencollective.com/unified" + } + ], + "dependencies": { + "@types/debug": "^4.0.0", + "debug": "^4.0.0", + "decode-named-character-reference": "^1.0.0", + "micromark-core-commonmark": "^1.0.1", + "micromark-factory-space": "^1.0.0", + "micromark-util-character": "^1.0.0", + "micromark-util-chunked": "^1.0.0", + "micromark-util-combine-extensions": "^1.0.0", + "micromark-util-decode-numeric-character-reference": "^1.0.0", + "micromark-util-encode": "^1.0.0", + "micromark-util-normalize-identifier": "^1.0.0", + "micromark-util-resolve-all": "^1.0.0", + "micromark-util-sanitize-uri": "^1.0.0", + "micromark-util-subtokenize": "^1.0.0", + "micromark-util-symbol": "^1.0.0", + "micromark-util-types": "^1.0.1", + "uvu": "^0.5.0" + } + }, + "node_modules/micromark-core-commonmark": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/micromark-core-commonmark/-/micromark-core-commonmark-1.1.0.tgz", + "integrity": "sha512-BgHO1aRbolh2hcrzL2d1La37V0Aoz73ymF8rAcKnohLy93titmv62E0gP8Hrx9PKcKrqCZ1BbLGbP3bEhoXYlw==", + "funding": [ + { + "type": "GitHub Sponsors", + "url": "https://github.com/sponsors/unifiedjs" + }, + { + "type": "OpenCollective", + "url": "https://opencollective.com/unified" + } + ], + "dependencies": { + "decode-named-character-reference": "^1.0.0", + "micromark-factory-destination": "^1.0.0", + "micromark-factory-label": "^1.0.0", + "micromark-factory-space": "^1.0.0", + "micromark-factory-title": "^1.0.0", + "micromark-factory-whitespace": "^1.0.0", + "micromark-util-character": "^1.0.0", + "micromark-util-chunked": "^1.0.0", + "micromark-util-classify-character": "^1.0.0", + "micromark-util-html-tag-name": "^1.0.0", + "micromark-util-normalize-identifier": "^1.0.0", + "micromark-util-resolve-all": "^1.0.0", + "micromark-util-subtokenize": "^1.0.0", + "micromark-util-symbol": "^1.0.0", + "micromark-util-types": "^1.0.1", + "uvu": "^0.5.0" + } + }, + "node_modules/micromark-extension-frontmatter": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/micromark-extension-frontmatter/-/micromark-extension-frontmatter-1.1.1.tgz", + "integrity": "sha512-m2UH9a7n3W8VAH9JO9y01APpPKmNNNs71P0RbknEmYSaZU5Ghogv38BYO94AI5Xw6OYfxZRdHZZ2nYjs/Z+SZQ==", + "dependencies": { + "fault": "^2.0.0", + "micromark-util-character": "^1.0.0", + "micromark-util-symbol": "^1.0.0", + "micromark-util-types": "^1.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/micromark-extension-gfm": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/micromark-extension-gfm/-/micromark-extension-gfm-2.0.3.tgz", + "integrity": "sha512-vb9OoHqrhCmbRidQv/2+Bc6pkP0FrtlhurxZofvOEy5o8RtuuvTq+RQ1Vw5ZDNrVraQZu3HixESqbG+0iKk/MQ==", + "dependencies": { + "micromark-extension-gfm-autolink-literal": "^1.0.0", + "micromark-extension-gfm-footnote": "^1.0.0", + "micromark-extension-gfm-strikethrough": "^1.0.0", + "micromark-extension-gfm-table": "^1.0.0", + "micromark-extension-gfm-tagfilter": "^1.0.0", + "micromark-extension-gfm-task-list-item": "^1.0.0", + "micromark-util-combine-extensions": "^1.0.0", + "micromark-util-types": "^1.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/micromark-extension-gfm-autolink-literal": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/micromark-extension-gfm-autolink-literal/-/micromark-extension-gfm-autolink-literal-1.0.5.tgz", + "integrity": "sha512-z3wJSLrDf8kRDOh2qBtoTRD53vJ+CWIyo7uyZuxf/JAbNJjiHsOpG1y5wxk8drtv3ETAHutCu6N3thkOOgueWg==", + "dependencies": { + "micromark-util-character": "^1.0.0", + "micromark-util-sanitize-uri": "^1.0.0", + "micromark-util-symbol": "^1.0.0", + "micromark-util-types": "^1.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/micromark-extension-gfm-footnote": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/micromark-extension-gfm-footnote/-/micromark-extension-gfm-footnote-1.1.2.tgz", + "integrity": "sha512-Yxn7z7SxgyGWRNa4wzf8AhYYWNrwl5q1Z8ii+CSTTIqVkmGZF1CElX2JI8g5yGoM3GAman9/PVCUFUSJ0kB/8Q==", + "dependencies": { + "micromark-core-commonmark": "^1.0.0", + "micromark-factory-space": "^1.0.0", + "micromark-util-character": "^1.0.0", + "micromark-util-normalize-identifier": "^1.0.0", + "micromark-util-sanitize-uri": "^1.0.0", + "micromark-util-symbol": "^1.0.0", + "micromark-util-types": "^1.0.0", + "uvu": "^0.5.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/micromark-extension-gfm-strikethrough": { + "version": "1.0.7", + "resolved": "https://registry.npmjs.org/micromark-extension-gfm-strikethrough/-/micromark-extension-gfm-strikethrough-1.0.7.tgz", + "integrity": "sha512-sX0FawVE1o3abGk3vRjOH50L5TTLr3b5XMqnP9YDRb34M0v5OoZhG+OHFz1OffZ9dlwgpTBKaT4XW/AsUVnSDw==", + "dependencies": { + "micromark-util-chunked": "^1.0.0", + "micromark-util-classify-character": "^1.0.0", + "micromark-util-resolve-all": "^1.0.0", + "micromark-util-symbol": "^1.0.0", + "micromark-util-types": "^1.0.0", + "uvu": "^0.5.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/micromark-extension-gfm-table": { + "version": "1.0.7", + "resolved": "https://registry.npmjs.org/micromark-extension-gfm-table/-/micromark-extension-gfm-table-1.0.7.tgz", + "integrity": "sha512-3ZORTHtcSnMQEKtAOsBQ9/oHp9096pI/UvdPtN7ehKvrmZZ2+bbWhi0ln+I9drmwXMt5boocn6OlwQzNXeVeqw==", + "dependencies": { + "micromark-factory-space": "^1.0.0", + "micromark-util-character": "^1.0.0", + "micromark-util-symbol": "^1.0.0", + "micromark-util-types": "^1.0.0", + "uvu": "^0.5.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/micromark-extension-gfm-tagfilter": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/micromark-extension-gfm-tagfilter/-/micromark-extension-gfm-tagfilter-1.0.2.tgz", + "integrity": "sha512-5XWB9GbAUSHTn8VPU8/1DBXMuKYT5uOgEjJb8gN3mW0PNW5OPHpSdojoqf+iq1xo7vWzw/P8bAHY0n6ijpXF7g==", + "dependencies": { + "micromark-util-types": "^1.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/micromark-extension-gfm-task-list-item": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/micromark-extension-gfm-task-list-item/-/micromark-extension-gfm-task-list-item-1.0.5.tgz", + "integrity": "sha512-RMFXl2uQ0pNQy6Lun2YBYT9g9INXtWJULgbt01D/x8/6yJ2qpKyzdZD3pi6UIkzF++Da49xAelVKUeUMqd5eIQ==", + "dependencies": { + "micromark-factory-space": "^1.0.0", + "micromark-util-character": "^1.0.0", + "micromark-util-symbol": "^1.0.0", + "micromark-util-types": "^1.0.0", + "uvu": "^0.5.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/micromark-extension-math": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/micromark-extension-math/-/micromark-extension-math-2.1.2.tgz", + "integrity": "sha512-es0CcOV89VNS9wFmyn+wyFTKweXGW4CEvdaAca6SWRWPyYCbBisnjaHLjWO4Nszuiud84jCpkHsqAJoa768Pvg==", + "dependencies": { + "@types/katex": "^0.16.0", + "katex": "^0.16.0", + "micromark-factory-space": "^1.0.0", + "micromark-util-character": "^1.0.0", + "micromark-util-symbol": "^1.0.0", + "micromark-util-types": "^1.0.0", + "uvu": "^0.5.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/micromark-extension-math/node_modules/@types/katex": { + "version": "0.16.0", + "resolved": "https://registry.npmjs.org/@types/katex/-/katex-0.16.0.tgz", + "integrity": "sha512-hz+S3nV6Mym5xPbT9fnO8dDhBFQguMYpY0Ipxv06JMi1ORgnEM4M1ymWDUhUNer3ElLmT583opRo4RzxKmh9jw==" + }, + "node_modules/micromark-factory-destination": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/micromark-factory-destination/-/micromark-factory-destination-1.1.0.tgz", + "integrity": "sha512-XaNDROBgx9SgSChd69pjiGKbV+nfHGDPVYFs5dOoDd7ZnMAE+Cuu91BCpsY8RT2NP9vo/B8pds2VQNCLiu0zhg==", + "funding": [ + { + "type": "GitHub Sponsors", + "url": "https://github.com/sponsors/unifiedjs" + }, + { + "type": "OpenCollective", + "url": "https://opencollective.com/unified" + } + ], + "dependencies": { + "micromark-util-character": "^1.0.0", + "micromark-util-symbol": "^1.0.0", + "micromark-util-types": "^1.0.0" + } + }, + "node_modules/micromark-factory-label": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/micromark-factory-label/-/micromark-factory-label-1.1.0.tgz", + "integrity": "sha512-OLtyez4vZo/1NjxGhcpDSbHQ+m0IIGnT8BoPamh+7jVlzLJBH98zzuCoUeMxvM6WsNeh8wx8cKvqLiPHEACn0w==", + "funding": [ + { + "type": "GitHub Sponsors", + "url": "https://github.com/sponsors/unifiedjs" + }, + { + "type": "OpenCollective", + "url": "https://opencollective.com/unified" + } + ], + "dependencies": { + "micromark-util-character": "^1.0.0", + "micromark-util-symbol": "^1.0.0", + "micromark-util-types": "^1.0.0", + "uvu": "^0.5.0" + } + }, + "node_modules/micromark-factory-space": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/micromark-factory-space/-/micromark-factory-space-1.1.0.tgz", + "integrity": "sha512-cRzEj7c0OL4Mw2v6nwzttyOZe8XY/Z8G0rzmWQZTBi/jjwyw/U4uqKtUORXQrR5bAZZnbTI/feRV/R7hc4jQYQ==", + "funding": [ + { + "type": "GitHub Sponsors", + "url": "https://github.com/sponsors/unifiedjs" + }, + { + "type": "OpenCollective", + "url": "https://opencollective.com/unified" + } + ], + "dependencies": { + "micromark-util-character": "^1.0.0", + "micromark-util-types": "^1.0.0" + } + }, + "node_modules/micromark-factory-title": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/micromark-factory-title/-/micromark-factory-title-1.1.0.tgz", + "integrity": "sha512-J7n9R3vMmgjDOCY8NPw55jiyaQnH5kBdV2/UXCtZIpnHH3P6nHUKaH7XXEYuWwx/xUJcawa8plLBEjMPU24HzQ==", + "funding": [ + { + "type": "GitHub Sponsors", + "url": "https://github.com/sponsors/unifiedjs" + }, + { + "type": "OpenCollective", + "url": "https://opencollective.com/unified" + } + ], + "dependencies": { + "micromark-factory-space": "^1.0.0", + "micromark-util-character": "^1.0.0", + "micromark-util-symbol": "^1.0.0", + "micromark-util-types": "^1.0.0" + } + }, + "node_modules/micromark-factory-whitespace": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/micromark-factory-whitespace/-/micromark-factory-whitespace-1.1.0.tgz", + "integrity": "sha512-v2WlmiymVSp5oMg+1Q0N1Lxmt6pMhIHD457whWM7/GUlEks1hI9xj5w3zbc4uuMKXGisksZk8DzP2UyGbGqNsQ==", + "funding": [ + { + "type": "GitHub Sponsors", + "url": "https://github.com/sponsors/unifiedjs" + }, + { + "type": "OpenCollective", + "url": "https://opencollective.com/unified" + } + ], + "dependencies": { + "micromark-factory-space": "^1.0.0", + "micromark-util-character": "^1.0.0", + "micromark-util-symbol": "^1.0.0", + "micromark-util-types": "^1.0.0" + } + }, + "node_modules/micromark-util-character": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/micromark-util-character/-/micromark-util-character-1.2.0.tgz", + "integrity": "sha512-lXraTwcX3yH/vMDaFWCQJP1uIszLVebzUa3ZHdrgxr7KEU/9mL4mVgCpGbyhvNLNlauROiNUq7WN5u7ndbY6xg==", + "funding": [ + { + "type": "GitHub Sponsors", + "url": "https://github.com/sponsors/unifiedjs" + }, + { + "type": "OpenCollective", + "url": "https://opencollective.com/unified" + } + ], + "dependencies": { + "micromark-util-symbol": "^1.0.0", + "micromark-util-types": "^1.0.0" + } + }, + "node_modules/micromark-util-chunked": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/micromark-util-chunked/-/micromark-util-chunked-1.1.0.tgz", + "integrity": "sha512-Ye01HXpkZPNcV6FiyoW2fGZDUw4Yc7vT0E9Sad83+bEDiCJ1uXu0S3mr8WLpsz3HaG3x2q0HM6CTuPdcZcluFQ==", + "funding": [ + { + "type": "GitHub Sponsors", + "url": "https://github.com/sponsors/unifiedjs" + }, + { + "type": "OpenCollective", + "url": "https://opencollective.com/unified" + } + ], + "dependencies": { + "micromark-util-symbol": "^1.0.0" + } + }, + "node_modules/micromark-util-classify-character": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/micromark-util-classify-character/-/micromark-util-classify-character-1.1.0.tgz", + "integrity": "sha512-SL0wLxtKSnklKSUplok1WQFoGhUdWYKggKUiqhX+Swala+BtptGCu5iPRc+xvzJ4PXE/hwM3FNXsfEVgoZsWbw==", + "funding": [ + { + "type": "GitHub Sponsors", + "url": "https://github.com/sponsors/unifiedjs" + }, + { + "type": "OpenCollective", + "url": "https://opencollective.com/unified" + } + ], + "dependencies": { + "micromark-util-character": "^1.0.0", + "micromark-util-symbol": "^1.0.0", + "micromark-util-types": "^1.0.0" + } + }, + "node_modules/micromark-util-combine-extensions": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/micromark-util-combine-extensions/-/micromark-util-combine-extensions-1.1.0.tgz", + "integrity": "sha512-Q20sp4mfNf9yEqDL50WwuWZHUrCO4fEyeDCnMGmG5Pr0Cz15Uo7KBs6jq+dq0EgX4DPwwrh9m0X+zPV1ypFvUA==", + "funding": [ + { + "type": "GitHub Sponsors", + "url": "https://github.com/sponsors/unifiedjs" + }, + { + "type": "OpenCollective", + "url": "https://opencollective.com/unified" + } + ], + "dependencies": { + "micromark-util-chunked": "^1.0.0", + "micromark-util-types": "^1.0.0" + } + }, + "node_modules/micromark-util-decode-numeric-character-reference": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/micromark-util-decode-numeric-character-reference/-/micromark-util-decode-numeric-character-reference-1.1.0.tgz", + "integrity": "sha512-m9V0ExGv0jB1OT21mrWcuf4QhP46pH1KkfWy9ZEezqHKAxkj4mPCy3nIH1rkbdMlChLHX531eOrymlwyZIf2iw==", + "funding": [ + { + "type": "GitHub Sponsors", + "url": "https://github.com/sponsors/unifiedjs" + }, + { + "type": "OpenCollective", + "url": "https://opencollective.com/unified" + } + ], + "dependencies": { + "micromark-util-symbol": "^1.0.0" + } + }, + "node_modules/micromark-util-decode-string": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/micromark-util-decode-string/-/micromark-util-decode-string-1.1.0.tgz", + "integrity": "sha512-YphLGCK8gM1tG1bd54azwyrQRjCFcmgj2S2GoJDNnh4vYtnL38JS8M4gpxzOPNyHdNEpheyWXCTnnTDY3N+NVQ==", + "funding": [ + { + "type": "GitHub Sponsors", + "url": "https://github.com/sponsors/unifiedjs" + }, + { + "type": "OpenCollective", + "url": "https://opencollective.com/unified" + } + ], + "dependencies": { + "decode-named-character-reference": "^1.0.0", + "micromark-util-character": "^1.0.0", + "micromark-util-decode-numeric-character-reference": "^1.0.0", + "micromark-util-symbol": "^1.0.0" + } + }, + "node_modules/micromark-util-encode": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/micromark-util-encode/-/micromark-util-encode-1.1.0.tgz", + "integrity": "sha512-EuEzTWSTAj9PA5GOAs992GzNh2dGQO52UvAbtSOMvXTxv3Criqb6IOzJUBCmEqrrXSblJIJBbFFv6zPxpreiJw==", + "funding": [ + { + "type": "GitHub Sponsors", + "url": "https://github.com/sponsors/unifiedjs" + }, + { + "type": "OpenCollective", + "url": "https://opencollective.com/unified" + } + ] + }, + "node_modules/micromark-util-html-tag-name": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/micromark-util-html-tag-name/-/micromark-util-html-tag-name-1.2.0.tgz", + "integrity": "sha512-VTQzcuQgFUD7yYztuQFKXT49KghjtETQ+Wv/zUjGSGBioZnkA4P1XXZPT1FHeJA6RwRXSF47yvJ1tsJdoxwO+Q==", + "funding": [ + { + "type": "GitHub Sponsors", + "url": "https://github.com/sponsors/unifiedjs" + }, + { + "type": "OpenCollective", + "url": "https://opencollective.com/unified" + } + ] + }, + "node_modules/micromark-util-normalize-identifier": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/micromark-util-normalize-identifier/-/micromark-util-normalize-identifier-1.1.0.tgz", + "integrity": "sha512-N+w5vhqrBihhjdpM8+5Xsxy71QWqGn7HYNUvch71iV2PM7+E3uWGox1Qp90loa1ephtCxG2ftRV/Conitc6P2Q==", + "funding": [ + { + "type": "GitHub Sponsors", + "url": "https://github.com/sponsors/unifiedjs" + }, + { + "type": "OpenCollective", + "url": "https://opencollective.com/unified" + } + ], + "dependencies": { + "micromark-util-symbol": "^1.0.0" + } + }, + "node_modules/micromark-util-resolve-all": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/micromark-util-resolve-all/-/micromark-util-resolve-all-1.1.0.tgz", + "integrity": "sha512-b/G6BTMSg+bX+xVCshPTPyAu2tmA0E4X98NSR7eIbeC6ycCqCeE7wjfDIgzEbkzdEVJXRtOG4FbEm/uGbCRouA==", + "funding": [ + { + "type": "GitHub Sponsors", + "url": "https://github.com/sponsors/unifiedjs" + }, + { + "type": "OpenCollective", + "url": "https://opencollective.com/unified" + } + ], + "dependencies": { + "micromark-util-types": "^1.0.0" + } + }, + "node_modules/micromark-util-sanitize-uri": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/micromark-util-sanitize-uri/-/micromark-util-sanitize-uri-1.2.0.tgz", + "integrity": "sha512-QO4GXv0XZfWey4pYFndLUKEAktKkG5kZTdUNaTAkzbuJxn2tNBOr+QtxR2XpWaMhbImT2dPzyLrPXLlPhph34A==", + "funding": [ + { + "type": "GitHub Sponsors", + "url": "https://github.com/sponsors/unifiedjs" + }, + { + "type": "OpenCollective", + "url": "https://opencollective.com/unified" + } + ], + "dependencies": { + "micromark-util-character": "^1.0.0", + "micromark-util-encode": "^1.0.0", + "micromark-util-symbol": "^1.0.0" + } + }, + "node_modules/micromark-util-subtokenize": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/micromark-util-subtokenize/-/micromark-util-subtokenize-1.1.0.tgz", + "integrity": "sha512-kUQHyzRoxvZO2PuLzMt2P/dwVsTiivCK8icYTeR+3WgbuPqfHgPPy7nFKbeqRivBvn/3N3GBiNC+JRTMSxEC7A==", + "funding": [ + { + "type": "GitHub Sponsors", + "url": "https://github.com/sponsors/unifiedjs" + }, + { + "type": "OpenCollective", + "url": "https://opencollective.com/unified" + } + ], + "dependencies": { + "micromark-util-chunked": "^1.0.0", + "micromark-util-symbol": "^1.0.0", + "micromark-util-types": "^1.0.0", + "uvu": "^0.5.0" + } + }, + "node_modules/micromark-util-symbol": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/micromark-util-symbol/-/micromark-util-symbol-1.1.0.tgz", + "integrity": "sha512-uEjpEYY6KMs1g7QfJ2eX1SQEV+ZT4rUD3UcF6l57acZvLNK7PBZL+ty82Z1qhK1/yXIY4bdx04FKMgR0g4IAag==", + "funding": [ + { + "type": "GitHub Sponsors", + "url": "https://github.com/sponsors/unifiedjs" + }, + { + "type": "OpenCollective", + "url": "https://opencollective.com/unified" + } + ] + }, + "node_modules/micromark-util-types": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/micromark-util-types/-/micromark-util-types-1.1.0.tgz", + "integrity": "sha512-ukRBgie8TIAcacscVHSiddHjO4k/q3pnedmzMQ4iwDcK0FtFCohKOlFbaOL/mPgfnPsL3C1ZyxJa4sbWrBl3jg==", + "funding": [ + { + "type": "GitHub Sponsors", + "url": "https://github.com/sponsors/unifiedjs" + }, + { + "type": "OpenCollective", + "url": "https://opencollective.com/unified" + } + ] + }, + "node_modules/micromatch": { + "version": "4.0.5", + "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-4.0.5.tgz", + "integrity": "sha512-DMy+ERcEW2q8Z2Po+WNXuw3c5YaUSFjAO5GsJqfEl7UjvtIuFKO6ZrKvcItdy98dwFI2N1tg3zNIdKaQT+aNdA==", + "dependencies": { + "braces": "^3.0.2", + "picomatch": "^2.3.1" + }, + "engines": { + "node": ">=8.6" + } + }, + "node_modules/micromorph": { + "version": "0.4.5", + "resolved": "https://registry.npmjs.org/micromorph/-/micromorph-0.4.5.tgz", + "integrity": "sha512-Erasr0xiDvDeEhh7B/k7RFTwwfaAX10D7BMorNpokkwDh6XsRLYWDPaWF1m5JQeMSkGdqlEtQ8s68NcdDWuGgw==" + }, + "node_modules/mime-db": { + "version": "1.33.0", + "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.33.0.tgz", + "integrity": "sha512-BHJ/EKruNIqJf/QahvxwQZXKygOQ256myeN/Ew+THcAa5q+PjyTTMMeNQC4DZw5AwfvelsUrA6B67NKMqXDbzQ==", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/mime-types": { + "version": "2.1.18", + "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.18.tgz", + "integrity": "sha512-lc/aahn+t4/SWV/qcmumYjymLsWfN3ELhpmVuUFjgsORruuZPVSwAQryq+HHGvO/SI2KVX26bx+En+zhM8g8hQ==", + "dependencies": { + "mime-db": "~1.33.0" + }, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/minimatch": { + "version": "9.0.2", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-9.0.2.tgz", + "integrity": "sha512-PZOT9g5v2ojiTL7r1xF6plNHLtOeTpSlDI007As2NlA2aYBMfVom17yqa6QzhmDP8QOhn7LjHTg7DFCVSSa6yg==", + "dependencies": { + "brace-expansion": "^2.0.1" + }, + "engines": { + "node": ">=16 || 14 >=14.17" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/minipass": { + "version": "6.0.2", + "resolved": "https://registry.npmjs.org/minipass/-/minipass-6.0.2.tgz", + "integrity": "sha512-MzWSV5nYVT7mVyWCwn2o7JH13w2TBRmmSqSRCKzTw+lmft9X4z+3wjvs06Tzijo5z4W/kahUCDpRXTF+ZrmF/w==", + "engines": { + "node": ">=16 || 14 >=14.17" + } + }, + "node_modules/mj-context-menu": { + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/mj-context-menu/-/mj-context-menu-0.6.1.tgz", + "integrity": "sha512-7NO5s6n10TIV96d4g2uDpG7ZDpIhMh0QNfGdJw/W47JswFcosz457wqz/b5sAKvl12sxINGFCn80NZHKwxQEXA==" + }, + "node_modules/mri": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/mri/-/mri-1.2.0.tgz", + "integrity": "sha512-tzzskb3bG8LvYGFF/mDTpq3jpI6Q9wc3LEmBaghu+DdCssd1FakN7Bc0hVNmEyGq1bq3RgfkCb3cmQLpNPOroA==", + "engines": { + "node": ">=4" + } + }, + "node_modules/ms": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", + "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==" + }, + "node_modules/nlcst-to-string": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/nlcst-to-string/-/nlcst-to-string-3.1.1.tgz", + "integrity": "sha512-63mVyqaqt0cmn2VcI2aH6kxe1rLAmSROqHMA0i4qqg1tidkfExgpb0FGMikMCn86mw5dFtBtEANfmSSK7TjNHw==", + "dependencies": { + "@types/nlcst": "^1.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/normalize-path": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-3.0.0.tgz", + "integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/nwsapi": { + "version": "2.2.7", + "resolved": "https://registry.npmjs.org/nwsapi/-/nwsapi-2.2.7.tgz", + "integrity": "sha512-ub5E4+FBPKwAZx0UwIQOjYWGHTEq5sPqHQNRN8Z9e4A7u3Tj1weLJsL59yH9vmvqEtBHaOmT6cYQKIZOxp35FQ==" + }, + "node_modules/parse-latin": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/parse-latin/-/parse-latin-5.0.1.tgz", + "integrity": "sha512-b/K8ExXaWC9t34kKeDV8kGXBkXZ1HCSAZRYE7HR14eA1GlXX5L8iWhs8USJNhQU9q5ci413jCKF0gOyovvyRBg==", + "dependencies": { + "nlcst-to-string": "^3.0.0", + "unist-util-modify-children": "^3.0.0", + "unist-util-visit-children": "^2.0.0" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/wooorm" + } + }, + "node_modules/parse-numeric-range": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/parse-numeric-range/-/parse-numeric-range-1.3.0.tgz", + "integrity": "sha512-twN+njEipszzlMJd4ONUYgSfZPDxgHhT9Ahed5uTigpQn90FggW4SA/AIPq/6a149fTbE9qBEcSwE3FAEp6wQQ==" + }, + "node_modules/parse5": { + "version": "7.1.2", + "resolved": "https://registry.npmjs.org/parse5/-/parse5-7.1.2.tgz", + "integrity": "sha512-Czj1WaSVpaoj0wbhMzLmWD69anp2WH7FXMB9n1Sy8/ZFF9jolSQVMu1Ij5WIyGmcBmhk7EOndpO4mIpihVqAXw==", + "dependencies": { + "entities": "^4.4.0" + }, + "funding": { + "url": "https://github.com/inikulin/parse5?sponsor=1" + } + }, + "node_modules/path-is-inside": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/path-is-inside/-/path-is-inside-1.0.2.tgz", + "integrity": "sha512-DUWJr3+ULp4zXmol/SZkFf3JGsS9/SIv+Y3Rt93/UjPpDpklB5f1er4O3POIbUuUJ3FXgqte2Q7SrU6zAqwk8w==" + }, + "node_modules/path-key": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/path-key/-/path-key-3.1.1.tgz", + "integrity": "sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q==", + "engines": { + "node": ">=8" + } + }, + "node_modules/path-parse": { + "version": "1.0.7", + "resolved": "https://registry.npmjs.org/path-parse/-/path-parse-1.0.7.tgz", + "integrity": "sha512-LDJzPVEEEPR+y48z93A0Ed0yXb8pAByGWo/k5YYdYgpY2/2EsOsksJrq7lOHxryrVOn1ejG6oAp8ahvOIQD8sw==" + }, + "node_modules/path-scurry": { + "version": "1.10.0", + "resolved": "https://registry.npmjs.org/path-scurry/-/path-scurry-1.10.0.tgz", + "integrity": "sha512-tZFEaRQbMLjwrsmidsGJ6wDMv0iazJWk6SfIKnY4Xru8auXgmJkOBa5DUbYFcFD2Rzk2+KDlIiF0GVXNCbgC7g==", + "dependencies": { + "lru-cache": "^9.1.1 || ^10.0.0", + "minipass": "^5.0.0 || ^6.0.2" + }, + "engines": { + "node": ">=16 || 14 >=14.17" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/path-to-regexp": { + "version": "2.2.1", + "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-2.2.1.tgz", + "integrity": "sha512-gu9bD6Ta5bwGrrU8muHzVOBFFREpp2iRkVfhBJahwJ6p6Xw20SjT0MxLnwkjOibQmGSYhiUnf2FLe7k+jcFmGQ==" + }, + "node_modules/path-type": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/path-type/-/path-type-4.0.0.tgz", + "integrity": "sha512-gDKb8aZMDeD/tZWs9P6+q0J9Mwkdl6xMV8TjnGP3qJVJ06bdMgkbBlLU8IdfOsIsFz2BW1rNVT3XuNEl8zPAvw==", + "engines": { + "node": ">=8" + } + }, + "node_modules/picocolors": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/picocolors/-/picocolors-1.0.0.tgz", + "integrity": "sha512-1fygroTLlHu66zi26VoTDv8yRgm0Fccecssto+MhsZ0D/DGW2sm8E8AjW7NU5VVTRt5GxbeZ5qBuJr+HyLYkjQ==" + }, + "node_modules/picomatch": { + "version": "2.3.1", + "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.1.tgz", + "integrity": "sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==", + "engines": { + "node": ">=8.6" + }, + "funding": { + "url": "https://github.com/sponsors/jonschlinkert" + } + }, + "node_modules/plausible-tracker": { + "version": "0.3.8", + "resolved": "https://registry.npmjs.org/plausible-tracker/-/plausible-tracker-0.3.8.tgz", + "integrity": "sha512-lmOWYQ7s9KOUJ1R+YTOR3HrjdbxIS2Z4de0P/Jx2dQPteznJl2eX3tXxKClpvbfyGP59B5bbhW8ftN59HbbFSg==", + "engines": { + "node": ">=10" + } + }, + "node_modules/preact": { + "version": "10.15.1", + "resolved": "https://registry.npmjs.org/preact/-/preact-10.15.1.tgz", + "integrity": "sha512-qs2ansoQEwzNiV5eAcRT1p1EC/dmEzaATVDJNiB3g2sRDWdA7b7MurXdJjB2+/WQktGWZwxvDrnuRFbWuIr64g==", + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/preact" + } + }, + "node_modules/preact-render-to-string": { + "version": "6.1.0", + "resolved": "https://registry.npmjs.org/preact-render-to-string/-/preact-render-to-string-6.1.0.tgz", + "integrity": "sha512-/AsKU4Q4R8r4aKwwNQrkQQNUVEDmTeZr6IwesDffobFRPcTk4dSQrfo1VOcXjtlcUss6QYEe7JShUGbQIhaw+A==", + "dependencies": { + "pretty-format": "^3.8.0" + }, + "peerDependencies": { + "preact": ">=10" + } + }, + "node_modules/prettier": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/prettier/-/prettier-3.0.0.tgz", + "integrity": "sha512-zBf5eHpwHOGPC47h0zrPyNn+eAEIdEzfywMoYn2XPi0P44Zp0tSq64rq0xAREh4auw2cJZHo9QUob+NqCQky4g==", + "dev": true, + "bin": { + "prettier": "bin/prettier.cjs" + }, + "engines": { + "node": ">=14" + }, + "funding": { + "url": "https://github.com/prettier/prettier?sponsor=1" + } + }, + "node_modules/pretty-bytes": { + "version": "6.1.0", + "resolved": "https://registry.npmjs.org/pretty-bytes/-/pretty-bytes-6.1.0.tgz", + "integrity": "sha512-Rk753HI8f4uivXi4ZCIYdhmG1V+WKzvRMg/X+M42a6t7D07RcmopXJMDNk6N++7Bl75URRGsb40ruvg7Hcp2wQ==", + "engines": { + "node": "^14.13.1 || >=16.0.0" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/pretty-format": { + "version": "3.8.0", + "resolved": "https://registry.npmjs.org/pretty-format/-/pretty-format-3.8.0.tgz", + "integrity": "sha512-WuxUnVtlWL1OfZFQFuqvnvs6MiAGk9UNsBostyBOB0Is9wb5uRESevA6rnl/rkksXaGX3GzZhPup5d6Vp1nFew==" + }, + "node_modules/pretty-time": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/pretty-time/-/pretty-time-1.1.0.tgz", + "integrity": "sha512-28iF6xPQrP8Oa6uxE6a1biz+lWeTOAPKggvjB8HAs6nVMKZwf5bG++632Dx614hIWgUPkgivRfG+a8uAXGTIbA==", + "engines": { + "node": ">=4" + } + }, + "node_modules/property-information": { + "version": "6.2.0", + "resolved": "https://registry.npmjs.org/property-information/-/property-information-6.2.0.tgz", + "integrity": "sha512-kma4U7AFCTwpqq5twzC1YVIDXSqg6qQK6JN0smOw8fgRy1OkMi0CYSzFmsy6dnqSenamAtj0CyXMUJ1Mf6oROg==", + "funding": { + "type": "github", + "url": "https://github.com/sponsors/wooorm" + } + }, + "node_modules/psl": { + "version": "1.9.0", + "resolved": "https://registry.npmjs.org/psl/-/psl-1.9.0.tgz", + "integrity": "sha512-E/ZsdU4HLs/68gYzgGTkMicWTLPdAftJLfJFlLUAAKZGkStNU72sZjT66SnMDVOfOWY/YAoiD7Jxa9iHvngcag==" + }, + "node_modules/punycode": { + "version": "1.4.1", + "resolved": "https://registry.npmjs.org/punycode/-/punycode-1.4.1.tgz", + "integrity": "sha512-jmYNElW7yvO7TV33CjSmvSiE2yco3bV2czu/OzDKdMNVZQWfxCblURLhf+47syQRBntjfLdd/H0egrzIG+oaFQ==" + }, + "node_modules/querystringify": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/querystringify/-/querystringify-2.2.0.tgz", + "integrity": "sha512-FIqgj2EUvTa7R50u0rGsyTftzjYmv/a3hO345bZNrqabNqjtgiDMgmo4mkUjd+nzU5oF3dClKqFIPUKybUyqoQ==" + }, + "node_modules/queue-microtask": { + "version": "1.2.3", + "resolved": "https://registry.npmjs.org/queue-microtask/-/queue-microtask-1.2.3.tgz", + "integrity": "sha512-NuaNSa6flKT5JaSYQzJok04JzTL1CA6aGhv5rfLW3PgqA+M2ChpZQnAC8h8i4ZFkBS8X5RqkDBHA7r4hej3K9A==", + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/feross" + }, + { + "type": "patreon", + "url": "https://www.patreon.com/feross" + }, + { + "type": "consulting", + "url": "https://feross.org/support" + } + ] + }, + "node_modules/range-parser": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/range-parser/-/range-parser-1.2.0.tgz", + "integrity": "sha512-kA5WQoNVo4t9lNx2kQNFCxKeBl5IbbSNBl1M/tLkw9WCn+hxNBAW5Qh8gdhs63CJnhjJ2zQWFoqPJP2sK1AV5A==", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/readdirp": { + "version": "3.6.0", + "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-3.6.0.tgz", + "integrity": "sha512-hOS089on8RduqdbhvQ5Z37A0ESjsqz6qnRcffsMU3495FuTdqSm+7bhJ29JvIOsBDEEnan5DPu9t3To9VRlMzA==", + "dependencies": { + "picomatch": "^2.2.1" + }, + "engines": { + "node": ">=8.10.0" + } + }, + "node_modules/reading-time": { + "version": "1.5.0", + "resolved": "https://registry.npmjs.org/reading-time/-/reading-time-1.5.0.tgz", + "integrity": "sha512-onYyVhBNr4CmAxFsKS7bz+uTLRakypIe4R+5A824vBSkQy/hB3fZepoVEf8OVAxzLvK+H/jm9TzpI3ETSm64Kg==" + }, + "node_modules/rehype-autolink-headings": { + "version": "6.1.1", + "resolved": "https://registry.npmjs.org/rehype-autolink-headings/-/rehype-autolink-headings-6.1.1.tgz", + "integrity": "sha512-NMYzZIsHM3sA14nC5rAFuUPIOfg+DFmf9EY1YMhaNlB7+3kK/ZlE6kqPfuxr1tsJ1XWkTrMtMoyHosU70d35mA==", + "dependencies": { + "@types/hast": "^2.0.0", + "extend": "^3.0.0", + "hast-util-has-property": "^2.0.0", + "hast-util-heading-rank": "^2.0.0", + "hast-util-is-element": "^2.0.0", + "unified": "^10.0.0", + "unist-util-visit": "^4.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/rehype-katex": { + "version": "6.0.3", + "resolved": "https://registry.npmjs.org/rehype-katex/-/rehype-katex-6.0.3.tgz", + "integrity": "sha512-ByZlRwRUcWegNbF70CVRm2h/7xy7jQ3R9LaY4VVSvjnoVWwWVhNL60DiZsBpC5tSzYQOCvDbzncIpIjPZWodZA==", + "dependencies": { + "@types/hast": "^2.0.0", + "@types/katex": "^0.14.0", + "hast-util-from-html-isomorphic": "^1.0.0", + "hast-util-to-text": "^3.1.0", + "katex": "^0.16.0", + "unist-util-visit": "^4.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/rehype-mathjax": { + "version": "4.0.3", + "resolved": "https://registry.npmjs.org/rehype-mathjax/-/rehype-mathjax-4.0.3.tgz", + "integrity": "sha512-QIwWH9U+r54nMQklVkT1qluxhKyzdPWz9dFwgel3BrseQsWZafRTDTUj8VR8/14nFuRIV2ChuCMz4zpACPoYvg==", + "dependencies": { + "@types/hast": "^2.0.0", + "@types/mathjax": "^0.0.37", + "hast-util-from-dom": "^4.0.0", + "hast-util-to-text": "^3.1.0", + "jsdom": "^20.0.0", + "mathjax-full": "^3.0.0", + "unified": "^10.0.0", + "unist-util-visit": "^4.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/rehype-pretty-code": { + "version": "0.10.0", + "resolved": "https://registry.npmjs.org/rehype-pretty-code/-/rehype-pretty-code-0.10.0.tgz", + "integrity": "sha512-qCD071Y+vUxEy9yyrATPk2+W9q7qCbzZgtc9suZhu75bmRQvOlBhJt4d3WvqSMTamkKoFkvqtCjyAk+ggH+aXQ==", + "dependencies": { + "@types/hast": "^2.0.0", + "hash-obj": "^4.0.0", + "parse-numeric-range": "^1.3.0" + }, + "engines": { + "node": ">=16" + }, + "peerDependencies": { + "shiki": "0.x" + } + }, + "node_modules/rehype-raw": { + "version": "6.1.1", + "resolved": "https://registry.npmjs.org/rehype-raw/-/rehype-raw-6.1.1.tgz", + "integrity": "sha512-d6AKtisSRtDRX4aSPsJGTfnzrX2ZkHQLE5kiUuGOeEoLpbEulFF4hj0mLPbsa+7vmguDKOVVEQdHKDSwoaIDsQ==", + "dependencies": { + "@types/hast": "^2.0.0", + "hast-util-raw": "^7.2.0", + "unified": "^10.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/rehype-slug": { + "version": "5.1.0", + "resolved": "https://registry.npmjs.org/rehype-slug/-/rehype-slug-5.1.0.tgz", + "integrity": "sha512-Gf91dJoXneiorNEnn+Phx97CO7oRMrpi+6r155tTxzGuLtm+QrI4cTwCa9e1rtePdL4i9tSO58PeSS6HWfgsiw==", + "dependencies": { + "@types/hast": "^2.0.0", + "github-slugger": "^2.0.0", + "hast-util-has-property": "^2.0.0", + "hast-util-heading-rank": "^2.0.0", + "hast-util-to-string": "^2.0.0", + "unified": "^10.0.0", + "unist-util-visit": "^4.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/remark": { + "version": "14.0.3", + "resolved": "https://registry.npmjs.org/remark/-/remark-14.0.3.tgz", + "integrity": "sha512-bfmJW1dmR2LvaMJuAnE88pZP9DktIFYXazkTfOIKZzi3Knk9lT0roItIA24ydOucI3bV/g/tXBA6hzqq3FV9Ew==", + "dependencies": { + "@types/mdast": "^3.0.0", + "remark-parse": "^10.0.0", + "remark-stringify": "^10.0.0", + "unified": "^10.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/remark-breaks": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/remark-breaks/-/remark-breaks-3.0.3.tgz", + "integrity": "sha512-C7VkvcUp1TPUc2eAYzsPdaUh8Xj4FSbQnYA5A9f80diApLZscTDeG7efiWP65W8hV2sEy3JuGVU0i6qr5D8Hug==", + "dependencies": { + "@types/mdast": "^3.0.0", + "mdast-util-newline-to-break": "^1.0.0", + "unified": "^10.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/remark-frontmatter": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/remark-frontmatter/-/remark-frontmatter-4.0.1.tgz", + "integrity": "sha512-38fJrB0KnmD3E33a5jZC/5+gGAC2WKNiPw1/fdXJvijBlhA7RCsvJklrYJakS0HedninvaCYW8lQGf9C918GfA==", + "dependencies": { + "@types/mdast": "^3.0.0", + "mdast-util-frontmatter": "^1.0.0", + "micromark-extension-frontmatter": "^1.0.0", + "unified": "^10.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/remark-gfm": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/remark-gfm/-/remark-gfm-3.0.1.tgz", + "integrity": "sha512-lEFDoi2PICJyNrACFOfDD3JlLkuSbOa5Wd8EPt06HUdptv8Gn0bxYTdbU/XXQ3swAPkEaGxxPN9cbnMHvVu1Ig==", + "dependencies": { + "@types/mdast": "^3.0.0", + "mdast-util-gfm": "^2.0.0", + "micromark-extension-gfm": "^2.0.0", + "unified": "^10.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/remark-math": { + "version": "5.1.1", + "resolved": "https://registry.npmjs.org/remark-math/-/remark-math-5.1.1.tgz", + "integrity": "sha512-cE5T2R/xLVtfFI4cCePtiRn+e6jKMtFDR3P8V3qpv8wpKjwvHoBA4eJzvX+nVrnlNy0911bdGmuspCSwetfYHw==", + "dependencies": { + "@types/mdast": "^3.0.0", + "mdast-util-math": "^2.0.0", + "micromark-extension-math": "^2.0.0", + "unified": "^10.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/remark-parse": { + "version": "10.0.2", + "resolved": "https://registry.npmjs.org/remark-parse/-/remark-parse-10.0.2.tgz", + "integrity": "sha512-3ydxgHa/ZQzG8LvC7jTXccARYDcRld3VfcgIIFs7bI6vbRSxJJmzgLEIIoYKyrfhaY+ujuWaf/PJiMZXoiCXgw==", + "dependencies": { + "@types/mdast": "^3.0.0", + "mdast-util-from-markdown": "^1.0.0", + "unified": "^10.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/remark-rehype": { + "version": "10.1.0", + "resolved": "https://registry.npmjs.org/remark-rehype/-/remark-rehype-10.1.0.tgz", + "integrity": "sha512-EFmR5zppdBp0WQeDVZ/b66CWJipB2q2VLNFMabzDSGR66Z2fQii83G5gTBbgGEnEEA0QRussvrFHxk1HWGJskw==", + "dependencies": { + "@types/hast": "^2.0.0", + "@types/mdast": "^3.0.0", + "mdast-util-to-hast": "^12.1.0", + "unified": "^10.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/remark-smartypants": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/remark-smartypants/-/remark-smartypants-2.0.0.tgz", + "integrity": "sha512-Rc0VDmr/yhnMQIz8n2ACYXlfw/P/XZev884QU1I5u+5DgJls32o97Vc1RbK3pfumLsJomS2yy8eT4Fxj/2MDVA==", + "dependencies": { + "retext": "^8.1.0", + "retext-smartypants": "^5.1.0", + "unist-util-visit": "^4.1.0" + }, + "engines": { + "node": "^12.20.0 || ^14.13.1 || >=16.0.0" + } + }, + "node_modules/remark-stringify": { + "version": "10.0.3", + "resolved": "https://registry.npmjs.org/remark-stringify/-/remark-stringify-10.0.3.tgz", + "integrity": "sha512-koyOzCMYoUHudypbj4XpnAKFbkddRMYZHwghnxd7ue5210WzGw6kOBwauJTRUMq16jsovXx8dYNvSSWP89kZ3A==", + "dependencies": { + "@types/mdast": "^3.0.0", + "mdast-util-to-markdown": "^1.0.0", + "unified": "^10.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/require-directory": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/require-directory/-/require-directory-2.1.1.tgz", + "integrity": "sha512-fGxEI7+wsG9xrvdjsrlmL22OMTTiHRwAMroiEeMgq8gzoLC/PQr7RsRDSTLUg/bZAZtF+TVIkHc6/4RIKrui+Q==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/requires-port": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/requires-port/-/requires-port-1.0.0.tgz", + "integrity": "sha512-KigOCHcocU3XODJxsu8i/j8T9tzT4adHiecwORRQ0ZZFcp7ahwXuRU1m+yuO90C5ZUyGeGfocHDI14M3L3yDAQ==" + }, + "node_modules/resolve": { + "version": "1.22.2", + "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.22.2.tgz", + "integrity": "sha512-Sb+mjNHOULsBv818T40qSPeRiuWLyaGMa5ewydRLFimneixmVy2zdivRl+AF6jaYPC8ERxGDmFSiqui6SfPd+g==", + "dependencies": { + "is-core-module": "^2.11.0", + "path-parse": "^1.0.7", + "supports-preserve-symlinks-flag": "^1.0.0" + }, + "bin": { + "resolve": "bin/resolve" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/resolve-pkg-maps": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/resolve-pkg-maps/-/resolve-pkg-maps-1.0.0.tgz", + "integrity": "sha512-seS2Tj26TBVOC2NIc2rOe2y2ZO7efxITtLZcGSOnHHNOQ7CkiUBfw0Iw2ck6xkIhPwLhKNLS8BO+hEpngQlqzw==", + "dev": true, + "funding": { + "url": "https://github.com/privatenumber/resolve-pkg-maps?sponsor=1" + } + }, + "node_modules/retext": { + "version": "8.1.0", + "resolved": "https://registry.npmjs.org/retext/-/retext-8.1.0.tgz", + "integrity": "sha512-N9/Kq7YTn6ZpzfiGW45WfEGJqFf1IM1q8OsRa1CGzIebCJBNCANDRmOrholiDRGKo/We7ofKR4SEvcGAWEMD3Q==", + "dependencies": { + "@types/nlcst": "^1.0.0", + "retext-latin": "^3.0.0", + "retext-stringify": "^3.0.0", + "unified": "^10.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/retext-latin": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/retext-latin/-/retext-latin-3.1.0.tgz", + "integrity": "sha512-5MrD1tuebzO8ppsja5eEu+ZbBeUNCjoEarn70tkXOS7Bdsdf6tNahsv2bY0Z8VooFF6cw7/6S+d3yI/TMlMVVQ==", + "dependencies": { + "@types/nlcst": "^1.0.0", + "parse-latin": "^5.0.0", + "unherit": "^3.0.0", + "unified": "^10.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/retext-smartypants": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/retext-smartypants/-/retext-smartypants-5.2.0.tgz", + "integrity": "sha512-Do8oM+SsjrbzT2UNIKgheP0hgUQTDDQYyZaIY3kfq0pdFzoPk+ZClYJ+OERNXveog4xf1pZL4PfRxNoVL7a/jw==", + "dependencies": { + "@types/nlcst": "^1.0.0", + "nlcst-to-string": "^3.0.0", + "unified": "^10.0.0", + "unist-util-visit": "^4.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/retext-stringify": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/retext-stringify/-/retext-stringify-3.1.0.tgz", + "integrity": "sha512-767TLOaoXFXyOnjx/EggXlb37ZD2u4P1n0GJqVdpipqACsQP+20W+BNpMYrlJkq7hxffnFk+jc6mAK9qrbuB8w==", + "dependencies": { + "@types/nlcst": "^1.0.0", + "nlcst-to-string": "^3.0.0", + "unified": "^10.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/reusify": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/reusify/-/reusify-1.0.4.tgz", + "integrity": "sha512-U9nH88a3fc/ekCF1l0/UP1IosiuIjyTh7hBvXVMHYgVcfGvt897Xguj2UOLDeI5BG2m7/uwyaLVT6fbtCwTyzw==", + "engines": { + "iojs": ">=1.0.0", + "node": ">=0.10.0" + } + }, + "node_modules/rimraf": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-5.0.1.tgz", + "integrity": "sha512-OfFZdwtd3lZ+XZzYP/6gTACubwFcHdLRqS9UX3UwpU2dnGQYkPFISRwvM3w9IiB2w7bW5qGo/uAwE4SmXXSKvg==", + "dependencies": { + "glob": "^10.2.5" + }, + "bin": { + "rimraf": "dist/cjs/src/bin.js" + }, + "engines": { + "node": ">=14" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/robust-predicates": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/robust-predicates/-/robust-predicates-3.0.2.tgz", + "integrity": "sha512-IXgzBWvWQwE6PrDI05OvmXUIruQTcoMDzRsOd5CDvHCVLcLHMTSYvOK5Cm46kWqlV3yAbuSpBZdJ5oP5OUoStg==" + }, + "node_modules/run-parallel": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/run-parallel/-/run-parallel-1.2.0.tgz", + "integrity": "sha512-5l4VyZR86LZ/lDxZTR6jqL8AFE2S0IFLMP26AbjsLVADxHdhB/c0GUsH+y39UfCi3dzz8OlQuPmnaJOMoDHQBA==", + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/feross" + }, + { + "type": "patreon", + "url": "https://www.patreon.com/feross" + }, + { + "type": "consulting", + "url": "https://feross.org/support" + } + ], + "dependencies": { + "queue-microtask": "^1.2.2" + } + }, + "node_modules/rw": { + "version": "1.3.3", + "resolved": "https://registry.npmjs.org/rw/-/rw-1.3.3.tgz", + "integrity": "sha512-PdhdWy89SiZogBLaw42zdeqtRJ//zFd2PgQavcICDUgJT5oW10QCRKbJ6bg4r0/UY2M6BWd5tkxuGFRvCkgfHQ==" + }, + "node_modules/sade": { + "version": "1.8.1", + "resolved": "https://registry.npmjs.org/sade/-/sade-1.8.1.tgz", + "integrity": "sha512-xal3CZX1Xlo/k4ApwCFrHVACi9fBqJ7V+mwhBsuf/1IOKbBy098Fex+Wa/5QMubw09pSZ/u8EY8PWgevJsXp1A==", + "dependencies": { + "mri": "^1.1.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/safer-buffer": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz", + "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==" + }, + "node_modules/sass": { + "version": "1.66.1", + "resolved": "https://registry.npmjs.org/sass/-/sass-1.66.1.tgz", + "integrity": "sha512-50c+zTsZOJVgFfTgwwEzkjA3/QACgdNsKueWPyAR0mRINIvLAStVQBbPg14iuqEQ74NPDbXzJARJ/O4SI1zftA==", + "dependencies": { + "chokidar": ">=3.0.0 <4.0.0", + "immutable": "^4.0.0", + "source-map-js": ">=0.6.2 <2.0.0" + }, + "bin": { + "sass": "sass.js" + }, + "engines": { + "node": ">=14.0.0" + } + }, + "node_modules/saxes": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/saxes/-/saxes-6.0.0.tgz", + "integrity": "sha512-xAg7SOnEhrm5zI3puOOKyy1OMcMlIJZYNJY7xLBwSze0UjhPLnWfj2GF2EpT0jmzaJKIWKHLsaSSajf35bcYnA==", + "dependencies": { + "xmlchars": "^2.2.0" + }, + "engines": { + "node": ">=v12.22.7" + } + }, + "node_modules/section-matter": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/section-matter/-/section-matter-1.0.0.tgz", + "integrity": "sha512-vfD3pmTzGpufjScBh50YHKzEu2lxBWhVEHsNGoEXmCmn2hKGfeNLYMzCJpe8cD7gqX7TJluOVpBkAequ6dgMmA==", + "dependencies": { + "extend-shallow": "^2.0.1", + "kind-of": "^6.0.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/serve-handler": { + "version": "6.1.5", + "resolved": "https://registry.npmjs.org/serve-handler/-/serve-handler-6.1.5.tgz", + "integrity": "sha512-ijPFle6Hwe8zfmBxJdE+5fta53fdIY0lHISJvuikXB3VYFafRjMRpOffSPvCYsbKyBA7pvy9oYr/BT1O3EArlg==", + "dependencies": { + "bytes": "3.0.0", + "content-disposition": "0.5.2", + "fast-url-parser": "1.1.3", + "mime-types": "2.1.18", + "minimatch": "3.1.2", + "path-is-inside": "1.0.2", + "path-to-regexp": "2.2.1", + "range-parser": "1.2.0" + } + }, + "node_modules/serve-handler/node_modules/brace-expansion": { + "version": "1.1.11", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz", + "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==", + "dependencies": { + "balanced-match": "^1.0.0", + "concat-map": "0.0.1" + } + }, + "node_modules/serve-handler/node_modules/minimatch": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", + "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", + "dependencies": { + "brace-expansion": "^1.1.7" + }, + "engines": { + "node": "*" + } + }, + "node_modules/shebang-command": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-2.0.0.tgz", + "integrity": "sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA==", + "dependencies": { + "shebang-regex": "^3.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/shebang-regex": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-3.0.0.tgz", + "integrity": "sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==", + "engines": { + "node": ">=8" + } + }, + "node_modules/shiki": { + "version": "0.14.3", + "resolved": "https://registry.npmjs.org/shiki/-/shiki-0.14.3.tgz", + "integrity": "sha512-U3S/a+b0KS+UkTyMjoNojvTgrBHjgp7L6ovhFVZsXmBGnVdQ4K4U9oK0z63w538S91ATngv1vXigHCSWOwnr+g==", + "peer": true, + "dependencies": { + "ansi-sequence-parser": "^1.1.0", + "jsonc-parser": "^3.2.0", + "vscode-oniguruma": "^1.7.0", + "vscode-textmate": "^8.0.0" + } + }, + "node_modules/signal-exit": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-4.0.2.tgz", + "integrity": "sha512-MY2/qGx4enyjprQnFaZsHib3Yadh3IXyV2C321GY0pjGfVBu4un0uDJkwgdxqO+Rdx8JMT8IfJIRwbYVz3Ob3Q==", + "engines": { + "node": ">=14" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/sisteransi": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/sisteransi/-/sisteransi-1.0.5.tgz", + "integrity": "sha512-bLGGlR1QxBcynn2d5YmDX4MGjlZvy2MRBDRNHLJ8VI6l6+9FUiyTFNJ0IveOSP0bcXgVDPRcfGqA0pjaqUpfVg==" + }, + "node_modules/slash": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/slash/-/slash-4.0.0.tgz", + "integrity": "sha512-3dOsAHXXUkQTpOYcoAxLIorMTp4gIQr5IW3iVb7A7lFIp0VHhnynm9izx6TssdrIcVIESAlVjtnO2K8bg+Coew==", + "engines": { + "node": ">=12" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/sort-keys": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/sort-keys/-/sort-keys-5.0.0.tgz", + "integrity": "sha512-Pdz01AvCAottHTPQGzndktFNdbRA75BgOfeT1hH+AMnJFv8lynkPi42rfeEhpx1saTEI3YNMWxfqu0sFD1G8pw==", + "dependencies": { + "is-plain-obj": "^4.0.0" + }, + "engines": { + "node": ">=12" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/source-map-js": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/source-map-js/-/source-map-js-1.0.2.tgz", + "integrity": "sha512-R0XvVJ9WusLiqTCEiGCmICCMplcCkIwwR11mOSD9CR5u+IXYdiseeEuXCVAjS54zqwkLcPNnmU4OeJ6tUrWhDw==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/source-map-support": { + "version": "0.5.21", + "resolved": "https://registry.npmjs.org/source-map-support/-/source-map-support-0.5.21.tgz", + "integrity": "sha512-uBHU3L3czsIyYXKX88fdrGovxdSCoTGDRZ6SYXtSRxLZUzHg5P/66Ht6uoUlHu9EZod+inXhKo3qQgwXUT/y1w==", + "dependencies": { + "buffer-from": "^1.0.0", + "source-map": "^0.6.0" + } + }, + "node_modules/space-separated-tokens": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/space-separated-tokens/-/space-separated-tokens-2.0.2.tgz", + "integrity": "sha512-PEGlAwrG8yXGXRjW32fGbg66JAlOAwbObuqVoJpv/mRgoWDQfgH1wDPvtzWyUSNAXBGSk8h755YDbbcEy3SH2Q==", + "funding": { + "type": "github", + "url": "https://github.com/sponsors/wooorm" + } + }, + "node_modules/speech-rule-engine": { + "version": "4.0.7", + "resolved": "https://registry.npmjs.org/speech-rule-engine/-/speech-rule-engine-4.0.7.tgz", + "integrity": "sha512-sJrL3/wHzNwJRLBdf6CjJWIlxC04iYKkyXvYSVsWVOiC2DSkHmxsqOhEeMsBA9XK+CHuNcsdkbFDnoUfAsmp9g==", + "dependencies": { + "commander": "9.2.0", + "wicked-good-xpath": "1.3.0", + "xmldom-sre": "0.1.31" + }, + "bin": { + "sre": "bin/sre" + } + }, + "node_modules/speech-rule-engine/node_modules/commander": { + "version": "9.2.0", + "resolved": "https://registry.npmjs.org/commander/-/commander-9.2.0.tgz", + "integrity": "sha512-e2i4wANQiSXgnrBlIatyHtP1odfUp0BbV5Y5nEGbxtIrStkEOAAzCUirvLBNXHLr7kwLvJl6V+4V3XV9x7Wd9w==", + "engines": { + "node": "^12.20.0 || >=14" + } + }, + "node_modules/sprintf-js": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/sprintf-js/-/sprintf-js-1.0.3.tgz", + "integrity": "sha512-D9cPgkvLlV3t3IzL0D0YLvGA9Ahk4PcvVwUbN0dSGr1aP0Nrt4AEnTUbuGvquEC0mA64Gqt1fzirlRs5ibXx8g==" + }, + "node_modules/string-width": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-5.1.2.tgz", + "integrity": "sha512-HnLOCR3vjcY8beoNLtcjZ5/nxn2afmME6lhrDrebokqMap+XbeW8n9TXpPDOqdGK5qcI3oT0GKTW6wC7EMiVqA==", + "dependencies": { + "eastasianwidth": "^0.2.0", + "emoji-regex": "^9.2.2", + "strip-ansi": "^7.0.1" + }, + "engines": { + "node": ">=12" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/string-width-cjs": { + "name": "string-width", + "version": "4.2.3", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.3.tgz", + "integrity": "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==", + "dependencies": { + "emoji-regex": "^8.0.0", + "is-fullwidth-code-point": "^3.0.0", + "strip-ansi": "^6.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/string-width-cjs/node_modules/ansi-regex": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz", + "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==", + "engines": { + "node": ">=8" + } + }, + "node_modules/string-width-cjs/node_modules/emoji-regex": { + "version": "8.0.0", + "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz", + "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==" + }, + "node_modules/string-width-cjs/node_modules/strip-ansi": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", + "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", + "dependencies": { + "ansi-regex": "^5.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/stringify-entities": { + "version": "4.0.3", + "resolved": "https://registry.npmjs.org/stringify-entities/-/stringify-entities-4.0.3.tgz", + "integrity": "sha512-BP9nNHMhhfcMbiuQKCqMjhDP5yBCAxsPu4pHFFzJ6Alo9dZgY4VLDPutXqIjpRiMoKdp7Av85Gr73Q5uH9k7+g==", + "dependencies": { + "character-entities-html4": "^2.0.0", + "character-entities-legacy": "^3.0.0" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/wooorm" + } + }, + "node_modules/strip-ansi": { + "version": "7.1.0", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-7.1.0.tgz", + "integrity": "sha512-iq6eVVI64nQQTRYq2KtEg2d2uU7LElhTJwsH4YzIHZshxlgZms/wIc4VoDQTlG/IvVIrBKG06CrZnp0qv7hkcQ==", + "dependencies": { + "ansi-regex": "^6.0.1" + }, + "engines": { + "node": ">=12" + }, + "funding": { + "url": "https://github.com/chalk/strip-ansi?sponsor=1" + } + }, + "node_modules/strip-ansi-cjs": { + "name": "strip-ansi", + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", + "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", + "dependencies": { + "ansi-regex": "^5.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/strip-ansi-cjs/node_modules/ansi-regex": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz", + "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==", + "engines": { + "node": ">=8" + } + }, + "node_modules/strip-bom-string": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/strip-bom-string/-/strip-bom-string-1.0.0.tgz", + "integrity": "sha512-uCC2VHvQRYu+lMh4My/sFNmF2klFymLX1wHJeXnbEJERpV/ZsVuonzerjfrGpIGF7LBVa1O7i9kjiWvJiFck8g==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/style-to-object": { + "version": "0.4.1", + "resolved": "https://registry.npmjs.org/style-to-object/-/style-to-object-0.4.1.tgz", + "integrity": "sha512-HFpbb5gr2ypci7Qw+IOhnP2zOU7e77b+rzM+wTzXzfi1PrtBCX0E7Pk4wL4iTLnhzZ+JgEGAhX81ebTg/aYjQw==", + "dependencies": { + "inline-style-parser": "0.1.1" + } + }, + "node_modules/supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dependencies": { + "has-flag": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/supports-preserve-symlinks-flag": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/supports-preserve-symlinks-flag/-/supports-preserve-symlinks-flag-1.0.0.tgz", + "integrity": "sha512-ot0WnXS9fgdkgIcePe6RHNk1WA8+muPa6cSjeR3V8K27q9BB1rTE3R1p7Hv0z1ZyAc8s6Vvv8DIyWf681MAt0w==", + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/symbol-tree": { + "version": "3.2.4", + "resolved": "https://registry.npmjs.org/symbol-tree/-/symbol-tree-3.2.4.tgz", + "integrity": "sha512-9QNk5KwDF+Bvz+PyObkmSYjI5ksVUYtjW7AU22r2NKcfLJcXp96hkDWU3+XndOsUb+AQ9QhfzfCT2O+CNWT5Tw==" + }, + "node_modules/to-regex-range": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz", + "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==", + "dependencies": { + "is-number": "^7.0.0" + }, + "engines": { + "node": ">=8.0" + } + }, + "node_modules/to-vfile": { + "version": "7.2.4", + "resolved": "https://registry.npmjs.org/to-vfile/-/to-vfile-7.2.4.tgz", + "integrity": "sha512-2eQ+rJ2qGbyw3senPI0qjuM7aut8IYXK6AEoOWb+fJx/mQYzviTckm1wDjq91QYHAPBTYzmdJXxMFA6Mk14mdw==", + "dependencies": { + "is-buffer": "^2.0.0", + "vfile": "^5.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/toml": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/toml/-/toml-3.0.0.tgz", + "integrity": "sha512-y/mWCZinnvxjTKYhJ+pYxwD0mRLVvOtdS2Awbgxln6iEnt4rk0yBxeSBHkGJcPucRiG0e55mwWp+g/05rsrd6w==" + }, + "node_modules/tough-cookie": { + "version": "4.1.3", + "resolved": "https://registry.npmjs.org/tough-cookie/-/tough-cookie-4.1.3.tgz", + "integrity": "sha512-aX/y5pVRkfRnfmuX+OdbSdXvPe6ieKX/G2s7e98f4poJHnqH3281gDPm/metm6E/WRamfx7WC4HUqkWHfQHprw==", + "dependencies": { + "psl": "^1.1.33", + "punycode": "^2.1.1", + "universalify": "^0.2.0", + "url-parse": "^1.5.3" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/tough-cookie/node_modules/punycode": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/punycode/-/punycode-2.3.0.tgz", + "integrity": "sha512-rRV+zQD8tVFys26lAGR9WUuS4iUAngJScM+ZRSKtvl5tKeZ2t5bvdNFdNHBW9FWR4guGHlgmsZ1G7BSm2wTbuA==", + "engines": { + "node": ">=6" + } + }, + "node_modules/tr46": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/tr46/-/tr46-3.0.0.tgz", + "integrity": "sha512-l7FvfAHlcmulp8kr+flpQZmVwtu7nfRV7NZujtN0OqES8EL4O4e0qqzL0DC5gAvx/ZC/9lk6rhcUwYvkBnBnYA==", + "dependencies": { + "punycode": "^2.1.1" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/tr46/node_modules/punycode": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/punycode/-/punycode-2.3.0.tgz", + "integrity": "sha512-rRV+zQD8tVFys26lAGR9WUuS4iUAngJScM+ZRSKtvl5tKeZ2t5bvdNFdNHBW9FWR4guGHlgmsZ1G7BSm2wTbuA==", + "engines": { + "node": ">=6" + } + }, + "node_modules/trim-lines": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/trim-lines/-/trim-lines-3.0.1.tgz", + "integrity": "sha512-kRj8B+YHZCc9kQYdWfJB2/oUl9rA99qbowYYBtr4ui4mZyAQ2JpvVBd/6U2YloATfqBhBTSMhTpgBHtU0Mf3Rg==", + "funding": { + "type": "github", + "url": "https://github.com/sponsors/wooorm" + } + }, + "node_modules/trough": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/trough/-/trough-2.1.0.tgz", + "integrity": "sha512-AqTiAOLcj85xS7vQ8QkAV41hPDIJ71XJB4RCUrzo/1GM2CQwhkJGaf9Hgr7BOugMRpgGUrqRg/DrBDl4H40+8g==", + "funding": { + "type": "github", + "url": "https://github.com/sponsors/wooorm" + } + }, + "node_modules/tslib": { + "version": "2.6.2", + "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.6.2.tgz", + "integrity": "sha512-AEYxH93jGFPn/a2iVAwW87VuUIkR1FVUKB77NwMF7nBTDkDrrT/Hpt/IrCJ0QXhW27jTBDcf5ZY7w6RiqTMw2Q==" + }, + "node_modules/tsx": { + "version": "3.12.7", + "resolved": "https://registry.npmjs.org/tsx/-/tsx-3.12.7.tgz", + "integrity": "sha512-C2Ip+jPmqKd1GWVQDvz/Eyc6QJbGfE7NrR3fx5BpEHMZsEHoIxHL1j+lKdGobr8ovEyqeNkPLSKp6SCSOt7gmw==", + "dev": true, + "dependencies": { + "@esbuild-kit/cjs-loader": "^2.4.2", + "@esbuild-kit/core-utils": "^3.0.0", + "@esbuild-kit/esm-loader": "^2.5.5" + }, + "bin": { + "tsx": "dist/cli.js" + }, + "optionalDependencies": { + "fsevents": "~2.3.2" + } + }, + "node_modules/type-fest": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-1.4.0.tgz", + "integrity": "sha512-yGSza74xk0UG8k+pLh5oeoYirvIiWo5t0/o3zHHAO2tRDiZcxWP7fywNlXhqb6/r6sWvwi+RsyQMWhVLe4BVuA==", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/typescript": { + "version": "5.1.6", + "resolved": "https://registry.npmjs.org/typescript/-/typescript-5.1.6.tgz", + "integrity": "sha512-zaWCozRZ6DLEWAWFrVDz1H6FVXzUSfTy5FUMWsQlU8Ym5JP9eO4xkTIROFCQvhQf61z6O/G6ugw3SgAnvvm+HA==", + "dev": true, + "bin": { + "tsc": "bin/tsc", + "tsserver": "bin/tsserver" + }, + "engines": { + "node": ">=14.17" + } + }, + "node_modules/unherit": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/unherit/-/unherit-3.0.1.tgz", + "integrity": "sha512-akOOQ/Yln8a2sgcLj4U0Jmx0R5jpIg2IUyRrWOzmEbjBtGzBdHtSeFKgoEcoH4KYIG/Pb8GQ/BwtYm0GCq1Sqg==", + "funding": { + "type": "github", + "url": "https://github.com/sponsors/wooorm" + } + }, + "node_modules/unified": { + "version": "10.1.2", + "resolved": "https://registry.npmjs.org/unified/-/unified-10.1.2.tgz", + "integrity": "sha512-pUSWAi/RAnVy1Pif2kAoeWNBa3JVrx0MId2LASj8G+7AiHWoKZNTomq6LG326T68U7/e263X6fTdcXIy7XnF7Q==", + "dependencies": { + "@types/unist": "^2.0.0", + "bail": "^2.0.0", + "extend": "^3.0.0", + "is-buffer": "^2.0.0", + "is-plain-obj": "^4.0.0", + "trough": "^2.0.0", + "vfile": "^5.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/unist-util-find-after": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/unist-util-find-after/-/unist-util-find-after-4.0.1.tgz", + "integrity": "sha512-QO/PuPMm2ERxC6vFXEPtmAutOopy5PknD+Oq64gGwxKtk4xwo9Z97t9Av1obPmGU0IyTa6EKYUfTrK2QJS3Ozw==", + "dependencies": { + "@types/unist": "^2.0.0", + "unist-util-is": "^5.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/unist-util-generated": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/unist-util-generated/-/unist-util-generated-2.0.1.tgz", + "integrity": "sha512-qF72kLmPxAw0oN2fwpWIqbXAVyEqUzDHMsbtPvOudIlUzXYFIeQIuxXQCRCFh22B7cixvU0MG7m3MW8FTq/S+A==", + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/unist-util-is": { + "version": "5.2.1", + "resolved": "https://registry.npmjs.org/unist-util-is/-/unist-util-is-5.2.1.tgz", + "integrity": "sha512-u9njyyfEh43npf1M+yGKDGVPbY/JWEemg5nH05ncKPfi+kBbKBJoTdsogMu33uhytuLlv9y0O7GH7fEdwLdLQw==", + "dependencies": { + "@types/unist": "^2.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/unist-util-modify-children": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/unist-util-modify-children/-/unist-util-modify-children-3.1.1.tgz", + "integrity": "sha512-yXi4Lm+TG5VG+qvokP6tpnk+r1EPwyYL04JWDxLvgvPV40jANh7nm3udk65OOWquvbMDe+PL9+LmkxDpTv/7BA==", + "dependencies": { + "@types/unist": "^2.0.0", + "array-iterate": "^2.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/unist-util-position": { + "version": "4.0.4", + "resolved": "https://registry.npmjs.org/unist-util-position/-/unist-util-position-4.0.4.tgz", + "integrity": "sha512-kUBE91efOWfIVBo8xzh/uZQ7p9ffYRtUbMRZBNFYwf0RK8koUMx6dGUfwylLOKmaT2cs4wSW96QoYUSXAyEtpg==", + "dependencies": { + "@types/unist": "^2.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/unist-util-remove-position": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/unist-util-remove-position/-/unist-util-remove-position-4.0.2.tgz", + "integrity": "sha512-TkBb0HABNmxzAcfLf4qsIbFbaPDvMO6wa3b3j4VcEzFVaw1LBKwnW4/sRJ/atSLSzoIg41JWEdnE7N6DIhGDGQ==", + "dependencies": { + "@types/unist": "^2.0.0", + "unist-util-visit": "^4.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/unist-util-stringify-position": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/unist-util-stringify-position/-/unist-util-stringify-position-3.0.3.tgz", + "integrity": "sha512-k5GzIBZ/QatR8N5X2y+drfpWG8IDBzdnVj6OInRNWm1oXrzydiaAT2OQiA8DPRRZyAKb9b6I2a6PxYklZD0gKg==", + "dependencies": { + "@types/unist": "^2.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/unist-util-visit": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/unist-util-visit/-/unist-util-visit-4.1.2.tgz", + "integrity": "sha512-MSd8OUGISqHdVvfY9TPhyK2VdUrPgxkUtWSuMHF6XAAFuL4LokseigBnZtPnJMu+FbynTkFNnFlyjxpVKujMRg==", + "dependencies": { + "@types/unist": "^2.0.0", + "unist-util-is": "^5.0.0", + "unist-util-visit-parents": "^5.1.1" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/unist-util-visit-children": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/unist-util-visit-children/-/unist-util-visit-children-2.0.2.tgz", + "integrity": "sha512-+LWpMFqyUwLGpsQxpumsQ9o9DG2VGLFrpz+rpVXYIEdPy57GSy5HioC0g3bg/8WP9oCLlapQtklOzQ8uLS496Q==", + "dependencies": { + "@types/unist": "^2.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/unist-util-visit-parents": { + "version": "5.1.3", + "resolved": "https://registry.npmjs.org/unist-util-visit-parents/-/unist-util-visit-parents-5.1.3.tgz", + "integrity": "sha512-x6+y8g7wWMyQhL1iZfhIPhDAs7Xwbn9nRosDXl7qoPTSCy0yNxnKc+hWokFifWQIDGi154rdUqKvbCa4+1kLhg==", + "dependencies": { + "@types/unist": "^2.0.0", + "unist-util-is": "^5.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/universalify": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/universalify/-/universalify-0.2.0.tgz", + "integrity": "sha512-CJ1QgKmNg3CwvAv/kOFmtnEN05f0D/cn9QntgNOQlQF9dgvVTHj3t+8JPdjqawCHk7V/KA+fbUqzZ9XWhcqPUg==", + "engines": { + "node": ">= 4.0.0" + } + }, + "node_modules/url-parse": { + "version": "1.5.10", + "resolved": "https://registry.npmjs.org/url-parse/-/url-parse-1.5.10.tgz", + "integrity": "sha512-WypcfiRhfeUP9vvF0j6rw0J3hrWrw6iZv3+22h6iRMJ/8z1Tj6XfLP4DsUix5MhMPnXpiHDoKyoZ/bdCkwBCiQ==", + "dependencies": { + "querystringify": "^2.1.1", + "requires-port": "^1.0.0" + } + }, + "node_modules/uvu": { + "version": "0.5.6", + "resolved": "https://registry.npmjs.org/uvu/-/uvu-0.5.6.tgz", + "integrity": "sha512-+g8ENReyr8YsOc6fv/NVJs2vFdHBnBNdfE49rshrTzDWOlUx4Gq7KOS2GD8eqhy2j+Ejq29+SbKH8yjkAqXqoA==", + "dependencies": { + "dequal": "^2.0.0", + "diff": "^5.0.0", + "kleur": "^4.0.3", + "sade": "^1.7.3" + }, + "bin": { + "uvu": "bin.js" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/vfile": { + "version": "5.3.7", + "resolved": "https://registry.npmjs.org/vfile/-/vfile-5.3.7.tgz", + "integrity": "sha512-r7qlzkgErKjobAmyNIkkSpizsFPYiUPuJb5pNW1RB4JcYVZhs4lIbVqk8XPk033CV/1z8ss5pkax8SuhGpcG8g==", + "dependencies": { + "@types/unist": "^2.0.0", + "is-buffer": "^2.0.0", + "unist-util-stringify-position": "^3.0.0", + "vfile-message": "^3.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/vfile-location": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/vfile-location/-/vfile-location-4.1.0.tgz", + "integrity": "sha512-YF23YMyASIIJXpktBa4vIGLJ5Gs88UB/XePgqPmTa7cDA+JeO3yclbpheQYCHjVHBn/yePzrXuygIL+xbvRYHw==", + "dependencies": { + "@types/unist": "^2.0.0", + "vfile": "^5.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/vfile-message": { + "version": "3.1.4", + "resolved": "https://registry.npmjs.org/vfile-message/-/vfile-message-3.1.4.tgz", + "integrity": "sha512-fa0Z6P8HUrQN4BZaX05SIVXic+7kE3b05PWAtPuYP9QLHsLKYR7/AlLW3NtOrpXRLeawpDLMsVkmk5DG0NXgWw==", + "dependencies": { + "@types/unist": "^2.0.0", + "unist-util-stringify-position": "^3.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/vscode-oniguruma": { + "version": "1.7.0", + "resolved": "https://registry.npmjs.org/vscode-oniguruma/-/vscode-oniguruma-1.7.0.tgz", + "integrity": "sha512-L9WMGRfrjOhgHSdOYgCt/yRMsXzLDJSL7BPrOZt73gU0iWO4mpqzqQzOz5srxqTvMBaR0XZTSrVWo4j55Rc6cA==", + "peer": true + }, + "node_modules/vscode-textmate": { + "version": "8.0.0", + "resolved": "https://registry.npmjs.org/vscode-textmate/-/vscode-textmate-8.0.0.tgz", + "integrity": "sha512-AFbieoL7a5LMqcnOF04ji+rpXadgOXnZsxQr//r83kLPr7biP7am3g9zbaZIaBGwBRWeSvoMD4mgPdX3e4NWBg==", + "peer": true + }, + "node_modules/w3c-xmlserializer": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/w3c-xmlserializer/-/w3c-xmlserializer-4.0.0.tgz", + "integrity": "sha512-d+BFHzbiCx6zGfz0HyQ6Rg69w9k19nviJspaj4yNscGjrHu94sVP+aRm75yEbCh+r2/yR+7q6hux9LVtbuTGBw==", + "dependencies": { + "xml-name-validator": "^4.0.0" + }, + "engines": { + "node": ">=14" + } + }, + "node_modules/web-namespaces": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/web-namespaces/-/web-namespaces-2.0.1.tgz", + "integrity": "sha512-bKr1DkiNa2krS7qxNtdrtHAmzuYGFQLiQ13TsorsdT6ULTkPLKuu5+GsFpDlg6JFjUTwX2DyhMPG2be8uPrqsQ==", + "funding": { + "type": "github", + "url": "https://github.com/sponsors/wooorm" + } + }, + "node_modules/webidl-conversions": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/webidl-conversions/-/webidl-conversions-7.0.0.tgz", + "integrity": "sha512-VwddBukDzu71offAQR975unBIGqfKZpM+8ZX6ySk8nYhVoo5CYaZyzt3YBvYtRtO+aoGlqxPg/B87NGVZ/fu6g==", + "engines": { + "node": ">=12" + } + }, + "node_modules/whatwg-encoding": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/whatwg-encoding/-/whatwg-encoding-2.0.0.tgz", + "integrity": "sha512-p41ogyeMUrw3jWclHWTQg1k05DSVXPLcVxRTYsXUk+ZooOCZLcoYgPZ/HL/D/N+uQPOtcp1me1WhBEaX02mhWg==", + "dependencies": { + "iconv-lite": "0.6.3" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/whatwg-mimetype": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/whatwg-mimetype/-/whatwg-mimetype-3.0.0.tgz", + "integrity": "sha512-nt+N2dzIutVRxARx1nghPKGv1xHikU7HKdfafKkLNLindmPU/ch3U31NOCGGA/dmPcmb1VlofO0vnKAcsm0o/Q==", + "engines": { + "node": ">=12" + } + }, + "node_modules/whatwg-url": { + "version": "11.0.0", + "resolved": "https://registry.npmjs.org/whatwg-url/-/whatwg-url-11.0.0.tgz", + "integrity": "sha512-RKT8HExMpoYx4igMiVMY83lN6UeITKJlBQ+vR/8ZJ8OCdSiN3RwCq+9gH0+Xzj0+5IrM6i4j/6LuvzbZIQgEcQ==", + "dependencies": { + "tr46": "^3.0.0", + "webidl-conversions": "^7.0.0" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/which": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz", + "integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==", + "dependencies": { + "isexe": "^2.0.0" + }, + "bin": { + "node-which": "bin/node-which" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/wicked-good-xpath": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/wicked-good-xpath/-/wicked-good-xpath-1.3.0.tgz", + "integrity": "sha512-Gd9+TUn5nXdwj/hFsPVx5cuHHiF5Bwuc30jZ4+ronF1qHK5O7HD0sgmXWSEgwKquT3ClLoKPVbO6qGwVwLzvAw==" + }, + "node_modules/workerpool": { + "version": "6.4.0", + "resolved": "https://registry.npmjs.org/workerpool/-/workerpool-6.4.0.tgz", + "integrity": "sha512-i3KR1mQMNwY2wx20ozq2EjISGtQWDIfV56We+yGJ5yDs8jTwQiLLaqHlkBHITlCuJnYlVRmXegxFxZg7gqI++A==" + }, + "node_modules/wrap-ansi": { + "version": "8.1.0", + "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-8.1.0.tgz", + "integrity": "sha512-si7QWI6zUMq56bESFvagtmzMdGOtoxfR+Sez11Mobfc7tm+VkUckk9bW2UeffTGVUbOksxmSw0AA2gs8g71NCQ==", + "dependencies": { + "ansi-styles": "^6.1.0", + "string-width": "^5.0.1", + "strip-ansi": "^7.0.1" + }, + "engines": { + "node": ">=12" + }, + "funding": { + "url": "https://github.com/chalk/wrap-ansi?sponsor=1" + } + }, + "node_modules/wrap-ansi-cjs": { + "name": "wrap-ansi", + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-7.0.0.tgz", + "integrity": "sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q==", + "dependencies": { + "ansi-styles": "^4.0.0", + "string-width": "^4.1.0", + "strip-ansi": "^6.0.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/wrap-ansi?sponsor=1" + } + }, + "node_modules/wrap-ansi-cjs/node_modules/ansi-regex": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz", + "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==", + "engines": { + "node": ">=8" + } + }, + "node_modules/wrap-ansi-cjs/node_modules/emoji-regex": { + "version": "8.0.0", + "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz", + "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==" + }, + "node_modules/wrap-ansi-cjs/node_modules/string-width": { + "version": "4.2.3", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.3.tgz", + "integrity": "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==", + "dependencies": { + "emoji-regex": "^8.0.0", + "is-fullwidth-code-point": "^3.0.0", + "strip-ansi": "^6.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/wrap-ansi-cjs/node_modules/strip-ansi": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", + "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", + "dependencies": { + "ansi-regex": "^5.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/wrap-ansi/node_modules/ansi-styles": { + "version": "6.2.1", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-6.2.1.tgz", + "integrity": "sha512-bN798gFfQX+viw3R7yrGWRqnrN2oRkEkUjjl4JNn4E8GxxbjtG3FbrEIIY3l8/hrwUwIeCZvi4QuOTP4MErVug==", + "engines": { + "node": ">=12" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/ws": { + "version": "8.13.0", + "resolved": "https://registry.npmjs.org/ws/-/ws-8.13.0.tgz", + "integrity": "sha512-x9vcZYTrFPC7aSIbj7sRCYo7L/Xb8Iy+pW0ng0wt2vCJv7M9HOMy0UoN3rr+IFC7hb7vXoqS+P9ktyLLLhO+LA==", + "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 + } + } + }, + "node_modules/xml-name-validator": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/xml-name-validator/-/xml-name-validator-4.0.0.tgz", + "integrity": "sha512-ICP2e+jsHvAj2E2lIHxa5tjXRlKDJo4IdvPvCXbXQGdzSfmSpNVyIKMvoZHjDY9DP0zV17iI85o90vRFXNccRw==", + "engines": { + "node": ">=12" + } + }, + "node_modules/xmlchars": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/xmlchars/-/xmlchars-2.2.0.tgz", + "integrity": "sha512-JZnDKK8B0RCDw84FNdDAIpZK+JuJw+s7Lz8nksI7SIuU3UXJJslUthsi+uWBUYOwPFwW7W7PRLRfUKpxjtjFCw==" + }, + "node_modules/xmldom-sre": { + "version": "0.1.31", + "resolved": "https://registry.npmjs.org/xmldom-sre/-/xmldom-sre-0.1.31.tgz", + "integrity": "sha512-f9s+fUkX04BxQf+7mMWAp5zk61pciie+fFLC9hX9UVvCeJQfNHRHXpeo5MPcR0EUf57PYLdt+ZO4f3Ipk2oZUw==", + "engines": { + "node": ">=0.1" + } + }, + "node_modules/y18n": { + "version": "5.0.8", + "resolved": "https://registry.npmjs.org/y18n/-/y18n-5.0.8.tgz", + "integrity": "sha512-0pfFzegeDWJHJIAmTLRP2DwHjdF5s7jo9tuztdQxAhINCdvS+3nGINqPd00AphqJR/0LhANUS6/+7SCb98YOfA==", + "engines": { + "node": ">=10" + } + }, + "node_modules/yargs": { + "version": "17.7.2", + "resolved": "https://registry.npmjs.org/yargs/-/yargs-17.7.2.tgz", + "integrity": "sha512-7dSzzRQ++CKnNI/krKnYRV7JKKPUXMEh61soaHKg9mrWEhzFWhFnxPxGl+69cD1Ou63C13NUPCnmIcrvqCuM6w==", + "dependencies": { + "cliui": "^8.0.1", + "escalade": "^3.1.1", + "get-caller-file": "^2.0.5", + "require-directory": "^2.1.1", + "string-width": "^4.2.3", + "y18n": "^5.0.5", + "yargs-parser": "^21.1.1" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/yargs-parser": { + "version": "21.1.1", + "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-21.1.1.tgz", + "integrity": "sha512-tVpsJW7DdjecAiFpbIB1e3qxIQsE6NoPc5/eTdrbbIC4h0LVsWhnoa3g+m2HclBIujHzsxZ4VJVA+GUuc2/LBw==", + "engines": { + "node": ">=12" + } + }, + "node_modules/yargs/node_modules/ansi-regex": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz", + "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==", + "engines": { + "node": ">=8" + } + }, + "node_modules/yargs/node_modules/emoji-regex": { + "version": "8.0.0", + "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz", + "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==" + }, + "node_modules/yargs/node_modules/string-width": { + "version": "4.2.3", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.3.tgz", + "integrity": "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==", + "dependencies": { + "emoji-regex": "^8.0.0", + "is-fullwidth-code-point": "^3.0.0", + "strip-ansi": "^6.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/yargs/node_modules/strip-ansi": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", + "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", + "dependencies": { + "ansi-regex": "^5.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/zwitch": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/zwitch/-/zwitch-2.0.4.tgz", + "integrity": "sha512-bXE4cR/kVZhKZX/RjPEflHaKVhUVl85noU3v6b8apfQEc1x4A+zBxjZ4lN8LqGd6WZ3dl98pY4o717VFmoPp+A==", + "funding": { + "type": "github", + "url": "https://github.com/sponsors/wooorm" + } + } + } +} diff --git a/package.json b/package.json new file mode 100644 index 0000000..3fa8c23 --- /dev/null +++ b/package.json @@ -0,0 +1,109 @@ +{ + "name": "@jackyzha0/quartz", + "description": "🌱 publish your digital garden and notes as a website", + "private": true, + "version": "4.1.0", + "type": "module", + "author": "jackyzha0 ", + "license": "MIT", + "homepage": "https://quartz.jzhao.xyz", + "repository": { + "type": "git", + "url": "https://github.com/jackyzha0/quartz.git" + }, + "scripts": { + "docs": "npx quartz build --serve -d docs", + "check": "tsc --noEmit && npx prettier . --check", + "format": "npx prettier . --write", + "test": "tsx ./quartz/util/path.test.ts", + "profile": "0x -D prof ./quartz/bootstrap-cli.mjs build --concurrency=1" + }, + "engines": { + "npm": ">=9.3.1", + "node": ">=18.14" + }, + "keywords": [ + "site generator", + "ssg", + "digital-garden", + "markdown", + "blog", + "quartz" + ], + "bin": { + "quartz": "./quartz/bootstrap-cli.mjs" + }, + "dependencies": { + "@clack/prompts": "^0.6.3", + "@floating-ui/dom": "^1.4.0", + "@napi-rs/simple-git": "0.1.9", + "async-mutex": "^0.4.0", + "chalk": "^4.1.2", + "chokidar": "^3.5.3", + "cli-spinner": "^0.2.10", + "d3": "^7.8.5", + "esbuild-sass-plugin": "^2.12.0", + "flexsearch": "0.7.21", + "github-slugger": "^2.0.0", + "globby": "^13.1.4", + "gray-matter": "^4.0.3", + "hast-util-to-html": "^8.0.4", + "hast-util-to-jsx-runtime": "^1.2.0", + "hast-util-to-string": "^2.0.0", + "is-absolute-url": "^4.0.1", + "js-yaml": "^4.1.0", + "lightningcss": "1.21.7", + "mdast-util-find-and-replace": "^2.2.2", + "mdast-util-to-hast": "^12.3.0", + "mdast-util-to-string": "^3.2.0", + "micromorph": "^0.4.5", + "plausible-tracker": "^0.3.8", + "preact": "^10.14.1", + "preact-render-to-string": "^6.0.3", + "pretty-bytes": "^6.1.0", + "pretty-time": "^1.1.0", + "reading-time": "^1.5.0", + "rehype-autolink-headings": "^6.1.1", + "rehype-katex": "^6.0.3", + "rehype-mathjax": "^4.0.3", + "rehype-pretty-code": "^0.10.0", + "rehype-raw": "^6.1.1", + "rehype-slug": "^5.1.0", + "remark": "^14.0.2", + "remark-breaks": "^3.0.3", + "remark-frontmatter": "^4.0.1", + "remark-gfm": "^3.0.1", + "remark-math": "^5.1.1", + "remark-parse": "^10.0.1", + "remark-rehype": "^10.1.0", + "remark-smartypants": "^2.0.0", + "rimraf": "^5.0.1", + "serve-handler": "^6.1.5", + "source-map-support": "^0.5.21", + "to-vfile": "^7.2.4", + "toml": "^3.0.0", + "unified": "^10.1.2", + "unist-util-visit": "^4.1.2", + "vfile": "^5.3.7", + "workerpool": "^6.4.0", + "ws": "^8.13.0", + "yargs": "^17.7.2" + }, + "devDependencies": { + "@types/cli-spinner": "^0.2.1", + "@types/d3": "^7.4.0", + "@types/flexsearch": "^0.7.3", + "@types/hast": "^2.3.4", + "@types/js-yaml": "^4.0.5", + "@types/node": "^20.1.2", + "@types/pretty-time": "^1.1.2", + "@types/source-map-support": "^0.5.6", + "@types/workerpool": "^6.4.0", + "@types/ws": "^8.5.5", + "@types/yargs": "^17.0.24", + "esbuild": "0.19.2", + "prettier": "^3.0.0", + "tsx": "^3.12.7", + "typescript": "^5.0.4" + } +} diff --git a/pages/About.md b/pages/About.md deleted file mode 100755 index a933a77..0000000 --- a/pages/About.md +++ /dev/null @@ -1,8 +0,0 @@ -alias:: TFTRocks Notes - -- This is an experimental set of notes for the Tools For Thought Rocks community -- It is a place to document the events, speakers, videos, chat transcripts, and other content around the community -- It also will hold information and documentation about the [[Tools For Thought Mastodon]] -- How To - - [[How to add content to the TFTRocks Notes site]] - - \ No newline at end of file diff --git a/pages/Butter.md b/pages/Butter.md deleted file mode 100755 index 1e54546..0000000 --- a/pages/Butter.md +++ /dev/null @@ -1 +0,0 @@ -link:: https://www.butter.us/ diff --git a/pages/Can I delete the contents within the %22bak%22 folder.md b/pages/Can I delete the contents within the %22bak%22 folder.md deleted file mode 100755 index dc08c42..0000000 --- a/pages/Can I delete the contents within the %22bak%22 folder.md +++ /dev/null @@ -1,6 +0,0 @@ -tags:: LogSeq - -- link:: https://discuss.logseq.com/t/can-i-delete-the-contents-within-the-bak-folder/8068 -- Answer by [[Ramses Oudt]]: -- > The  `bak`  folder contains backup files. These are created when there’s a conflict between the text file on disk and the page in the database—which mostly happens when using Logseq in combination with sync services like OneDrive, Dropbox, and iCloud. - > You can safely delete the files in the  `bak`  folder, though they may come in handy if you want to roll back a file version. \ No newline at end of file diff --git a/pages/Charles Chamberlain.md b/pages/Charles Chamberlain.md deleted file mode 100644 index b8f7a91..0000000 --- a/pages/Charles Chamberlain.md +++ /dev/null @@ -1,3 +0,0 @@ -link:: https://birch.ink/ -twitter:: https://twitter.com/charlesetc -tags:: person, speaker diff --git a/pages/Chartodon.md b/pages/Chartodon.md deleted file mode 100755 index 917f615..0000000 --- a/pages/Chartodon.md +++ /dev/null @@ -1,6 +0,0 @@ -tags:: #tool, #activitypub -author:: [[Colin the Mathmo]] -activitypub:: https://mathstodon.xyz/@Chartodon -- Experimental account to provide charts of discussions. Currently run by hand, might eventually become a full-blown 'bot. -- Invoke me by replying to a toot with a single line saying `Calling @Chartodon`. -- I may take a while to reply, depending on how busy I am, and how extensive the discussion. diff --git a/pages/Colin the Mathmo.md b/pages/Colin the Mathmo.md deleted file mode 100755 index 19cdaed..0000000 --- a/pages/Colin the Mathmo.md +++ /dev/null @@ -1,5 +0,0 @@ -pronouns:: he/him -link:: https://www.solipsys.co.uk/ -activitypub:: https://mathstodon.xyz/@ColinTheMathmo) -tags:: #math, #person, #developer -- maker of [[Chartodon]] for [[Mastodon]] \ No newline at end of file diff --git a/pages/Colophon.md b/pages/Colophon.md deleted file mode 100755 index 21f5c9e..0000000 --- a/pages/Colophon.md +++ /dev/null @@ -1,7 +0,0 @@ -- This site is managed with #LogSeq -- The source lives in the [[Tools for Thought Rocks]] Github organization at https://github.com/ToolsForThoughtRocks/ToolsForThoughtLogSeq -- Alternately, you can request be added to a Dropbox folder that contains a checkout of the site for easier editing that doesn't have to mean learning git -- There is a build script that turns it into static pages that are published via Github Pages -- You can access it via https://notes.toolsforthought.rocks -- {{embed ((63792362-9130-4138-ac94-31e5c6ff7708))}} -- \ No newline at end of file diff --git a/pages/Community Infrastructure.md b/pages/Community Infrastructure.md deleted file mode 100755 index 96fb3be..0000000 --- a/pages/Community Infrastructure.md +++ /dev/null @@ -1,24 +0,0 @@ -- This is all of the infrastructure, from people, to tools, to access, that helps run the [[Tools for Thought Rocks]] community -- Current organizers are [[Christina Bowen]], [[Boris Mann]], and [[Jess Martin]] - - Anyone can propose / organize events! - - What else can we do together? -- [[Luma]] https://lu.ma/toolsforthoughtrocks is the link for the community page on Luma - - There is also a shared org account https://lu.ma/toolsforthought -- [[github]] https://github.com/ToolsForThoughtRocks - - A number of people are [owners / members of the org](https://github.com/orgs/ToolsForThoughtRocks/people) (not all are publicly listed) - - [[Boris Mann]], Owner - - [[Jess Martin]], Owner - - [[Michael Gartner]], Owner -- Youtube https://youtube.com/@toolsforthought - id:: 637985aa-49bb-4521-ac29-7d37137d6d78 - - [[Boris Mann]], Owner - - [[Jess Martin]], Owner - - [[Christina Bowen]], Owner - - [[Michael Gartner]], Manager -- [[Open Collective]] https://opencollective.com/toolsforthought -- [[Tools For Thought Mastodon]] https://toolsforthought.rocks -- The domain `toolsforthought.rocks` is registered at and has DNS hosted by [[Google Domains]] - id:: 63792362-9130-4138-ac94-31e5c6ff7708 - - [[Boris Mann]] and [[Jess Martin]] have full permissions over it -- The [[Colophon]] describes the infrastructure for this [[TFTRocks Notes]] site -- Paid [[Zoom]] account for [[Jess Martin]] and [[Boris Mann]] as part of working at [[Fission]] \ No newline at end of file diff --git a/pages/Event Listing.md b/pages/Event Listing.md deleted file mode 100755 index b44bb2e..0000000 --- a/pages/Event Listing.md +++ /dev/null @@ -1,8 +0,0 @@ -- Upcoming - - TODO generate listing event based on `#event` and `date` property using Advanced query - id:: 62bcd24d-7ae0-42f9-bc4e-650d0b35559f - - https://discord.com/channels/725182569297215569/743139225746145311/832898837953642507 - - https://discord.com/channels/725182569297215569/743139225746145311/983417782643327096 - - -- Past - - {{embed ((62bcd24d-7ae0-42f9-bc4e-650d0b35559f))}} \ No newline at end of file diff --git a/pages/Events.md b/pages/Events.md deleted file mode 100755 index da0e4ac..0000000 --- a/pages/Events.md +++ /dev/null @@ -1,2 +0,0 @@ -- Registration for events is via the [[Luma]] community at https://calendar.toolsforthought.rocks -- \ No newline at end of file diff --git a/pages/February 2022 Meeting.md b/pages/February 2022 Meeting.md deleted file mode 100755 index c096bce..0000000 --- a/pages/February 2022 Meeting.md +++ /dev/null @@ -1,5 +0,0 @@ -date:: [[Feb 23rd, 2022]] -speakers:: [[Alexander Obernauer]], [[Geoffrey Litt]], [[Dalton Banks]] -video:: https://youtu.be/ORAFb0Lh7ys -link:: https://lu.ma/tftrocks-feb22 -tags:: event, monthly-meeting diff --git a/pages/Luma.md b/pages/Luma.md deleted file mode 100755 index 7667275..0000000 --- a/pages/Luma.md +++ /dev/null @@ -1,3 +0,0 @@ -link:: https://lu.ma - -- \ No newline at end of file diff --git a/pages/October 2021 Meeting.md b/pages/October 2021 Meeting.md deleted file mode 100755 index 1a15045..0000000 --- a/pages/October 2021 Meeting.md +++ /dev/null @@ -1,23 +0,0 @@ -date:: [[Oct 29th, 2021]] -speakers:: [[Rosano]] -video:: https://youtu.be/McKXW-bP2HQ -link:: https://lu.ma/tftrocks-oct -tags:: event, monthly-meeting - -- [[October 2021 Chat Log]] -- Lobby Chatter - - Let's try different collaborative apps, like [[Sprout]] that Wei Wei builds - - [[Butter]] is one that [[Boris]] tried the other day - - Bandwidth is 5x more expensive than storage costs, per GB - - [[Cloudflare]] released [[Cloudflare R2]], an [[Amazon S3]] competitor with no bandwidth costs -- Notes - - [[Rosano]] [[Wetware of Writing and Doing]] - - [[Joybox]] - - [[Hyperdraft]] - - [[Write of Passage]] by [[David Perell]] - - [[Cory Doctorow]], [[The Memex Method]] - - [[Bill Seitz]] [[POSSE]] and [[PESOS]] - - [[Bill Seitz]] [MashUp](http://webseitz.fluxent.com/wiki/MashUp) - - [[Marc-Antoine Parent]] - - [[Andy Matuschak]] [[Export considered harmful]] - - [[Linus Lee]] Twitter client, [[Lucerne]] \ No newline at end of file diff --git a/pages/Tools For Thought Mastodon.md b/pages/Tools For Thought Mastodon.md deleted file mode 100755 index 7dd0aba..0000000 --- a/pages/Tools For Thought Mastodon.md +++ /dev/null @@ -1,7 +0,0 @@ -link:: https://toolsforthought.rocks - -- opencollective:: https://opencollective.com/toolsforthought -- A #Mastodon server for the #[[Tools for Thought Rocks]] community -- Managed hosting by [[Federated Hosting by Spacebear]] -- Setup and initially administered by [[Boris Mann]] -- \ No newline at end of file diff --git a/pages/contents.md b/pages/contents.md deleted file mode 100755 index 19b7962..0000000 --- a/pages/contents.md +++ /dev/null @@ -1,11 +0,0 @@ -- [[About]] -- [[People]] - - [[Speaker Listing]] -- [[Events]] - - [[Monthly Meetings]] -- [[Tools for Thought App List]] -- [[Tools For Thought Mastodon]] - - [[Mastodon Resources]] -- [[Community Infrastructure]] -- [[Colophon]] -- [[Utilities]] \ No newline at end of file diff --git a/pages/favorites.md b/pages/favorites.md deleted file mode 100755 index 39cdd0d..0000000 --- a/pages/favorites.md +++ /dev/null @@ -1 +0,0 @@ -- diff --git a/quartz.config.ts b/quartz.config.ts new file mode 100644 index 0000000..9f5ec95 --- /dev/null +++ b/quartz.config.ts @@ -0,0 +1,77 @@ +import { QuartzConfig } from "./quartz/cfg" +import * as Plugin from "./quartz/plugins" + +const config: QuartzConfig = { + configuration: { + pageTitle: "🌱 Tools for Thought Rocks!", + enableSPA: true, + enablePopovers: true, + analytics: { + provider: "plausible", + }, + baseUrl: "notes.toolsforthought.rocks", + ignorePatterns: ["private", "templates", ".obsidian"], + defaultDateType: "created", + theme: { + typography: { + header: "Schibsted Grotesk", + body: "Source Sans Pro", + code: "IBM Plex Mono", + }, + colors: { + lightMode: { + light: "#faf8f8", + lightgray: "#e5e5e5", + gray: "#b8b8b8", + darkgray: "#4e4e4e", + dark: "#2b2b2b", + secondary: "#284b63", + tertiary: "#84a59d", + highlight: "rgba(143, 159, 169, 0.15)", + }, + darkMode: { + light: "#161618", + lightgray: "#393639", + gray: "#646464", + darkgray: "#d4d4d4", + dark: "#ebebec", + secondary: "#7b97aa", + tertiary: "#84a59d", + highlight: "rgba(143, 159, 169, 0.15)", + }, + }, + }, + }, + plugins: { + transformers: [ + Plugin.FrontMatter(), + Plugin.TableOfContents(), + Plugin.CreatedModifiedDate({ + priority: ["frontmatter", "filesystem"], // you can add 'git' here for last modified from Git but this makes the build slower + }), + Plugin.SyntaxHighlighting(), + Plugin.ObsidianFlavoredMarkdown({ enableInHtmlEmbed: false }), + Plugin.GitHubFlavoredMarkdown(), + Plugin.CrawlLinks({ markdownLinkResolution: "shortest" }), + Plugin.Latex({ renderEngine: "katex" }), + Plugin.Description(), + ], + filters: [Plugin.RemoveDrafts()], + emitters: [ + Plugin.AliasRedirects(), + Plugin.ComponentResources({ fontOrigin: "googleFonts" }), + Plugin.ContentPage(), + Plugin.FolderPage(), + Plugin.TagPage(), + Plugin.ContentIndex({ + enableSiteMap: true, + enableRSS: true, + }), + Plugin.Assets(), + Plugin.Static(), + Plugin.NotFoundPage(), + ], + }, +} + +export default config diff --git a/quartz.layout.ts b/quartz.layout.ts new file mode 100644 index 0000000..8b6edd8 --- /dev/null +++ b/quartz.layout.ts @@ -0,0 +1,48 @@ +import { PageLayout, SharedLayout } from "./quartz/cfg" +import * as Component from "./quartz/components" + +// components shared across all pages +export const sharedPageComponents: SharedLayout = { + head: Component.Head(), + header: [], + footer: Component.Footer({ + links: { + GitHub: "https://github.com/jackyzha0/quartz", + "Discord Community": "https://discord.gg/cRFFHYye7t", + }, + }), +} + +// components for pages that display a single page (e.g. a single note) +export const defaultContentPageLayout: PageLayout = { + beforeBody: [ + Component.Breadcrumbs(), + Component.ArticleTitle(), + Component.ContentMeta(), + Component.TagList(), + ], + left: [ + Component.PageTitle(), + Component.MobileOnly(Component.Spacer()), + Component.Search(), + Component.Darkmode(), + Component.DesktopOnly(Component.Explorer()), + ], + right: [ + Component.Graph(), + Component.DesktopOnly(Component.TableOfContents()), + Component.Backlinks(), + ], +} + +// components for pages that display lists of pages (e.g. tags or folders) +export const defaultListPageLayout: PageLayout = { + beforeBody: [Component.ArticleTitle()], + left: [ + Component.PageTitle(), + Component.MobileOnly(Component.Spacer()), + Component.Search(), + Component.Darkmode(), + ], + right: [], +} diff --git a/quartz/bootstrap-cli.mjs b/quartz/bootstrap-cli.mjs new file mode 100755 index 0000000..35d06af --- /dev/null +++ b/quartz/bootstrap-cli.mjs @@ -0,0 +1,41 @@ +#!/usr/bin/env node +import yargs from "yargs" +import { hideBin } from "yargs/helpers" +import { + handleBuild, + handleCreate, + handleUpdate, + handleRestore, + handleSync, +} from "./cli/handlers.js" +import { CommonArgv, BuildArgv, CreateArgv, SyncArgv } from "./cli/args.js" +import { version } from "./cli/constants.js" + +yargs(hideBin(process.argv)) + .scriptName("quartz") + .version(version) + .usage("$0 [args]") + .command("create", "Initialize Quartz", CreateArgv, async (argv) => { + await handleCreate(argv) + }) + .command("update", "Get the latest Quartz updates", CommonArgv, async (argv) => { + await handleUpdate(argv) + }) + .command( + "restore", + "Try to restore your content folder from the cache", + CommonArgv, + async (argv) => { + await handleRestore(argv) + }, + ) + .command("sync", "Sync your Quartz to and from GitHub.", SyncArgv, async (argv) => { + await handleSync(argv) + }) + .command("build", "Build Quartz into a bundle of static HTML files", BuildArgv, async (argv) => { + await handleBuild(argv) + }) + .showHelpOnFail(false) + .help() + .strict() + .demandCommand().argv diff --git a/quartz/bootstrap-worker.mjs b/quartz/bootstrap-worker.mjs new file mode 100644 index 0000000..b08689c --- /dev/null +++ b/quartz/bootstrap-worker.mjs @@ -0,0 +1,7 @@ +#!/usr/bin/env node +import workerpool from "workerpool" +const cacheFile = "./.quartz-cache/transpiled-worker.mjs" +const { parseFiles } = await import(cacheFile) +workerpool.worker({ + parseFiles, +}) diff --git a/quartz/build.ts b/quartz/build.ts new file mode 100644 index 0000000..5752caa --- /dev/null +++ b/quartz/build.ts @@ -0,0 +1,183 @@ +import sourceMapSupport from "source-map-support" +sourceMapSupport.install(options) +import path from "path" +import { PerfTimer } from "./util/perf" +import { rimraf } from "rimraf" +import { isGitIgnored } from "globby" +import chalk from "chalk" +import { parseMarkdown } from "./processors/parse" +import { filterContent } from "./processors/filter" +import { emitContent } from "./processors/emit" +import cfg from "../quartz.config" +import { FilePath, joinSegments, slugifyFilePath } from "./util/path" +import chokidar from "chokidar" +import { ProcessedContent } from "./plugins/vfile" +import { Argv, BuildCtx } from "./util/ctx" +import { glob, toPosixPath } from "./util/glob" +import { trace } from "./util/trace" +import { options } from "./util/sourcemap" +import { Mutex } from "async-mutex" + +async function buildQuartz(argv: Argv, mut: Mutex, clientRefresh: () => void) { + const ctx: BuildCtx = { + argv, + cfg, + allSlugs: [], + } + + const perf = new PerfTimer() + const output = argv.output + + const pluginCount = Object.values(cfg.plugins).flat().length + const pluginNames = (key: "transformers" | "filters" | "emitters") => + cfg.plugins[key].map((plugin) => plugin.name) + if (argv.verbose) { + console.log(`Loaded ${pluginCount} plugins`) + console.log(` Transformers: ${pluginNames("transformers").join(", ")}`) + console.log(` Filters: ${pluginNames("filters").join(", ")}`) + console.log(` Emitters: ${pluginNames("emitters").join(", ")}`) + } + + const release = await mut.acquire() + perf.addEvent("clean") + await rimraf(output) + console.log(`Cleaned output directory \`${output}\` in ${perf.timeSince("clean")}`) + + perf.addEvent("glob") + const allFiles = await glob("**/*.*", argv.directory, cfg.configuration.ignorePatterns) + const fps = allFiles.filter((fp) => fp.endsWith(".md")).sort() + console.log( + `Found ${fps.length} input files from \`${argv.directory}\` in ${perf.timeSince("glob")}`, + ) + + const filePaths = fps.map((fp) => joinSegments(argv.directory, fp) as FilePath) + ctx.allSlugs = allFiles.map((fp) => slugifyFilePath(fp as FilePath)) + + const parsedFiles = await parseMarkdown(ctx, filePaths) + const filteredContent = filterContent(ctx, parsedFiles) + await emitContent(ctx, filteredContent) + console.log(chalk.green(`Done processing ${fps.length} files in ${perf.timeSince()}`)) + release() + + if (argv.serve) { + return startServing(ctx, mut, parsedFiles, clientRefresh) + } +} + +// setup watcher for rebuilds +async function startServing( + ctx: BuildCtx, + mut: Mutex, + initialContent: ProcessedContent[], + clientRefresh: () => void, +) { + const { argv } = ctx + + const ignored = await isGitIgnored() + const contentMap = new Map() + for (const content of initialContent) { + const [_tree, vfile] = content + contentMap.set(vfile.data.filePath!, content) + } + + const initialSlugs = ctx.allSlugs + let lastBuildMs = 0 + const toRebuild: Set = new Set() + const toRemove: Set = new Set() + const trackedAssets: Set = new Set() + async function rebuild(fp: string, action: "add" | "change" | "delete") { + // don't do anything for gitignored files + if (ignored(fp)) { + return + } + + // dont bother rebuilding for non-content files, just track and refresh + fp = toPosixPath(fp) + const filePath = joinSegments(argv.directory, fp) as FilePath + if (path.extname(fp) !== ".md") { + if (action === "add" || action === "change") { + trackedAssets.add(filePath) + } else if (action === "delete") { + trackedAssets.delete(filePath) + } + clientRefresh() + return + } + + if (action === "add" || action === "change") { + toRebuild.add(filePath) + } else if (action === "delete") { + toRemove.add(filePath) + } + + // debounce rebuilds every 250ms + + const buildStart = new Date().getTime() + lastBuildMs = buildStart + const release = await mut.acquire() + if (lastBuildMs > buildStart) { + release() + return + } + + const perf = new PerfTimer() + console.log(chalk.yellow("Detected change, rebuilding...")) + try { + const filesToRebuild = [...toRebuild].filter((fp) => !toRemove.has(fp)) + + const trackedSlugs = [...new Set([...contentMap.keys(), ...toRebuild, ...trackedAssets])] + .filter((fp) => !toRemove.has(fp)) + .map((fp) => slugifyFilePath(path.posix.relative(argv.directory, fp) as FilePath)) + + ctx.allSlugs = [...new Set([...initialSlugs, ...trackedSlugs])] + const parsedContent = await parseMarkdown(ctx, filesToRebuild) + for (const content of parsedContent) { + const [_tree, vfile] = content + contentMap.set(vfile.data.filePath!, content) + } + + for (const fp of toRemove) { + contentMap.delete(fp) + } + + const parsedFiles = [...contentMap.values()] + const filteredContent = filterContent(ctx, parsedFiles) + + // TODO: we can probably traverse the link graph to figure out what's safe to delete here + // instead of just deleting everything + await rimraf(argv.output) + await emitContent(ctx, filteredContent) + console.log(chalk.green(`Done rebuilding in ${perf.timeSince()}`)) + } catch { + console.log(chalk.yellow(`Rebuild failed. Waiting on a change to fix the error...`)) + } + + clientRefresh() + toRebuild.clear() + toRemove.clear() + release() + } + + const watcher = chokidar.watch(".", { + persistent: true, + cwd: argv.directory, + ignoreInitial: true, + }) + + watcher + .on("add", (fp) => rebuild(fp, "add")) + .on("change", (fp) => rebuild(fp, "change")) + .on("unlink", (fp) => rebuild(fp, "delete")) + + return async () => { + await watcher.close() + } +} + +export default async (argv: Argv, mut: Mutex, clientRefresh: () => void) => { + try { + return await buildQuartz(argv, mut, clientRefresh) + } catch (err) { + trace("\nExiting Quartz due to a fatal error", err as Error) + } +} diff --git a/quartz/cfg.ts b/quartz/cfg.ts new file mode 100644 index 0000000..8371b5e --- /dev/null +++ b/quartz/cfg.ts @@ -0,0 +1,55 @@ +import { ValidDateType } from "./components/Date" +import { QuartzComponent } from "./components/types" +import { PluginTypes } from "./plugins/types" +import { Theme } from "./util/theme" + +export type Analytics = + | null + | { + provider: "plausible" + } + | { + provider: "google" + tagId: string + } + | { + provider: "umami" + websiteId: string + } + +export interface GlobalConfiguration { + pageTitle: string + /** Whether to enable single-page-app style rendering. this prevents flashes of unstyled content and improves smoothness of Quartz */ + enableSPA: boolean + /** Whether to display Wikipedia-style popovers when hovering over links */ + enablePopovers: boolean + /** Analytics mode */ + analytics: Analytics + /** Glob patterns to not search */ + ignorePatterns: string[] + /** Whether to use created, modified, or published as the default type of date */ + defaultDateType: ValidDateType + /** Base URL to use for CNAME files, sitemaps, and RSS feeds that require an absolute URL. + * Quartz will avoid using this as much as possible and use relative URLs most of the time + */ + baseUrl?: string + theme: Theme +} + +export interface QuartzConfig { + configuration: GlobalConfiguration + plugins: PluginTypes +} + +export interface FullPageLayout { + head: QuartzComponent + header: QuartzComponent[] + beforeBody: QuartzComponent[] + pageBody: QuartzComponent + left: QuartzComponent[] + right: QuartzComponent[] + footer: QuartzComponent +} + +export type PageLayout = Pick +export type SharedLayout = Pick diff --git a/quartz/cli/args.js b/quartz/cli/args.js new file mode 100644 index 0000000..3543e2e --- /dev/null +++ b/quartz/cli/args.js @@ -0,0 +1,98 @@ +export const CommonArgv = { + directory: { + string: true, + alias: ["d"], + default: "content", + describe: "directory to look for content files", + }, + verbose: { + boolean: true, + alias: ["v"], + default: false, + describe: "print out extra logging information", + }, +} + +export const CreateArgv = { + ...CommonArgv, + source: { + string: true, + alias: ["s"], + describe: "source directory to copy/create symlink from", + }, + strategy: { + string: true, + alias: ["X"], + choices: ["new", "copy", "symlink"], + describe: "strategy for content folder setup", + }, + links: { + string: true, + alias: ["l"], + choices: ["absolute", "shortest", "relative"], + describe: "strategy to resolve links", + }, +} + +export const SyncArgv = { + ...CommonArgv, + commit: { + boolean: true, + default: true, + describe: "create a git commit for your unsaved changes", + }, + push: { + boolean: true, + default: true, + describe: "push updates to your Quartz fork", + }, + pull: { + boolean: true, + default: true, + describe: "pull updates from your Quartz fork", + }, +} + +export const BuildArgv = { + ...CommonArgv, + output: { + string: true, + alias: ["o"], + default: "public", + describe: "output folder for files", + }, + serve: { + boolean: true, + default: false, + describe: "run a local server to live-preview your Quartz", + }, + baseDir: { + string: true, + default: "", + describe: "base path to serve your local server on", + }, + port: { + number: true, + default: 8080, + describe: "port to serve Quartz on", + }, + wsPort: { + number: true, + default: 3001, + describe: "port to use for WebSocket-based hot-reload notifications", + }, + remoteDevHost: { + string: true, + default: "", + describe: "A URL override for the websocket connection if you are not developing on localhost", + }, + bundleInfo: { + boolean: true, + default: false, + describe: "show detailed bundle information", + }, + concurrency: { + number: true, + describe: "how many threads to use to parse notes", + }, +} diff --git a/quartz/cli/constants.js b/quartz/cli/constants.js new file mode 100644 index 0000000..f4a9ce5 --- /dev/null +++ b/quartz/cli/constants.js @@ -0,0 +1,15 @@ +import path from "path" +import { readFileSync } from "fs" + +/** + * All constants relating to helpers or handlers + */ +export const ORIGIN_NAME = "origin" +export const UPSTREAM_NAME = "upstream" +export const QUARTZ_SOURCE_BRANCH = "v4" +export const cwd = process.cwd() +export const cacheDir = path.join(cwd, ".quartz-cache") +export const cacheFile = "./quartz/.quartz-cache/transpiled-build.mjs" +export const fp = "./quartz/build.ts" +export const { version } = JSON.parse(readFileSync("./package.json").toString()) +export const contentCacheFolder = path.join(cacheDir, "content-cache") diff --git a/quartz/cli/handlers.js b/quartz/cli/handlers.js new file mode 100644 index 0000000..48a44ec --- /dev/null +++ b/quartz/cli/handlers.js @@ -0,0 +1,511 @@ +import { promises } from "fs" +import path from "path" +import esbuild from "esbuild" +import chalk from "chalk" +import { sassPlugin } from "esbuild-sass-plugin" +import fs from "fs" +import { intro, outro, select, text } from "@clack/prompts" +import { rimraf } from "rimraf" +import chokidar from "chokidar" +import prettyBytes from "pretty-bytes" +import { execSync, spawnSync } from "child_process" +import http from "http" +import serveHandler from "serve-handler" +import { WebSocketServer } from "ws" +import { randomUUID } from "crypto" +import { Mutex } from "async-mutex" +import { CreateArgv } from "./args.js" +import { + exitIfCancel, + escapePath, + gitPull, + popContentFolder, + stashContentFolder, +} from "./helpers.js" +import { + UPSTREAM_NAME, + QUARTZ_SOURCE_BRANCH, + ORIGIN_NAME, + version, + fp, + cacheFile, + cwd, +} from "./constants.js" + +/** + * Handles `npx quartz create` + * @param {*} argv arguments for `create` + */ +export async function handleCreate(argv) { + console.log() + intro(chalk.bgGreen.black(` Quartz v${version} `)) + const contentFolder = path.join(cwd, argv.directory) + let setupStrategy = argv.strategy?.toLowerCase() + let linkResolutionStrategy = argv.links?.toLowerCase() + const sourceDirectory = argv.source + + // If all cmd arguments were provided, check if theyre valid + if (setupStrategy && linkResolutionStrategy) { + // If setup isn't, "new", source argument is required + if (setupStrategy !== "new") { + // Error handling + if (!sourceDirectory) { + outro( + chalk.red( + `Setup strategies (arg '${chalk.yellow( + `-${CreateArgv.strategy.alias[0]}`, + )}') other than '${chalk.yellow( + "new", + )}' require content folder argument ('${chalk.yellow( + `-${CreateArgv.source.alias[0]}`, + )}') to be set`, + ), + ) + process.exit(1) + } else { + if (!fs.existsSync(sourceDirectory)) { + outro( + chalk.red( + `Input directory to copy/symlink 'content' from not found ('${chalk.yellow( + sourceDirectory, + )}', invalid argument "${chalk.yellow(`-${CreateArgv.source.alias[0]}`)})`, + ), + ) + process.exit(1) + } else if (!fs.lstatSync(sourceDirectory).isDirectory()) { + outro( + chalk.red( + `Source directory to copy/symlink 'content' from is not a directory (found file at '${chalk.yellow( + sourceDirectory, + )}', invalid argument ${chalk.yellow(`-${CreateArgv.source.alias[0]}`)}")`, + ), + ) + process.exit(1) + } + } + } + } + + // Use cli process if cmd args werent provided + if (!setupStrategy) { + setupStrategy = exitIfCancel( + await select({ + message: `Choose how to initialize the content in \`${contentFolder}\``, + options: [ + { value: "new", label: "Empty Quartz" }, + { value: "copy", label: "Copy an existing folder", hint: "overwrites `content`" }, + { + value: "symlink", + label: "Symlink an existing folder", + hint: "don't select this unless you know what you are doing!", + }, + ], + }), + ) + } + + async function rmContentFolder() { + const contentStat = await fs.promises.lstat(contentFolder) + if (contentStat.isSymbolicLink()) { + await fs.promises.unlink(contentFolder) + } else { + await rimraf(contentFolder) + } + } + + await fs.promises.unlink(path.join(contentFolder, ".gitkeep")) + if (setupStrategy === "copy" || setupStrategy === "symlink") { + let originalFolder = sourceDirectory + + // If input directory was not passed, use cli + if (!sourceDirectory) { + originalFolder = escapePath( + exitIfCancel( + await text({ + message: "Enter the full path to existing content folder", + placeholder: + "On most terminal emulators, you can drag and drop a folder into the window and it will paste the full path", + validate(fp) { + const fullPath = escapePath(fp) + if (!fs.existsSync(fullPath)) { + return "The given path doesn't exist" + } else if (!fs.lstatSync(fullPath).isDirectory()) { + return "The given path is not a folder" + } + }, + }), + ), + ) + } + + await rmContentFolder() + if (setupStrategy === "copy") { + await fs.promises.cp(originalFolder, contentFolder, { + recursive: true, + preserveTimestamps: true, + }) + } else if (setupStrategy === "symlink") { + await fs.promises.symlink(originalFolder, contentFolder, "dir") + } + } else if (setupStrategy === "new") { + await fs.promises.writeFile( + path.join(contentFolder, "index.md"), + `--- +title: Welcome to Quartz +--- + +This is a blank Quartz installation. +See the [documentation](https://quartz.jzhao.xyz) for how to get started. +`, + ) + } + + // Use cli process if cmd args werent provided + if (!linkResolutionStrategy) { + // get a preferred link resolution strategy + linkResolutionStrategy = exitIfCancel( + await select({ + message: `Choose how Quartz should resolve links in your content. You can change this later in \`quartz.config.ts\`.`, + options: [ + { + value: "absolute", + label: "Treat links as absolute path", + hint: "for content made for Quartz 3 and Hugo", + }, + { + value: "shortest", + label: "Treat links as shortest path", + hint: "for most Obsidian vaults", + }, + { + value: "relative", + label: "Treat links as relative paths", + hint: "for just normal Markdown files", + }, + ], + }), + ) + } + + // now, do config changes + const configFilePath = path.join(cwd, "quartz.config.ts") + let configContent = await fs.promises.readFile(configFilePath, { encoding: "utf-8" }) + configContent = configContent.replace( + /markdownLinkResolution: '(.+)'/, + `markdownLinkResolution: '${linkResolutionStrategy}'`, + ) + await fs.promises.writeFile(configFilePath, configContent) + + outro(`You're all set! Not sure what to do next? Try: + • Customizing Quartz a bit more by editing \`quartz.config.ts\` + • Running \`npx quartz build --serve\` to preview your Quartz locally + • Hosting your Quartz online (see: https://quartz.jzhao.xyz/hosting) +`) +} + +/** + * Handles `npx quartz build` + * @param {*} argv arguments for `build` + */ +export async function handleBuild(argv) { + console.log(chalk.bgGreen.black(`\n Quartz v${version} \n`)) + const ctx = await esbuild.context({ + entryPoints: [fp], + outfile: cacheFile, + bundle: true, + keepNames: true, + minifyWhitespace: true, + minifySyntax: true, + platform: "node", + format: "esm", + jsx: "automatic", + jsxImportSource: "preact", + packages: "external", + metafile: true, + sourcemap: true, + sourcesContent: false, + plugins: [ + sassPlugin({ + type: "css-text", + cssImports: true, + }), + { + name: "inline-script-loader", + setup(build) { + build.onLoad({ filter: /\.inline\.(ts|js)$/ }, async (args) => { + let text = await promises.readFile(args.path, "utf8") + + // remove default exports that we manually inserted + text = text.replace("export default", "") + text = text.replace("export", "") + + const sourcefile = path.relative(path.resolve("."), args.path) + const resolveDir = path.dirname(sourcefile) + const transpiled = await esbuild.build({ + stdin: { + contents: text, + loader: "ts", + resolveDir, + sourcefile, + }, + write: false, + bundle: true, + platform: "browser", + format: "esm", + }) + const rawMod = transpiled.outputFiles[0].text + return { + contents: rawMod, + loader: "text", + } + }) + }, + }, + ], + }) + + const buildMutex = new Mutex() + let lastBuildMs = 0 + let cleanupBuild = null + const build = async (clientRefresh) => { + const buildStart = new Date().getTime() + lastBuildMs = buildStart + const release = await buildMutex.acquire() + if (lastBuildMs > buildStart) { + release() + return + } + + if (cleanupBuild) { + await cleanupBuild() + console.log(chalk.yellow("Detected a source code change, doing a hard rebuild...")) + } + + const result = await ctx.rebuild().catch((err) => { + console.error(`${chalk.red("Couldn't parse Quartz configuration:")} ${fp}`) + console.log(`Reason: ${chalk.grey(err)}`) + process.exit(1) + }) + release() + + if (argv.bundleInfo) { + const outputFileName = "quartz/.quartz-cache/transpiled-build.mjs" + const meta = result.metafile.outputs[outputFileName] + console.log( + `Successfully transpiled ${Object.keys(meta.inputs).length} files (${prettyBytes( + meta.bytes, + )})`, + ) + console.log(await esbuild.analyzeMetafile(result.metafile, { color: true })) + } + + // bypass module cache + // https://github.com/nodejs/modules/issues/307 + const { default: buildQuartz } = await import(`../../${cacheFile}?update=${randomUUID()}`) + // ^ this import is relative, so base "cacheFile" path can't be used + + cleanupBuild = await buildQuartz(argv, buildMutex, clientRefresh) + clientRefresh() + } + + if (argv.serve) { + const connections = [] + const clientRefresh = () => connections.forEach((conn) => conn.send("rebuild")) + + if (argv.baseDir !== "" && !argv.baseDir.startsWith("/")) { + argv.baseDir = "/" + argv.baseDir + } + + await build(clientRefresh) + const server = http.createServer(async (req, res) => { + if (argv.baseDir && !req.url?.startsWith(argv.baseDir)) { + console.log( + chalk.red( + `[404] ${req.url} (warning: link outside of site, this is likely a Quartz bug)`, + ), + ) + res.writeHead(404) + res.end() + return + } + + // strip baseDir prefix + req.url = req.url?.slice(argv.baseDir.length) + + const serve = async () => { + const release = await buildMutex.acquire() + await serveHandler(req, res, { + public: argv.output, + directoryListing: false, + headers: [ + { + source: "**/*.html", + headers: [{ key: "Content-Disposition", value: "inline" }], + }, + ], + }) + const status = res.statusCode + const statusString = + status >= 200 && status < 300 ? chalk.green(`[${status}]`) : chalk.red(`[${status}]`) + console.log(statusString + chalk.grey(` ${argv.baseDir}${req.url}`)) + release() + } + + const redirect = (newFp) => { + newFp = argv.baseDir + newFp + res.writeHead(302, { + Location: newFp, + }) + console.log(chalk.yellow("[302]") + chalk.grey(` ${argv.baseDir}${req.url} -> ${newFp}`)) + res.end() + } + + let fp = req.url?.split("?")[0] ?? "/" + + // handle redirects + if (fp.endsWith("/")) { + // /trailing/ + // does /trailing/index.html exist? if so, serve it + const indexFp = path.posix.join(fp, "index.html") + if (fs.existsSync(path.posix.join(argv.output, indexFp))) { + req.url = fp + return serve() + } + + // does /trailing.html exist? if so, redirect to /trailing + let base = fp.slice(0, -1) + if (path.extname(base) === "") { + base += ".html" + } + if (fs.existsSync(path.posix.join(argv.output, base))) { + return redirect(fp.slice(0, -1)) + } + } else { + // /regular + // does /regular.html exist? if so, serve it + let base = fp + if (path.extname(base) === "") { + base += ".html" + } + if (fs.existsSync(path.posix.join(argv.output, base))) { + req.url = fp + return serve() + } + + // does /regular/index.html exist? if so, redirect to /regular/ + let indexFp = path.posix.join(fp, "index.html") + if (fs.existsSync(path.posix.join(argv.output, indexFp))) { + return redirect(fp + "/") + } + } + + return serve() + }) + server.listen(argv.port) + const wss = new WebSocketServer({ port: argv.wsPort }) + wss.on("connection", (ws) => connections.push(ws)) + console.log( + chalk.cyan( + `Started a Quartz server listening at http://localhost:${argv.port}${argv.baseDir}`, + ), + ) + console.log("hint: exit with ctrl+c") + chokidar + .watch(["**/*.ts", "**/*.tsx", "**/*.scss", "package.json"], { + ignoreInitial: true, + }) + .on("all", async () => { + build(clientRefresh) + }) + } else { + await build(() => {}) + ctx.dispose() + } +} + +/** + * Handles `npx quartz update` + * @param {*} argv arguments for `update` + */ +export async function handleUpdate(argv) { + const contentFolder = path.join(cwd, argv.directory) + console.log(chalk.bgGreen.black(`\n Quartz v${version} \n`)) + console.log("Backing up your content") + execSync( + `git remote show upstream || git remote add upstream https://github.com/jackyzha0/quartz.git`, + ) + await stashContentFolder(contentFolder) + console.log( + "Pulling updates... you may need to resolve some `git` conflicts if you've made changes to components or plugins.", + ) + gitPull(UPSTREAM_NAME, QUARTZ_SOURCE_BRANCH) + await popContentFolder(contentFolder) + console.log("Ensuring dependencies are up to date") + spawnSync("npm", ["i"], { stdio: "inherit" }) + console.log(chalk.green("Done!")) +} + +/** + * Handles `npx quartz restore` + * @param {*} argv arguments for `restore` + */ +export async function handleRestore(argv) { + const contentFolder = path.join(cwd, argv.directory) + await popContentFolder(contentFolder) +} + +/** + * Handles `npx quartz sync` + * @param {*} argv arguments for `sync` + */ +export async function handleSync(argv) { + const contentFolder = path.join(cwd, argv.directory) + console.log(chalk.bgGreen.black(`\n Quartz v${version} \n`)) + console.log("Backing up your content") + + if (argv.commit) { + const contentStat = await fs.promises.lstat(contentFolder) + if (contentStat.isSymbolicLink()) { + const linkTarg = await fs.promises.readlink(contentFolder) + console.log(chalk.yellow("Detected symlink, trying to dereference before committing")) + + // stash symlink file + await stashContentFolder(contentFolder) + + // follow symlink and copy content + await fs.promises.cp(linkTarg, contentFolder, { + recursive: true, + preserveTimestamps: true, + }) + } + + const currentTimestamp = new Date().toLocaleString("en-US", { + dateStyle: "medium", + timeStyle: "short", + }) + spawnSync("git", ["add", "."], { stdio: "inherit" }) + spawnSync("git", ["commit", "-m", `Quartz sync: ${currentTimestamp}`], { stdio: "inherit" }) + + if (contentStat.isSymbolicLink()) { + // put symlink back + await popContentFolder(contentFolder) + } + } + + await stashContentFolder(contentFolder) + + if (argv.pull) { + console.log( + "Pulling updates from your repository. You may need to resolve some `git` conflicts if you've made changes to components or plugins.", + ) + gitPull(ORIGIN_NAME, QUARTZ_SOURCE_BRANCH) + } + + await popContentFolder(contentFolder) + if (argv.push) { + console.log("Pushing your changes") + spawnSync("git", ["push", "-f", ORIGIN_NAME, QUARTZ_SOURCE_BRANCH], { stdio: "inherit" }) + } + + console.log(chalk.green("Done!")) +} diff --git a/quartz/cli/helpers.js b/quartz/cli/helpers.js new file mode 100644 index 0000000..b07d19e --- /dev/null +++ b/quartz/cli/helpers.js @@ -0,0 +1,52 @@ +import { isCancel, outro } from "@clack/prompts" +import chalk from "chalk" +import { contentCacheFolder } from "./constants.js" +import { spawnSync } from "child_process" +import fs from "fs" + +export function escapePath(fp) { + return fp + .replace(/\\ /g, " ") // unescape spaces + .replace(/^".*"$/, "$1") + .replace(/^'.*"$/, "$1") + .trim() +} + +export function exitIfCancel(val) { + if (isCancel(val)) { + outro(chalk.red("Exiting")) + process.exit(0) + } else { + return val + } +} + +export async function stashContentFolder(contentFolder) { + await fs.promises.rm(contentCacheFolder, { force: true, recursive: true }) + await fs.promises.cp(contentFolder, contentCacheFolder, { + force: true, + recursive: true, + verbatimSymlinks: true, + preserveTimestamps: true, + }) + await fs.promises.rm(contentFolder, { force: true, recursive: true }) +} + +export function gitPull(origin, branch) { + const flags = ["--no-rebase", "--autostash", "-s", "recursive", "-X", "ours", "--no-edit"] + const out = spawnSync("git", ["pull", ...flags, origin, branch], { stdio: "inherit" }) + if (out.stderr) { + throw new Error(`Error while pulling updates: ${out.stderr}`) + } +} + +export async function popContentFolder(contentFolder) { + await fs.promises.rm(contentFolder, { force: true, recursive: true }) + await fs.promises.cp(contentCacheFolder, contentFolder, { + force: true, + recursive: true, + verbatimSymlinks: true, + preserveTimestamps: true, + }) + await fs.promises.rm(contentCacheFolder, { force: true, recursive: true }) +} diff --git a/quartz/components/ArticleTitle.tsx b/quartz/components/ArticleTitle.tsx new file mode 100644 index 0000000..a52b2a4 --- /dev/null +++ b/quartz/components/ArticleTitle.tsx @@ -0,0 +1,17 @@ +import { QuartzComponentConstructor, QuartzComponentProps } from "./types" + +function ArticleTitle({ fileData, displayClass }: QuartzComponentProps) { + const title = fileData.frontmatter?.title + if (title) { + return

{title}

+ } else { + return null + } +} +ArticleTitle.css = ` +.article-title { + margin: 2rem 0 0 0; +} +` + +export default (() => ArticleTitle) satisfies QuartzComponentConstructor diff --git a/quartz/components/Backlinks.tsx b/quartz/components/Backlinks.tsx new file mode 100644 index 0000000..c4172ce --- /dev/null +++ b/quartz/components/Backlinks.tsx @@ -0,0 +1,29 @@ +import { QuartzComponentConstructor, QuartzComponentProps } from "./types" +import style from "./styles/backlinks.scss" +import { resolveRelative, simplifySlug } from "../util/path" + +function Backlinks({ fileData, allFiles, displayClass }: QuartzComponentProps) { + const slug = simplifySlug(fileData.slug!) + const backlinkFiles = allFiles.filter((file) => file.links?.includes(slug)) + return ( + + ) +} + +Backlinks.css = style +export default (() => Backlinks) satisfies QuartzComponentConstructor diff --git a/quartz/components/Body.tsx b/quartz/components/Body.tsx new file mode 100644 index 0000000..fbb8572 --- /dev/null +++ b/quartz/components/Body.tsx @@ -0,0 +1,13 @@ +// @ts-ignore +import clipboardScript from "./scripts/clipboard.inline" +import clipboardStyle from "./styles/clipboard.scss" +import { QuartzComponentConstructor, QuartzComponentProps } from "./types" + +function Body({ children }: QuartzComponentProps) { + return
{children}
+} + +Body.afterDOMLoaded = clipboardScript +Body.css = clipboardStyle + +export default (() => Body) satisfies QuartzComponentConstructor diff --git a/quartz/components/Breadcrumbs.tsx b/quartz/components/Breadcrumbs.tsx new file mode 100644 index 0000000..467b5a5 --- /dev/null +++ b/quartz/components/Breadcrumbs.tsx @@ -0,0 +1,117 @@ +import { QuartzComponentConstructor, QuartzComponentProps } from "./types" +import breadcrumbsStyle from "./styles/breadcrumbs.scss" +import { FullSlug, SimpleSlug, resolveRelative } from "../util/path" +import { QuartzPluginData } from "../plugins/vfile" + +type CrumbData = { + displayName: string + path: string +} + +interface BreadcrumbOptions { + /** + * Symbol between crumbs + */ + spacerSymbol: string + /** + * Name of first crumb + */ + rootName: string + /** + * wether to look up frontmatter title for folders (could cause performance problems with big vaults) + */ + resolveFrontmatterTitle: boolean + /** + * Wether to display breadcrumbs on root `index.md` + */ + hideOnRoot: boolean +} + +const defaultOptions: BreadcrumbOptions = { + spacerSymbol: ">", + rootName: "Home", + resolveFrontmatterTitle: false, + hideOnRoot: true, +} + +function formatCrumb(displayName: string, baseSlug: FullSlug, currentSlug: SimpleSlug): CrumbData { + return { + displayName: displayName.replaceAll("-", " "), + path: resolveRelative(baseSlug, currentSlug), + } +} + +// given a folderName (e.g. "features"), search for the corresponding `index.md` file +function findCurrentFile(allFiles: QuartzPluginData[], folderName: string) { + return allFiles.find((file) => { + if (file.slug?.endsWith("index")) { + const folderParts = file.filePath?.split("/") + if (folderParts) { + const name = folderParts[folderParts?.length - 2] + if (name === folderName) { + return true + } + } + } + }) +} + +export default ((opts?: Partial) => { + // Merge options with defaults + const options: BreadcrumbOptions = { ...defaultOptions, ...opts } + + function Breadcrumbs({ fileData, allFiles, displayClass }: QuartzComponentProps) { + // Hide crumbs on root if enabled + if (options.hideOnRoot && fileData.slug === "index") { + return <> + } + + // Format entry for root element + const firstEntry = formatCrumb(options.rootName, fileData.slug!, "/" as SimpleSlug) + const crumbs: CrumbData[] = [firstEntry] + + // Split slug into hierarchy/parts + const slugParts = fileData.slug?.split("/") + if (slugParts) { + // full path until current part + let currentPath = "" + for (let i = 0; i < slugParts.length - 1; i++) { + let currentTitle = slugParts[i] + + // TODO: performance optimizations/memoizing + // Try to resolve frontmatter folder title + if (options?.resolveFrontmatterTitle) { + // try to find file for current path + const currentFile = findCurrentFile(allFiles, currentTitle) + if (currentFile) { + currentTitle = currentFile.frontmatter!.title + } + } + // Add current slug to full path + currentPath += slugParts[i] + "/" + + // Format and add current crumb + const crumb = formatCrumb(currentTitle, fileData.slug!, currentPath as SimpleSlug) + crumbs.push(crumb) + } + + // Add current file to crumb (can directly use frontmatter title) + crumbs.push({ + displayName: fileData.frontmatter!.title, + path: "", + }) + } + return ( + + ) + } + Breadcrumbs.css = breadcrumbsStyle + return Breadcrumbs +}) satisfies QuartzComponentConstructor diff --git a/quartz/components/ContentMeta.tsx b/quartz/components/ContentMeta.tsx new file mode 100644 index 0000000..21dc13f --- /dev/null +++ b/quartz/components/ContentMeta.tsx @@ -0,0 +1,30 @@ +import { formatDate, getDate } from "./Date" +import { QuartzComponentConstructor, QuartzComponentProps } from "./types" +import readingTime from "reading-time" + +export default (() => { + function ContentMetadata({ cfg, fileData, displayClass }: QuartzComponentProps) { + const text = fileData.text + if (text) { + const segments: string[] = [] + const { text: timeTaken, words: _words } = readingTime(text) + + if (fileData.dates) { + segments.push(formatDate(getDate(cfg, fileData)!)) + } + + segments.push(timeTaken) + return

{segments.join(", ")}

+ } else { + return null + } + } + + ContentMetadata.css = ` + .content-meta { + margin-top: 0; + color: var(--gray); + } + ` + return ContentMetadata +}) satisfies QuartzComponentConstructor diff --git a/quartz/components/Darkmode.tsx b/quartz/components/Darkmode.tsx new file mode 100644 index 0000000..a668c5b --- /dev/null +++ b/quartz/components/Darkmode.tsx @@ -0,0 +1,51 @@ +// @ts-ignore: this is safe, we don't want to actually make darkmode.inline.ts a module as +// modules are automatically deferred and we don't want that to happen for critical beforeDOMLoads +// see: https://v8.dev/features/modules#defer +import darkmodeScript from "./scripts/darkmode.inline" +import styles from "./styles/darkmode.scss" +import { QuartzComponentConstructor, QuartzComponentProps } from "./types" + +function Darkmode({ displayClass }: QuartzComponentProps) { + return ( +
+ + + +
+ ) +} + +Darkmode.beforeDOMLoaded = darkmodeScript +Darkmode.css = styles + +export default (() => Darkmode) satisfies QuartzComponentConstructor diff --git a/quartz/components/Date.tsx b/quartz/components/Date.tsx new file mode 100644 index 0000000..8713cfd --- /dev/null +++ b/quartz/components/Date.tsx @@ -0,0 +1,29 @@ +import { GlobalConfiguration } from "../cfg" +import { QuartzPluginData } from "../plugins/vfile" + +interface Props { + date: Date +} + +export type ValidDateType = keyof Required["dates"] + +export function getDate(cfg: GlobalConfiguration, data: QuartzPluginData): Date | undefined { + if (!cfg.defaultDateType) { + throw new Error( + `Field 'defaultDateType' was not set in the configuration object of quartz.config.ts. See https://quartz.jzhao.xyz/configuration#general-configuration for more details.`, + ) + } + return data.dates?.[cfg.defaultDateType] +} + +export function formatDate(d: Date): string { + return d.toLocaleDateString("en-US", { + year: "numeric", + month: "short", + day: "2-digit", + }) +} + +export function Date({ date }: Props) { + return <>{formatDate(date)} +} diff --git a/quartz/components/DesktopOnly.tsx b/quartz/components/DesktopOnly.tsx new file mode 100644 index 0000000..a11f23f --- /dev/null +++ b/quartz/components/DesktopOnly.tsx @@ -0,0 +1,18 @@ +import { QuartzComponent, QuartzComponentConstructor, QuartzComponentProps } from "./types" + +export default ((component?: QuartzComponent) => { + if (component) { + const Component = component + function DesktopOnly(props: QuartzComponentProps) { + return + } + + DesktopOnly.displayName = component.displayName + DesktopOnly.afterDOMLoaded = component?.afterDOMLoaded + DesktopOnly.beforeDOMLoaded = component?.beforeDOMLoaded + DesktopOnly.css = component?.css + return DesktopOnly + } else { + return () => <> + } +}) satisfies QuartzComponentConstructor diff --git a/quartz/components/Explorer.tsx b/quartz/components/Explorer.tsx new file mode 100644 index 0000000..95eac43 --- /dev/null +++ b/quartz/components/Explorer.tsx @@ -0,0 +1,126 @@ +import { QuartzComponentConstructor, QuartzComponentProps } from "./types" +import explorerStyle from "./styles/explorer.scss" + +// @ts-ignore +import script from "./scripts/explorer.inline" +import { ExplorerNode, FileNode, Options } from "./ExplorerNode" +import { QuartzPluginData } from "../plugins/vfile" + +// Options interface defined in `ExplorerNode` to avoid circular dependency +const defaultOptions = { + title: "Explorer", + folderClickBehavior: "collapse", + folderDefaultState: "collapsed", + useSavedState: true, + sortFn: (a, b) => { + // Sort order: folders first, then files. Sort folders and files alphabetically + if ((!a.file && !b.file) || (a.file && b.file)) { + // numeric: true: Whether numeric collation should be used, such that "1" < "2" < "10" + // sensitivity: "base": Only strings that differ in base letters compare as unequal. Examples: a ≠ b, a = á, a = A + return a.displayName.localeCompare(b.displayName, undefined, { + numeric: true, + sensitivity: "base", + }) + } + if (a.file && !b.file) { + return 1 + } else { + return -1 + } + }, + filterFn: (node) => node.name !== "tags", + order: ["filter", "map", "sort"], +} satisfies Options + +export default ((userOpts?: Partial) => { + // Parse config + const opts: Options = { ...defaultOptions, ...userOpts } + + // memoized + let fileTree: FileNode + let jsonTree: string + + function constructFileTree(allFiles: QuartzPluginData[]) { + if (!fileTree) { + // Construct tree from allFiles + fileTree = new FileNode("") + allFiles.forEach((file) => fileTree.add(file, 1)) + + /** + * Keys of this object must match corresponding function name of `FileNode`, + * while values must be the argument that will be passed to the function. + * + * e.g. entry for FileNode.sort: `sort: opts.sortFn` (value is sort function from options) + */ + const functions = { + map: opts.mapFn, + sort: opts.sortFn, + filter: opts.filterFn, + } + + // Execute all functions (sort, filter, map) that were provided (if none were provided, only default "sort" is applied) + if (opts.order) { + // Order is important, use loop with index instead of order.map() + for (let i = 0; i < opts.order.length; i++) { + const functionName = opts.order[i] + if (functions[functionName]) { + // for every entry in order, call matching function in FileNode and pass matching argument + // e.g. i = 0; functionName = "filter" + // converted to: (if opts.filterFn) => fileTree.filter(opts.filterFn) + + // @ts-ignore + // typescript cant statically check these dynamic references, so manually make sure reference is valid and ignore warning + fileTree[functionName].call(fileTree, functions[functionName]) + } + } + } + + // Get all folders of tree. Initialize with collapsed state + const folders = fileTree.getFolderPaths(opts.folderDefaultState === "collapsed") + + // Stringify to pass json tree as data attribute ([data-tree]) + jsonTree = JSON.stringify(folders) + } + } + + function Explorer({ allFiles, displayClass, fileData }: QuartzComponentProps) { + constructFileTree(allFiles) + return ( +
+ +
+
    + +
  • +
+
+
+ ) + } + Explorer.css = explorerStyle + Explorer.afterDOMLoaded = script + return Explorer +}) satisfies QuartzComponentConstructor diff --git a/quartz/components/ExplorerNode.tsx b/quartz/components/ExplorerNode.tsx new file mode 100644 index 0000000..e5ceb0b --- /dev/null +++ b/quartz/components/ExplorerNode.tsx @@ -0,0 +1,224 @@ +// @ts-ignore +import { QuartzPluginData } from "../plugins/vfile" +import { resolveRelative } from "../util/path" + +type OrderEntries = "sort" | "filter" | "map" + +export interface Options { + title: string + folderDefaultState: "collapsed" | "open" + folderClickBehavior: "collapse" | "link" + useSavedState: boolean + sortFn: (a: FileNode, b: FileNode) => number + filterFn?: (node: FileNode) => boolean + mapFn?: (node: FileNode) => void + order?: OrderEntries[] +} + +type DataWrapper = { + file: QuartzPluginData + path: string[] +} + +export type FolderState = { + path: string + collapsed: boolean +} + +// Structure to add all files into a tree +export class FileNode { + children: FileNode[] + name: string + displayName: string + file: QuartzPluginData | null + depth: number + + constructor(name: string, file?: QuartzPluginData, depth?: number) { + this.children = [] + this.name = name + this.displayName = name + this.file = file ? structuredClone(file) : null + this.depth = depth ?? 0 + } + + private insert(file: DataWrapper) { + if (file.path.length === 1) { + if (file.path[0] !== "index.md") { + this.children.push(new FileNode(file.file.frontmatter!.title, file.file, this.depth + 1)) + } else { + const title = file.file.frontmatter?.title + if (title && title !== "index" && file.path[0] === "index.md") { + this.displayName = title + } + } + } else { + const next = file.path[0] + file.path = file.path.splice(1) + for (const child of this.children) { + if (child.name === next) { + child.insert(file) + return + } + } + + const newChild = new FileNode(next, undefined, this.depth + 1) + newChild.insert(file) + this.children.push(newChild) + } + } + + // Add new file to tree + add(file: QuartzPluginData, splice: number = 0) { + this.insert({ file, path: file.filePath!.split("/").splice(splice) }) + } + + // Print tree structure (for debugging) + print(depth: number = 0) { + let folderChar = "" + if (!this.file) folderChar = "|" + console.log("-".repeat(depth), folderChar, this.name, this.depth) + this.children.forEach((e) => e.print(depth + 1)) + } + + /** + * Filter FileNode tree. Behaves similar to `Array.prototype.filter()`, but modifies tree in place + * @param filterFn function to filter tree with + */ + filter(filterFn: (node: FileNode) => boolean) { + this.children = this.children.filter(filterFn) + this.children.forEach((child) => child.filter(filterFn)) + } + + /** + * Filter FileNode tree. Behaves similar to `Array.prototype.map()`, but modifies tree in place + * @param mapFn function to use for mapping over tree + */ + map(mapFn: (node: FileNode) => void) { + mapFn(this) + + this.children.forEach((child) => child.map(mapFn)) + } + + /** + * Get folder representation with state of tree. + * Intended to only be called on root node before changes to the tree are made + * @param collapsed default state of folders (collapsed by default or not) + * @returns array containing folder state for tree + */ + getFolderPaths(collapsed: boolean): FolderState[] { + const folderPaths: FolderState[] = [] + + const traverse = (node: FileNode, currentPath: string) => { + if (!node.file) { + const folderPath = currentPath + (currentPath ? "/" : "") + node.name + if (folderPath !== "") { + folderPaths.push({ path: folderPath, collapsed }) + } + node.children.forEach((child) => traverse(child, folderPath)) + } + } + + traverse(this, "") + + return folderPaths + } + + // Sort order: folders first, then files. Sort folders and files alphabetically + /** + * Sorts tree according to sort/compare function + * @param sortFn compare function used for `.sort()`, also used recursively for children + */ + sort(sortFn: (a: FileNode, b: FileNode) => number) { + this.children = this.children.sort(sortFn) + this.children.forEach((e) => e.sort(sortFn)) + } +} + +type ExplorerNodeProps = { + node: FileNode + opts: Options + fileData: QuartzPluginData + fullPath?: string +} + +export function ExplorerNode({ node, opts, fullPath, fileData }: ExplorerNodeProps) { + // Get options + const folderBehavior = opts.folderClickBehavior + const isDefaultOpen = opts.folderDefaultState === "open" + + // Calculate current folderPath + let pathOld = fullPath ? fullPath : "" + let folderPath = "" + if (node.name !== "") { + folderPath = `${pathOld}/${node.name}` + } + + return ( +
  • + {node.file ? ( + // Single file node +
  • + + {node.displayName} + +
  • + ) : ( +
    + {node.name !== "" && ( + // Node with entire folder + // Render svg button + folder name, then children + +
    + )} + {/* Recursively render children of folder */} +
    +
      + {node.children.map((childNode, i) => ( + + ))} +
    +
    + + )} + + ) +} diff --git a/quartz/components/Footer.tsx b/quartz/components/Footer.tsx new file mode 100644 index 0000000..54440cf --- /dev/null +++ b/quartz/components/Footer.tsx @@ -0,0 +1,32 @@ +import { QuartzComponentConstructor, QuartzComponentProps } from "./types" +import style from "./styles/footer.scss" +import { version } from "../../package.json" + +interface Options { + links: Record +} + +export default ((opts?: Options) => { + function Footer({ displayClass }: QuartzComponentProps) { + const year = new Date().getFullYear() + const links = opts?.links ?? [] + return ( +
    +
    +

    + Created with Quartz v{version}, © {year} +

    +
      + {Object.entries(links).map(([text, link]) => ( +
    • + {text} +
    • + ))} +
    +
    + ) + } + + Footer.css = style + return Footer +}) satisfies QuartzComponentConstructor diff --git a/quartz/components/Graph.tsx b/quartz/components/Graph.tsx new file mode 100644 index 0000000..0c2647f --- /dev/null +++ b/quartz/components/Graph.tsx @@ -0,0 +1,100 @@ +import { QuartzComponentConstructor, QuartzComponentProps } from "./types" +// @ts-ignore +import script from "./scripts/graph.inline" +import style from "./styles/graph.scss" + +export interface D3Config { + drag: boolean + zoom: boolean + depth: number + scale: number + repelForce: number + centerForce: number + linkDistance: number + fontSize: number + opacityScale: number + removeTags: string[] + showTags: boolean +} + +interface GraphOptions { + localGraph: Partial | undefined + globalGraph: Partial | undefined +} + +const defaultOptions: GraphOptions = { + localGraph: { + drag: true, + zoom: true, + depth: 1, + scale: 1.1, + repelForce: 0.5, + centerForce: 0.3, + linkDistance: 30, + fontSize: 0.6, + opacityScale: 1, + showTags: true, + removeTags: [], + }, + globalGraph: { + drag: true, + zoom: true, + depth: -1, + scale: 0.9, + repelForce: 0.5, + centerForce: 0.3, + linkDistance: 30, + fontSize: 0.6, + opacityScale: 1, + showTags: true, + removeTags: [], + }, +} + +export default ((opts?: GraphOptions) => { + function Graph({ displayClass }: QuartzComponentProps) { + const localGraph = { ...defaultOptions.localGraph, ...opts?.localGraph } + const globalGraph = { ...defaultOptions.globalGraph, ...opts?.globalGraph } + return ( +
    +

    Graph View

    +
    +
    + + + +
    +
    +
    +
    +
    + ) + } + + Graph.css = style + Graph.afterDOMLoaded = script + + return Graph +}) satisfies QuartzComponentConstructor diff --git a/quartz/components/Head.tsx b/quartz/components/Head.tsx new file mode 100644 index 0000000..2bf2638 --- /dev/null +++ b/quartz/components/Head.tsx @@ -0,0 +1,44 @@ +import { FullSlug, _stripSlashes, joinSegments, pathToRoot } from "../util/path" +import { JSResourceToScriptElement } from "../util/resources" +import { QuartzComponentConstructor, QuartzComponentProps } from "./types" + +export default (() => { + function Head({ cfg, fileData, externalResources }: QuartzComponentProps) { + const title = fileData.frontmatter?.title ?? "Untitled" + const description = fileData.description?.trim() ?? "No description provided" + const { css, js } = externalResources + + const url = new URL(`https://${cfg.baseUrl ?? "example.com"}`) + const path = url.pathname as FullSlug + const baseDir = fileData.slug === "404" ? path : pathToRoot(fileData.slug!) + + const iconPath = joinSegments(baseDir, "static/icon.png") + const ogImagePath = `https://${cfg.baseUrl}/static/og-image.png` + + return ( + + {title} + + + + + {cfg.baseUrl && } + + + + + + + + {css.map((href) => ( + + ))} + {js + .filter((resource) => resource.loadTime === "beforeDOMReady") + .map((res) => JSResourceToScriptElement(res, true))} + + ) + } + + return Head +}) satisfies QuartzComponentConstructor diff --git a/quartz/components/Header.tsx b/quartz/components/Header.tsx new file mode 100644 index 0000000..5281f72 --- /dev/null +++ b/quartz/components/Header.tsx @@ -0,0 +1,22 @@ +import { QuartzComponentConstructor, QuartzComponentProps } from "./types" + +function Header({ children }: QuartzComponentProps) { + return children.length > 0 ?
    {children}
    : null +} + +Header.css = ` +header { + display: flex; + flex-direction: row; + align-items: center; + margin: 2rem 0; + gap: 1.5rem; +} + +header h1 { + margin: 0; + flex: auto; +} +` + +export default (() => Header) satisfies QuartzComponentConstructor diff --git a/quartz/components/MobileOnly.tsx b/quartz/components/MobileOnly.tsx new file mode 100644 index 0000000..5a19095 --- /dev/null +++ b/quartz/components/MobileOnly.tsx @@ -0,0 +1,18 @@ +import { QuartzComponent, QuartzComponentConstructor, QuartzComponentProps } from "./types" + +export default ((component?: QuartzComponent) => { + if (component) { + const Component = component + function MobileOnly(props: QuartzComponentProps) { + return + } + + MobileOnly.displayName = component.displayName + MobileOnly.afterDOMLoaded = component?.afterDOMLoaded + MobileOnly.beforeDOMLoaded = component?.beforeDOMLoaded + MobileOnly.css = component?.css + return MobileOnly + } else { + return () => <> + } +}) satisfies QuartzComponentConstructor diff --git a/quartz/components/PageList.tsx b/quartz/components/PageList.tsx new file mode 100644 index 0000000..eb34f02 --- /dev/null +++ b/quartz/components/PageList.tsx @@ -0,0 +1,87 @@ +import { FullSlug, resolveRelative } from "../util/path" +import { QuartzPluginData } from "../plugins/vfile" +import { Date, getDate } from "./Date" +import { QuartzComponentProps } from "./types" +import { GlobalConfiguration } from "../cfg" + +export function byDateAndAlphabetical( + cfg: GlobalConfiguration, +): (f1: QuartzPluginData, f2: QuartzPluginData) => number { + return (f1, f2) => { + if (f1.dates && f2.dates) { + // sort descending + return getDate(cfg, f2)!.getTime() - getDate(cfg, f1)!.getTime() + } else if (f1.dates && !f2.dates) { + // prioritize files with dates + return -1 + } else if (!f1.dates && f2.dates) { + return 1 + } + + // otherwise, sort lexographically by title + const f1Title = f1.frontmatter?.title.toLowerCase() ?? "" + const f2Title = f2.frontmatter?.title.toLowerCase() ?? "" + return f1Title.localeCompare(f2Title) + } +} + +type Props = { + limit?: number +} & QuartzComponentProps + +export function PageList({ cfg, fileData, allFiles, limit }: Props) { + let list = allFiles.sort(byDateAndAlphabetical(cfg)) + if (limit) { + list = list.slice(0, limit) + } + + return ( +
      + {list.map((page) => { + const title = page.frontmatter?.title + const tags = page.frontmatter?.tags ?? [] + + return ( +
    • +
      + {page.dates && ( +

      + +

      + )} + + +
      +
    • + ) + })} +
    + ) +} + +PageList.css = ` +.section h3 { + margin: 0; +} + +.section > .tags { + margin: 0; +} +` diff --git a/quartz/components/PageTitle.tsx b/quartz/components/PageTitle.tsx new file mode 100644 index 0000000..81d80d1 --- /dev/null +++ b/quartz/components/PageTitle.tsx @@ -0,0 +1,20 @@ +import { pathToRoot } from "../util/path" +import { QuartzComponentConstructor, QuartzComponentProps } from "./types" + +function PageTitle({ fileData, cfg, displayClass }: QuartzComponentProps) { + const title = cfg?.pageTitle ?? "Untitled Quartz" + const baseDir = pathToRoot(fileData.slug!) + return ( +

    + {title} +

    + ) +} + +PageTitle.css = ` +.page-title { + margin: 0; +} +` + +export default (() => PageTitle) satisfies QuartzComponentConstructor diff --git a/quartz/components/RecentNotes.tsx b/quartz/components/RecentNotes.tsx new file mode 100644 index 0000000..63f040d --- /dev/null +++ b/quartz/components/RecentNotes.tsx @@ -0,0 +1,81 @@ +import { QuartzComponentConstructor, QuartzComponentProps } from "./types" +import { FullSlug, SimpleSlug, resolveRelative } from "../util/path" +import { QuartzPluginData } from "../plugins/vfile" +import { byDateAndAlphabetical } from "./PageList" +import style from "./styles/recentNotes.scss" +import { Date, getDate } from "./Date" +import { GlobalConfiguration } from "../cfg" + +interface Options { + title: string + limit: number + linkToMore: SimpleSlug | false + filter: (f: QuartzPluginData) => boolean + sort: (f1: QuartzPluginData, f2: QuartzPluginData) => number +} + +const defaultOptions = (cfg: GlobalConfiguration): Options => ({ + title: "Recent Notes", + limit: 3, + linkToMore: false, + filter: () => true, + sort: byDateAndAlphabetical(cfg), +}) + +export default ((userOpts?: Partial) => { + function RecentNotes({ allFiles, fileData, displayClass, cfg }: QuartzComponentProps) { + const opts = { ...defaultOptions(cfg), ...userOpts } + const pages = allFiles.filter(opts.filter).sort(opts.sort) + const remaining = Math.max(0, pages.length - opts.limit) + return ( +
    +

    {opts.title}

    +
      + {pages.slice(0, opts.limit).map((page) => { + const title = page.frontmatter?.title + const tags = page.frontmatter?.tags ?? [] + + return ( +
    • +
      + + {page.dates && ( +

      + +

      + )} + +
      +
    • + ) + })} +
    + {opts.linkToMore && remaining > 0 && ( +

    + See {remaining} more → +

    + )} +
    + ) + } + + RecentNotes.css = style + return RecentNotes +}) satisfies QuartzComponentConstructor diff --git a/quartz/components/Search.tsx b/quartz/components/Search.tsx new file mode 100644 index 0000000..9c1852d --- /dev/null +++ b/quartz/components/Search.tsx @@ -0,0 +1,49 @@ +import { QuartzComponentConstructor, QuartzComponentProps } from "./types" +import style from "./styles/search.scss" +// @ts-ignore +import script from "./scripts/search.inline" + +export default (() => { + function Search({ displayClass }: QuartzComponentProps) { + return ( +
    +
    +

    Search

    +
    + + Search + Search + + + + + +
    +
    +
    + +
    +
    +
    +
    + ) + } + + Search.afterDOMLoaded = script + Search.css = style + + return Search +}) satisfies QuartzComponentConstructor diff --git a/quartz/components/Spacer.tsx b/quartz/components/Spacer.tsx new file mode 100644 index 0000000..8359111 --- /dev/null +++ b/quartz/components/Spacer.tsx @@ -0,0 +1,7 @@ +import { QuartzComponentConstructor, QuartzComponentProps } from "./types" + +function Spacer({ displayClass }: QuartzComponentProps) { + return
    +} + +export default (() => Spacer) satisfies QuartzComponentConstructor diff --git a/quartz/components/TableOfContents.tsx b/quartz/components/TableOfContents.tsx new file mode 100644 index 0000000..3847726 --- /dev/null +++ b/quartz/components/TableOfContents.tsx @@ -0,0 +1,84 @@ +import { QuartzComponentConstructor, QuartzComponentProps } from "./types" +import legacyStyle from "./styles/legacyToc.scss" +import modernStyle from "./styles/toc.scss" + +// @ts-ignore +import script from "./scripts/toc.inline" + +interface Options { + layout: "modern" | "legacy" +} + +const defaultOptions: Options = { + layout: "modern", +} + +function TableOfContents({ fileData, displayClass }: QuartzComponentProps) { + if (!fileData.toc) { + return null + } + + return ( +
    + +
    + +
    +
    + ) +} +TableOfContents.css = modernStyle +TableOfContents.afterDOMLoaded = script + +function LegacyTableOfContents({ fileData }: QuartzComponentProps) { + if (!fileData.toc) { + return null + } + + return ( +
    + +

    Table of Contents

    +
    + +
    + ) +} +LegacyTableOfContents.css = legacyStyle + +export default ((opts?: Partial) => { + const layout = opts?.layout ?? defaultOptions.layout + return layout === "modern" ? TableOfContents : LegacyTableOfContents +}) satisfies QuartzComponentConstructor diff --git a/quartz/components/TagList.tsx b/quartz/components/TagList.tsx new file mode 100644 index 0000000..c763b14 --- /dev/null +++ b/quartz/components/TagList.tsx @@ -0,0 +1,53 @@ +import { pathToRoot, slugTag } from "../util/path" +import { QuartzComponentConstructor, QuartzComponentProps } from "./types" + +function TagList({ fileData, displayClass }: QuartzComponentProps) { + const tags = fileData.frontmatter?.tags + const baseDir = pathToRoot(fileData.slug!) + if (tags && tags.length > 0) { + return ( +
      + {tags.map((tag) => { + const display = `#${tag}` + const linkDest = baseDir + `/tags/${slugTag(tag)}` + return ( +
    • + + {display} + +
    • + ) + })} +
    + ) + } else { + return null + } +} + +TagList.css = ` +.tags { + list-style: none; + display:flex; + flex-wrap: wrap; + padding-left: 0; + gap: 0.4rem; + margin: 1rem 0; +} + +.tags > li { + display: inline-block; + white-space: nowrap; + margin: 0; + overflow-wrap: normal; +} + +a.tag-link { + border-radius: 8px; + background-color: var(--highlight); + padding: 0.2rem 0.4rem; + margin: 0 0.1rem; +} +` + +export default (() => TagList) satisfies QuartzComponentConstructor diff --git a/quartz/components/index.ts b/quartz/components/index.ts new file mode 100644 index 0000000..b3db76b --- /dev/null +++ b/quartz/components/index.ts @@ -0,0 +1,45 @@ +import Content from "./pages/Content" +import TagContent from "./pages/TagContent" +import FolderContent from "./pages/FolderContent" +import NotFound from "./pages/404" +import ArticleTitle from "./ArticleTitle" +import Darkmode from "./Darkmode" +import Head from "./Head" +import PageTitle from "./PageTitle" +import ContentMeta from "./ContentMeta" +import Spacer from "./Spacer" +import TableOfContents from "./TableOfContents" +import Explorer from "./Explorer" +import TagList from "./TagList" +import Graph from "./Graph" +import Backlinks from "./Backlinks" +import Search from "./Search" +import Footer from "./Footer" +import DesktopOnly from "./DesktopOnly" +import MobileOnly from "./MobileOnly" +import RecentNotes from "./RecentNotes" +import Breadcrumbs from "./Breadcrumbs" + +export { + ArticleTitle, + Content, + TagContent, + FolderContent, + Darkmode, + Head, + PageTitle, + ContentMeta, + Spacer, + TableOfContents, + Explorer, + TagList, + Graph, + Backlinks, + Search, + Footer, + DesktopOnly, + MobileOnly, + RecentNotes, + NotFound, + Breadcrumbs, +} diff --git a/quartz/components/pages/404.tsx b/quartz/components/pages/404.tsx new file mode 100644 index 0000000..c276f56 --- /dev/null +++ b/quartz/components/pages/404.tsx @@ -0,0 +1,12 @@ +import { QuartzComponentConstructor } from "../types" + +function NotFound() { + return ( +
    +

    404

    +

    Either this page is private or doesn't exist.

    +
    + ) +} + +export default (() => NotFound) satisfies QuartzComponentConstructor diff --git a/quartz/components/pages/Content.tsx b/quartz/components/pages/Content.tsx new file mode 100644 index 0000000..7490a7e --- /dev/null +++ b/quartz/components/pages/Content.tsx @@ -0,0 +1,11 @@ +import { QuartzComponentConstructor, QuartzComponentProps } from "../types" +import { Fragment, jsx, jsxs } from "preact/jsx-runtime" +import { toJsxRuntime } from "hast-util-to-jsx-runtime" + +function Content({ tree }: QuartzComponentProps) { + // @ts-ignore (preact makes it angry) + const content = toJsxRuntime(tree, { Fragment, jsx, jsxs, elementAttributeNameCase: "html" }) + return
    {content}
    +} + +export default (() => Content) satisfies QuartzComponentConstructor diff --git a/quartz/components/pages/FolderContent.tsx b/quartz/components/pages/FolderContent.tsx new file mode 100644 index 0000000..a766d4b --- /dev/null +++ b/quartz/components/pages/FolderContent.tsx @@ -0,0 +1,49 @@ +import { QuartzComponentConstructor, QuartzComponentProps } from "../types" +import { Fragment, jsx, jsxs } from "preact/jsx-runtime" +import { toJsxRuntime } from "hast-util-to-jsx-runtime" +import path from "path" + +import style from "../styles/listPage.scss" +import { PageList } from "../PageList" +import { _stripSlashes, simplifySlug } from "../../util/path" +import { Root } from "hast" +import { pluralize } from "../../util/lang" + +function FolderContent(props: QuartzComponentProps) { + const { tree, fileData, allFiles } = props + const folderSlug = _stripSlashes(simplifySlug(fileData.slug!)) + const allPagesInFolder = allFiles.filter((file) => { + const fileSlug = _stripSlashes(simplifySlug(file.slug!)) + const prefixed = fileSlug.startsWith(folderSlug) && fileSlug !== folderSlug + const folderParts = folderSlug.split(path.posix.sep) + const fileParts = fileSlug.split(path.posix.sep) + const isDirectChild = fileParts.length === folderParts.length + 1 + return prefixed && isDirectChild + }) + + const listProps = { + ...props, + allFiles: allPagesInFolder, + } + + const content = + (tree as Root).children.length === 0 + ? fileData.description + : // @ts-ignore + toJsxRuntime(tree, { Fragment, jsx, jsxs, elementAttributeNameCase: "html" }) + + return ( +
    +
    +

    {content}

    +
    +

    {pluralize(allPagesInFolder.length, "item")} under this folder.

    +
    + +
    +
    + ) +} + +FolderContent.css = style + PageList.css +export default (() => FolderContent) satisfies QuartzComponentConstructor diff --git a/quartz/components/pages/TagContent.tsx b/quartz/components/pages/TagContent.tsx new file mode 100644 index 0000000..9907e3f --- /dev/null +++ b/quartz/components/pages/TagContent.tsx @@ -0,0 +1,94 @@ +import { QuartzComponentConstructor, QuartzComponentProps } from "../types" +import { Fragment, jsx, jsxs } from "preact/jsx-runtime" +import { toJsxRuntime } from "hast-util-to-jsx-runtime" +import style from "../styles/listPage.scss" +import { PageList } from "../PageList" +import { FullSlug, getAllSegmentPrefixes, simplifySlug } from "../../util/path" +import { QuartzPluginData } from "../../plugins/vfile" +import { Root } from "hast" +import { pluralize } from "../../util/lang" + +const numPages = 10 +function TagContent(props: QuartzComponentProps) { + const { tree, fileData, allFiles } = props + const slug = fileData.slug + + if (!(slug?.startsWith("tags/") || slug === "tags")) { + throw new Error(`Component "TagContent" tried to render a non-tag page: ${slug}`) + } + + const tag = simplifySlug(slug.slice("tags/".length) as FullSlug) + const allPagesWithTag = (tag: string) => + allFiles.filter((file) => + (file.frontmatter?.tags ?? []).flatMap(getAllSegmentPrefixes).includes(tag), + ) + + const content = + (tree as Root).children.length === 0 + ? fileData.description + : // @ts-ignore + toJsxRuntime(tree, { Fragment, jsx, jsxs, elementAttributeNameCase: "html" }) + + if (tag === "") { + const tags = [...new Set(allFiles.flatMap((data) => data.frontmatter?.tags ?? []))] + const tagItemMap: Map = new Map() + for (const tag of tags) { + tagItemMap.set(tag, allPagesWithTag(tag)) + } + + return ( +
    +
    +

    {content}

    +
    +

    Found {tags.length} total tags.

    +
    + {tags.map((tag) => { + const pages = tagItemMap.get(tag)! + const listProps = { + ...props, + allFiles: pages, + } + + const contentPage = allFiles.filter((file) => file.slug === `tags/${tag}`)[0] + const content = contentPage?.description + return ( +
    +

    + + #{tag} + +

    + {content &&

    {content}

    } +

    + {pluralize(pages.length, "item")} with this tag.{" "} + {pages.length > numPages && `Showing first ${numPages}.`} +

    + +
    + ) + })} +
    +
    + ) + } else { + const pages = allPagesWithTag(tag) + const listProps = { + ...props, + allFiles: pages, + } + + return ( +
    +
    {content}
    +

    {pluralize(pages.length, "item")} with this tag.

    +
    + +
    +
    + ) + } +} + +TagContent.css = style + PageList.css +export default (() => TagContent) satisfies QuartzComponentConstructor diff --git a/quartz/components/renderPage.tsx b/quartz/components/renderPage.tsx new file mode 100644 index 0000000..451813b --- /dev/null +++ b/quartz/components/renderPage.tsx @@ -0,0 +1,154 @@ +import { render } from "preact-render-to-string" +import { QuartzComponent, QuartzComponentProps } from "./types" +import HeaderConstructor from "./Header" +import BodyConstructor from "./Body" +import { JSResourceToScriptElement, StaticResources } from "../util/resources" +import { FullSlug, RelativeURL, joinSegments } from "../util/path" +import { visit } from "unist-util-visit" +import { Root, Element } from "hast" + +interface RenderComponents { + head: QuartzComponent + header: QuartzComponent[] + beforeBody: QuartzComponent[] + pageBody: QuartzComponent + left: QuartzComponent[] + right: QuartzComponent[] + footer: QuartzComponent +} + +export function pageResources( + baseDir: FullSlug | RelativeURL, + staticResources: StaticResources, +): StaticResources { + const contentIndexPath = joinSegments(baseDir, "static/contentIndex.json") + const contentIndexScript = `const fetchData = fetch(\`${contentIndexPath}\`).then(data => data.json())` + + return { + css: [joinSegments(baseDir, "index.css"), ...staticResources.css], + js: [ + { + src: joinSegments(baseDir, "prescript.js"), + loadTime: "beforeDOMReady", + contentType: "external", + }, + { + loadTime: "beforeDOMReady", + contentType: "inline", + spaPreserve: true, + script: contentIndexScript, + }, + ...staticResources.js, + { + src: joinSegments(baseDir, "postscript.js"), + loadTime: "afterDOMReady", + moduleType: "module", + contentType: "external", + }, + ], + } +} + +export function renderPage( + slug: FullSlug, + componentData: QuartzComponentProps, + components: RenderComponents, + pageResources: StaticResources, +): string { + // process transcludes in componentData + visit(componentData.tree as Root, "element", (node, _index, _parent) => { + if (node.tagName === "blockquote") { + const classNames = (node.properties?.className ?? []) as string[] + if (classNames.includes("transclude")) { + const inner = node.children[0] as Element + const blockSlug = inner.properties?.["data-slug"] as FullSlug + const blockRef = node.properties!.dataBlock as string + + // TODO: avoid this expensive find operation and construct an index ahead of time + let blockNode = componentData.allFiles.find((f) => f.slug === blockSlug)?.blocks?.[blockRef] + if (blockNode) { + if (blockNode.tagName === "li") { + blockNode = { + type: "element", + tagName: "ul", + children: [blockNode], + } + } + + node.children = [ + blockNode, + { + type: "element", + tagName: "a", + properties: { href: inner.properties?.href, class: ["internal"] }, + children: [{ type: "text", value: `Link to original` }], + }, + ] + } + } + } + }) + + const { + head: Head, + header, + beforeBody, + pageBody: Content, + left, + right, + footer: Footer, + } = components + const Header = HeaderConstructor() + const Body = BodyConstructor() + + const LeftComponent = ( + + ) + + const RightComponent = ( + + ) + + const doc = ( + + + +
    + + {LeftComponent} +
    + + +
    + {RightComponent} + +
    +
    + + {pageResources.js + .filter((resource) => resource.loadTime === "afterDOMReady") + .map((res) => JSResourceToScriptElement(res))} + + ) + + return "\n" + render(doc) +} diff --git a/quartz/components/scripts/callout.inline.ts b/quartz/components/scripts/callout.inline.ts new file mode 100644 index 0000000..d8cf518 --- /dev/null +++ b/quartz/components/scripts/callout.inline.ts @@ -0,0 +1,44 @@ +function toggleCallout(this: HTMLElement) { + const outerBlock = this.parentElement! + outerBlock.classList.toggle(`is-collapsed`) + const collapsed = outerBlock.classList.contains(`is-collapsed`) + const height = collapsed ? this.scrollHeight : outerBlock.scrollHeight + outerBlock.style.maxHeight = height + `px` + + // walk and adjust height of all parents + let current = outerBlock + let parent = outerBlock.parentElement + while (parent) { + if (!parent.classList.contains(`callout`)) { + return + } + + const collapsed = parent.classList.contains(`is-collapsed`) + const height = collapsed ? parent.scrollHeight : parent.scrollHeight + current.scrollHeight + parent.style.maxHeight = height + `px` + + current = parent + parent = parent.parentElement + } +} + +function setupCallout() { + const collapsible = document.getElementsByClassName( + `callout is-collapsible`, + ) as HTMLCollectionOf + for (const div of collapsible) { + const title = div.firstElementChild + + if (title) { + title.removeEventListener(`click`, toggleCallout) + title.addEventListener(`click`, toggleCallout) + + const collapsed = div.classList.contains(`is-collapsed`) + const height = collapsed ? title.scrollHeight : div.scrollHeight + div.style.maxHeight = height + `px` + } + } +} + +document.addEventListener(`nav`, setupCallout) +window.addEventListener(`resize`, setupCallout) diff --git a/quartz/components/scripts/clipboard.inline.ts b/quartz/components/scripts/clipboard.inline.ts new file mode 100644 index 0000000..c604c9b --- /dev/null +++ b/quartz/components/scripts/clipboard.inline.ts @@ -0,0 +1,33 @@ +const svgCopy = + '' +const svgCheck = + '' + +document.addEventListener("nav", () => { + const els = document.getElementsByTagName("pre") + for (let i = 0; i < els.length; i++) { + const codeBlock = els[i].getElementsByTagName("code")[0] + if (codeBlock) { + const source = codeBlock.innerText.replace(/\n\n/g, "\n") + const button = document.createElement("button") + button.className = "clipboard-button" + button.type = "button" + button.innerHTML = svgCopy + button.ariaLabel = "Copy source" + button.addEventListener("click", () => { + navigator.clipboard.writeText(source).then( + () => { + button.blur() + button.innerHTML = svgCheck + setTimeout(() => { + button.innerHTML = svgCopy + button.style.borderColor = "" + }, 2000) + }, + (error) => console.error(error), + ) + }) + els[i].prepend(button) + } + } +}) diff --git a/quartz/components/scripts/darkmode.inline.ts b/quartz/components/scripts/darkmode.inline.ts new file mode 100644 index 0000000..c42a367 --- /dev/null +++ b/quartz/components/scripts/darkmode.inline.ts @@ -0,0 +1,32 @@ +const userPref = window.matchMedia("(prefers-color-scheme: light)").matches ? "light" : "dark" +const currentTheme = localStorage.getItem("theme") ?? userPref +document.documentElement.setAttribute("saved-theme", currentTheme) + +document.addEventListener("nav", () => { + const switchTheme = (e: any) => { + if (e.target.checked) { + document.documentElement.setAttribute("saved-theme", "dark") + localStorage.setItem("theme", "dark") + } else { + document.documentElement.setAttribute("saved-theme", "light") + localStorage.setItem("theme", "light") + } + } + + // Darkmode toggle + const toggleSwitch = document.querySelector("#darkmode-toggle") as HTMLInputElement + toggleSwitch.removeEventListener("change", switchTheme) + toggleSwitch.addEventListener("change", switchTheme) + if (currentTheme === "dark") { + toggleSwitch.checked = true + } + + // Listen for changes in prefers-color-scheme + const colorSchemeMediaQuery = window.matchMedia("(prefers-color-scheme: dark)") + colorSchemeMediaQuery.addEventListener("change", (e) => { + const newTheme = e.matches ? "dark" : "light" + document.documentElement.setAttribute("saved-theme", newTheme) + localStorage.setItem("theme", newTheme) + toggleSwitch.checked = e.matches + }) +}) diff --git a/quartz/components/scripts/explorer.inline.ts b/quartz/components/scripts/explorer.inline.ts new file mode 100644 index 0000000..9fe1865 --- /dev/null +++ b/quartz/components/scripts/explorer.inline.ts @@ -0,0 +1,164 @@ +import { FolderState } from "../ExplorerNode" + +// Current state of folders +let explorerState: FolderState[] + +const observer = new IntersectionObserver((entries) => { + // If last element is observed, remove gradient of "overflow" class so element is visible + const explorer = document.getElementById("explorer-ul") + for (const entry of entries) { + if (entry.isIntersecting) { + explorer?.classList.add("no-background") + } else { + explorer?.classList.remove("no-background") + } + } +}) + +function toggleExplorer(this: HTMLElement) { + // Toggle collapsed state of entire explorer + this.classList.toggle("collapsed") + const content = this.nextElementSibling as HTMLElement + content.classList.toggle("collapsed") + content.style.maxHeight = content.style.maxHeight === "0px" ? content.scrollHeight + "px" : "0px" +} + +function toggleFolder(evt: MouseEvent) { + evt.stopPropagation() + + // Element that was clicked + const target = evt.target as HTMLElement + + // Check if target was svg icon or button + const isSvg = target.nodeName === "svg" + + // corresponding