Compare commits

..

1 Commits

Author SHA1 Message Date
linesofcodes 8540d72993 KaTeX & Syntax Highlighting support 2025-09-09 13:17:53 +07:00
10 changed files with 709 additions and 152 deletions
+4 -2
View File
@@ -1,9 +1,11 @@
{ {
"useTabs": true, "useTabs": true,
"singleQuote": true, "singleQuote": false,
"trailingComma": "none", "trailingComma": "none",
"printWidth": 100, "printWidth": 100,
"plugins": ["prettier-plugin-svelte"], "plugins": [
"prettier-plugin-svelte"
],
"overrides": [ "overrides": [
{ {
"files": "*.svelte", "files": "*.svelte",
+5
View File
@@ -50,8 +50,13 @@
"@codemirror/state": "^6.5.2", "@codemirror/state": "^6.5.2",
"@codemirror/view": "^6.38.1", "@codemirror/view": "^6.38.1",
"dompurify": "^3.2.6", "dompurify": "^3.2.6",
"highlight.js": "^11.11.1",
"isomorphic-dompurify": "^2.26.0",
"jszip": "^3.10.1", "jszip": "^3.10.1",
"katex": "^0.16.22",
"marked": "^16.2.1", "marked": "^16.2.1",
"marked-highlight": "^2.2.2",
"marked-katex-extension": "^5.1.5",
"svelte-codemirror-editor": "^1.4.1", "svelte-codemirror-editor": "^1.4.1",
"thememirror": "^2.0.1" "thememirror": "^2.0.1"
} }
+413 -25
View File
@@ -23,12 +23,27 @@ importers:
dompurify: dompurify:
specifier: ^3.2.6 specifier: ^3.2.6
version: 3.2.6 version: 3.2.6
highlight.js:
specifier: ^11.11.1
version: 11.11.1
isomorphic-dompurify:
specifier: ^2.26.0
version: 2.26.0
jszip: jszip:
specifier: ^3.10.1 specifier: ^3.10.1
version: 3.10.1 version: 3.10.1
katex:
specifier: ^0.16.22
version: 0.16.22
marked: marked:
specifier: ^16.2.1 specifier: ^16.2.1
version: 16.2.1 version: 16.2.1
marked-highlight:
specifier: ^2.2.2
version: 2.2.2(marked@16.2.1)
marked-katex-extension:
specifier: ^5.1.5
version: 5.1.5(katex@0.16.22)(marked@16.2.1)
svelte-codemirror-editor: svelte-codemirror-editor:
specifier: ^1.4.1 specifier: ^1.4.1
version: 1.4.1(codemirror@6.0.2)(svelte@5.38.3) version: 1.4.1(codemirror@6.0.2)(svelte@5.38.3)
@@ -44,19 +59,19 @@ importers:
version: 9.34.0 version: 9.34.0
'@sveltejs/adapter-auto': '@sveltejs/adapter-auto':
specifier: ^6.0.0 specifier: ^6.0.0
version: 6.1.0(@sveltejs/kit@2.36.2(@sveltejs/vite-plugin-svelte@6.1.3(svelte@5.38.3)(vite@7.1.3(jiti@2.5.1)(lightningcss@1.30.1)))(svelte@5.38.3)(vite@7.1.3(jiti@2.5.1)(lightningcss@1.30.1))) version: 6.1.0(@sveltejs/kit@2.36.2(@sveltejs/vite-plugin-svelte@6.1.3(svelte@5.38.3)(vite@7.1.3(@types/node@24.3.1)(jiti@2.5.1)(lightningcss@1.30.1)))(svelte@5.38.3)(vite@7.1.3(@types/node@24.3.1)(jiti@2.5.1)(lightningcss@1.30.1)))
'@sveltejs/kit': '@sveltejs/kit':
specifier: ^2.22.0 specifier: ^2.22.0
version: 2.36.2(@sveltejs/vite-plugin-svelte@6.1.3(svelte@5.38.3)(vite@7.1.3(jiti@2.5.1)(lightningcss@1.30.1)))(svelte@5.38.3)(vite@7.1.3(jiti@2.5.1)(lightningcss@1.30.1)) version: 2.36.2(@sveltejs/vite-plugin-svelte@6.1.3(svelte@5.38.3)(vite@7.1.3(@types/node@24.3.1)(jiti@2.5.1)(lightningcss@1.30.1)))(svelte@5.38.3)(vite@7.1.3(@types/node@24.3.1)(jiti@2.5.1)(lightningcss@1.30.1))
'@sveltejs/vite-plugin-svelte': '@sveltejs/vite-plugin-svelte':
specifier: ^6.0.0 specifier: ^6.0.0
version: 6.1.3(svelte@5.38.3)(vite@7.1.3(jiti@2.5.1)(lightningcss@1.30.1)) version: 6.1.3(svelte@5.38.3)(vite@7.1.3(@types/node@24.3.1)(jiti@2.5.1)(lightningcss@1.30.1))
'@tailwindcss/typography': '@tailwindcss/typography':
specifier: ^0.5.15 specifier: ^0.5.15
version: 0.5.16(tailwindcss@4.1.12) version: 0.5.16(tailwindcss@4.1.12)
'@tailwindcss/vite': '@tailwindcss/vite':
specifier: ^4.0.0 specifier: ^4.0.0
version: 4.1.12(vite@7.1.3(jiti@2.5.1)(lightningcss@1.30.1)) version: 4.1.12(vite@7.1.3(@types/node@24.3.1)(jiti@2.5.1)(lightningcss@1.30.1))
'@types/html2pdf.js': '@types/html2pdf.js':
specifier: ^0.10.0 specifier: ^0.10.0
version: 0.10.0 version: 0.10.0
@@ -95,13 +110,16 @@ importers:
version: 8.41.0(eslint@9.34.0(jiti@2.5.1))(typescript@5.9.2) version: 8.41.0(eslint@9.34.0(jiti@2.5.1))(typescript@5.9.2)
vite: vite:
specifier: ^7.0.4 specifier: ^7.0.4
version: 7.1.3(jiti@2.5.1)(lightningcss@1.30.1) version: 7.1.3(@types/node@24.3.1)(jiti@2.5.1)(lightningcss@1.30.1)
vite-plugin-devtools-json: vite-plugin-devtools-json:
specifier: ^1.0.0 specifier: ^1.0.0
version: 1.0.0(vite@7.1.3(jiti@2.5.1)(lightningcss@1.30.1)) version: 1.0.0(vite@7.1.3(@types/node@24.3.1)(jiti@2.5.1)(lightningcss@1.30.1))
packages: packages:
'@asamuzakjp/css-color@3.2.0':
resolution: {integrity: sha512-K1A6z8tS3XsmCMM86xoWdn7Fkdn9m6RSVtocUrJYIwZnFVkng/PvkEoWtOWmP+Scc6saYWHWZYbndEEXxl24jw==}
'@codemirror/autocomplete@6.18.6': '@codemirror/autocomplete@6.18.6':
resolution: {integrity: sha512-PHHBXFomUs5DF+9tCOM/UoW6XQ4R44lLNNhRaW9PKPTU0D7lIjRg3ElxaJnTwsl/oHiR93WSXDBrekhoUGCPtg==} resolution: {integrity: sha512-PHHBXFomUs5DF+9tCOM/UoW6XQ4R44lLNNhRaW9PKPTU0D7lIjRg3ElxaJnTwsl/oHiR93WSXDBrekhoUGCPtg==}
@@ -135,6 +153,34 @@ packages:
'@codemirror/view@6.38.1': '@codemirror/view@6.38.1':
resolution: {integrity: sha512-RmTOkE7hRU3OVREqFVITWHz6ocgBjv08GoePscAakgVQfciA3SGCEk7mb9IzwW61cKKmlTpHXG6DUE5Ubx+MGQ==} resolution: {integrity: sha512-RmTOkE7hRU3OVREqFVITWHz6ocgBjv08GoePscAakgVQfciA3SGCEk7mb9IzwW61cKKmlTpHXG6DUE5Ubx+MGQ==}
'@csstools/color-helpers@5.1.0':
resolution: {integrity: sha512-S11EXWJyy0Mz5SYvRmY8nJYTFFd1LCNV+7cXyAgQtOOuzb4EsgfqDufL+9esx72/eLhsRdGZwaldu/h+E4t4BA==}
engines: {node: '>=18'}
'@csstools/css-calc@2.1.4':
resolution: {integrity: sha512-3N8oaj+0juUw/1H3YwmDDJXCgTB1gKU6Hc/bB502u9zR0q2vd786XJH9QfrKIEgFlZmhZiq6epXl4rHqhzsIgQ==}
engines: {node: '>=18'}
peerDependencies:
'@csstools/css-parser-algorithms': ^3.0.5
'@csstools/css-tokenizer': ^3.0.4
'@csstools/css-color-parser@3.1.0':
resolution: {integrity: sha512-nbtKwh3a6xNVIp/VRuXV64yTKnb1IjTAEEh3irzS+HkKjAOYLTGNb9pmVNntZ8iVBHcWDA2Dof0QtPgFI1BaTA==}
engines: {node: '>=18'}
peerDependencies:
'@csstools/css-parser-algorithms': ^3.0.5
'@csstools/css-tokenizer': ^3.0.4
'@csstools/css-parser-algorithms@3.0.5':
resolution: {integrity: sha512-DaDeUkXZKjdGhgYaHNJTV9pV7Y9B3b644jCLs9Upc3VeNGg6LWARAT6O+Q+/COo+2gg/bM5rhpMAtf70WqfBdQ==}
engines: {node: '>=18'}
peerDependencies:
'@csstools/css-tokenizer': ^3.0.4
'@csstools/css-tokenizer@3.0.4':
resolution: {integrity: sha512-Vd/9EVDiu6PPJt9yAh6roZP6El1xHrdvIVGjyBsHR0RYwNHgL7FJPyIIW4fANJNG6FtyZfvlRPpFI4ZM/lubvw==}
engines: {node: '>=18'}
'@esbuild/aix-ppc64@0.25.9': '@esbuild/aix-ppc64@0.25.9':
resolution: {integrity: sha512-OaGtL73Jck6pBKjNIe24BnFE6agGl+6KxDtTfHhy1HmhthfKouEcOhqpSL64K4/0WCtbKFLOdzD/44cJ4k9opA==} resolution: {integrity: sha512-OaGtL73Jck6pBKjNIe24BnFE6agGl+6KxDtTfHhy1HmhthfKouEcOhqpSL64K4/0WCtbKFLOdzD/44cJ4k9opA==}
engines: {node: '>=18'} engines: {node: '>=18'}
@@ -665,6 +711,9 @@ packages:
'@types/json-schema@7.0.15': '@types/json-schema@7.0.15':
resolution: {integrity: sha512-5+fP8P8MFNC+AyZCDxrB2pkZFPGzqQWUzpSeuuVLvm8VMcorNYavBqoFcxK8bQz4Qsbn4oUEEem4wDLfcysGHA==} resolution: {integrity: sha512-5+fP8P8MFNC+AyZCDxrB2pkZFPGzqQWUzpSeuuVLvm8VMcorNYavBqoFcxK8bQz4Qsbn4oUEEem4wDLfcysGHA==}
'@types/node@24.3.1':
resolution: {integrity: sha512-3vXmQDXy+woz+gnrTvuvNrPzekOi+Ds0ReMxw0LzBiK3a+1k0kQn9f2NWk+lgD4rJehFUmYy2gMhJ2ZI+7YP9g==}
'@types/trusted-types@2.0.7': '@types/trusted-types@2.0.7':
resolution: {integrity: sha512-ScaPdn1dQczgbl0QFTeTOmVHFULt394XJgOQNoyVhZ6r2vLnMLJfBPd53SB52T/3G36VI1/g2MZaX0cwDuXsfw==} resolution: {integrity: sha512-ScaPdn1dQczgbl0QFTeTOmVHFULt394XJgOQNoyVhZ6r2vLnMLJfBPd53SB52T/3G36VI1/g2MZaX0cwDuXsfw==}
@@ -737,6 +786,10 @@ packages:
engines: {node: '>=0.4.0'} engines: {node: '>=0.4.0'}
hasBin: true hasBin: true
agent-base@7.1.4:
resolution: {integrity: sha512-MnA+YT8fwfJPgBx3m60MNqakm30XOkyIoH1y6huTQvC0PwZG7ki8NacLBcrPbNoo8vEZy7Jpuk7+jMO+CUovTQ==}
engines: {node: '>= 14'}
ajv@6.12.6: ajv@6.12.6:
resolution: {integrity: sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g==} resolution: {integrity: sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g==}
@@ -798,6 +851,10 @@ packages:
color-name@1.1.4: color-name@1.1.4:
resolution: {integrity: sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==} resolution: {integrity: sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==}
commander@8.3.0:
resolution: {integrity: sha512-OkTL9umf+He2DZkUq8f8J9of7yL6RJKI24dVITBmNfZBmri9zYZQrKkuXiKhyfPSu8tUhnVBB1iKXevvnlR4Ww==}
engines: {node: '>= 12'}
concat-map@0.0.1: concat-map@0.0.1:
resolution: {integrity: sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg==} resolution: {integrity: sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg==}
@@ -820,6 +877,14 @@ packages:
engines: {node: '>=4'} engines: {node: '>=4'}
hasBin: true hasBin: true
cssstyle@4.6.0:
resolution: {integrity: sha512-2z+rWdzbbSZv6/rhtvzvqeZQHrBaqgogqt85sqFNbabZOuFbCVFb8kPeEtZjiKkbrm395irpNKiYeFeLiQnFPg==}
engines: {node: '>=18'}
data-urls@5.0.0:
resolution: {integrity: sha512-ZYP5VBHshaDAiVZxjbRVcFJpc+4xGgT0bK3vzy1HLN8jTO975HEbuYzZJcHoQEY5K1a0z8YayJkyVETa08eNTg==}
engines: {node: '>=18'}
debug@4.4.1: debug@4.4.1:
resolution: {integrity: sha512-KcKCqiftBJcZr++7ykoDIEwSa3XWowTfNPo92BYxjXiyYEVrUQh2aLyhxBCwww+heortUFxEJYcRzosstTEBYQ==} resolution: {integrity: sha512-KcKCqiftBJcZr++7ykoDIEwSa3XWowTfNPo92BYxjXiyYEVrUQh2aLyhxBCwww+heortUFxEJYcRzosstTEBYQ==}
engines: {node: '>=6.0'} engines: {node: '>=6.0'}
@@ -829,6 +894,9 @@ packages:
supports-color: supports-color:
optional: true optional: true
decimal.js@10.6.0:
resolution: {integrity: sha512-YpgQiITW3JXGntzdUmyUR1V812Hn8T1YVXhCu+wO3OpS4eU9l4YdD3qjyiKdV6mvV29zapkMeD390UVEf2lkUg==}
deep-is@0.1.4: deep-is@0.1.4:
resolution: {integrity: sha512-oIPzksmTg4/MriiaYGO+okXDT7ztn/w3Eptv/+gSIdMdKsJo0u4CfYNFJPy+4SKMuCqGw2wxnA+URMg3t8a/bQ==} resolution: {integrity: sha512-oIPzksmTg4/MriiaYGO+okXDT7ztn/w3Eptv/+gSIdMdKsJo0u4CfYNFJPy+4SKMuCqGw2wxnA+URMg3t8a/bQ==}
@@ -850,6 +918,10 @@ packages:
resolution: {integrity: sha512-d4lC8xfavMeBjzGr2vECC3fsGXziXZQyJxD868h2M/mBI3PwAuODxAkLkq5HYuvrPYcUtiLzsTo8U3PgX3Ocww==} resolution: {integrity: sha512-d4lC8xfavMeBjzGr2vECC3fsGXziXZQyJxD868h2M/mBI3PwAuODxAkLkq5HYuvrPYcUtiLzsTo8U3PgX3Ocww==}
engines: {node: '>=10.13.0'} engines: {node: '>=10.13.0'}
entities@6.0.1:
resolution: {integrity: sha512-aN97NXWF6AWBTahfVOIrB/NShkzi5H7F9r1s9mD3cDj4Ko5f2qhhVoYMibXF7GlLveb/D2ioWay8lxI97Ven3g==}
engines: {node: '>=0.12'}
esbuild@0.25.9: esbuild@0.25.9:
resolution: {integrity: sha512-CRbODhYyQx3qp7ZEwzxOk4JBqmD/seJrzPa/cGjY1VtIn5E09Oi9/dB4JwctnfZ8Q8iT7rioVv5k/FNT/uf54g==} resolution: {integrity: sha512-CRbODhYyQx3qp7ZEwzxOk4JBqmD/seJrzPa/cGjY1VtIn5E09Oi9/dB4JwctnfZ8Q8iT7rioVv5k/FNT/uf54g==}
engines: {node: '>=18'} engines: {node: '>=18'}
@@ -998,6 +1070,26 @@ packages:
resolution: {integrity: sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==} resolution: {integrity: sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==}
engines: {node: '>=8'} engines: {node: '>=8'}
highlight.js@11.11.1:
resolution: {integrity: sha512-Xwwo44whKBVCYoliBQwaPvtd/2tYFkRQtXDWj1nackaV2JPXx3L0+Jvd8/qCJ2p+ML0/XVkJ2q+Mr+UVdpJK5w==}
engines: {node: '>=12.0.0'}
html-encoding-sniffer@4.0.0:
resolution: {integrity: sha512-Y22oTqIU4uuPgEemfz7NDJz6OeKf12Lsu+QC+s3BVpda64lTiMYCyGwg5ki4vFxkMwQdeZDl2adZoqUgdFuTgQ==}
engines: {node: '>=18'}
http-proxy-agent@7.0.2:
resolution: {integrity: sha512-T1gkAiYYDWYx3V5Bmyu7HcfcvL7mUrTWiM6yOfa3PIphViJ/gFPbvidQ+veqSOHci/PxBcDabeUNCzpOODJZig==}
engines: {node: '>= 14'}
https-proxy-agent@7.0.6:
resolution: {integrity: sha512-vK9P5/iUfdl95AI+JVyUuIcVtd4ofvtrOr3HNtM2yxC9bnMbEdp3x01OhQNnjb8IJYi38VlTE3mBXwcfvywuSw==}
engines: {node: '>= 14'}
iconv-lite@0.6.3:
resolution: {integrity: sha512-4fCk79wshMdzMp2rH06qWrJE4iolqLhCUH+OiuIgU++RB0+94NlDL81atO7GX55uUKueo0txHNtvEyI6D7WdMw==}
engines: {node: '>=0.10.0'}
ignore@5.3.2: ignore@5.3.2:
resolution: {integrity: sha512-hsBTNUqQTDwkWtcdYI2i06Y/nUBEsNEDJKjWdigLvegy8kDuJAS8uRlpkkcQpyEXL0Z/pjDy5HBmMjRCJ2gq+g==} resolution: {integrity: sha512-hsBTNUqQTDwkWtcdYI2i06Y/nUBEsNEDJKjWdigLvegy8kDuJAS8uRlpkkcQpyEXL0Z/pjDy5HBmMjRCJ2gq+g==}
engines: {node: '>= 4'} engines: {node: '>= 4'}
@@ -1032,6 +1124,9 @@ packages:
resolution: {integrity: sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==} resolution: {integrity: sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==}
engines: {node: '>=0.12.0'} engines: {node: '>=0.12.0'}
is-potential-custom-element-name@1.0.1:
resolution: {integrity: sha512-bCYeRA2rVibKZd+s2625gGnGF/t7DSqDs4dP7CrLA1m7jKWz6pps0LpYLJN8Q64HtmPKJ1hrN3nzPNKFEKOUiQ==}
is-reference@3.0.3: is-reference@3.0.3:
resolution: {integrity: sha512-ixkJoqQvAP88E6wLydLGGqCJsrFUnqoH6HnaczB8XmDH1oaWU+xxdptvikTgaEhtZ53Ky6YXiBuUI2WXLMCwjw==} resolution: {integrity: sha512-ixkJoqQvAP88E6wLydLGGqCJsrFUnqoH6HnaczB8XmDH1oaWU+xxdptvikTgaEhtZ53Ky6YXiBuUI2WXLMCwjw==}
@@ -1041,6 +1136,10 @@ packages:
isexe@2.0.0: isexe@2.0.0:
resolution: {integrity: sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw==} resolution: {integrity: sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw==}
isomorphic-dompurify@2.26.0:
resolution: {integrity: sha512-nZmoK4wKdzPs5USq4JHBiimjdKSVAOm2T1KyDoadtMPNXYHxiENd19ou4iU/V4juFM6LVgYQnpxCYmxqNP4Obw==}
engines: {node: '>=18'}
jiti@2.5.1: jiti@2.5.1:
resolution: {integrity: sha512-twQoecYPiVA5K/h6SxtORw/Bs3ar+mLUtoPSc7iMXzQzK8d7eJ/R09wmTwAjiamETn1cXYPGfNnu7DMoHgu12w==} resolution: {integrity: sha512-twQoecYPiVA5K/h6SxtORw/Bs3ar+mLUtoPSc7iMXzQzK8d7eJ/R09wmTwAjiamETn1cXYPGfNnu7DMoHgu12w==}
hasBin: true hasBin: true
@@ -1049,6 +1148,15 @@ packages:
resolution: {integrity: sha512-wpxZs9NoxZaJESJGIZTyDEaYpl0FKSA+FB9aJiyemKhMwkxQg63h4T1KJgUGHpTqPDNRcmmYLugrRjJlBtWvRA==} resolution: {integrity: sha512-wpxZs9NoxZaJESJGIZTyDEaYpl0FKSA+FB9aJiyemKhMwkxQg63h4T1KJgUGHpTqPDNRcmmYLugrRjJlBtWvRA==}
hasBin: true hasBin: true
jsdom@26.1.0:
resolution: {integrity: sha512-Cvc9WUhxSMEo4McES3P7oK3QaXldCfNWp7pl2NNeiIFlCoLr3kfq9kb1fxftiwk1FLV7CvpvDfonxtzUDeSOPg==}
engines: {node: '>=18'}
peerDependencies:
canvas: ^3.0.0
peerDependenciesMeta:
canvas:
optional: true
json-buffer@3.0.1: json-buffer@3.0.1:
resolution: {integrity: sha512-4bV5BfR2mqfQTJm+V5tPPdf+ZpuhiIvTuAB5g8kcrXOZpTT/QwwVRWBywX1ozr6lEuPdbHxwaJlm9G6mI2sfSQ==} resolution: {integrity: sha512-4bV5BfR2mqfQTJm+V5tPPdf+ZpuhiIvTuAB5g8kcrXOZpTT/QwwVRWBywX1ozr6lEuPdbHxwaJlm9G6mI2sfSQ==}
@@ -1061,6 +1169,10 @@ packages:
jszip@3.10.1: jszip@3.10.1:
resolution: {integrity: sha512-xXDvecyTpGLrqFrvkrUSoxxfJI5AH7U8zxxtVclpsUtMCq4JQ290LY8AW5c7Ggnr/Y/oK+bQMbqK2qmtk3pN4g==} resolution: {integrity: sha512-xXDvecyTpGLrqFrvkrUSoxxfJI5AH7U8zxxtVclpsUtMCq4JQ290LY8AW5c7Ggnr/Y/oK+bQMbqK2qmtk3pN4g==}
katex@0.16.22:
resolution: {integrity: sha512-XCHRdUw4lf3SKBaJe4EvgqIuWwkPSo9XoeO8GjQW94Bp7TWv9hNhzZjZ+OH9yf1UmLygb7DIT5GSFQiyt16zYg==}
hasBin: true
keyv@4.5.4: keyv@4.5.4:
resolution: {integrity: sha512-oxVHkHR/EJf2CNXnWxRLW6mg7JyCCUcG0DtEGmL2ctUo1PNTin1PUil+r/+4r5MpVgC/fn1kjsx7mjSujKqIpw==} resolution: {integrity: sha512-oxVHkHR/EJf2CNXnWxRLW6mg7JyCCUcG0DtEGmL2ctUo1PNTin1PUil+r/+4r5MpVgC/fn1kjsx7mjSujKqIpw==}
@@ -1162,9 +1274,23 @@ packages:
lodash.merge@4.6.2: lodash.merge@4.6.2:
resolution: {integrity: sha512-0KpjqXRVvrYyCsX1swR/XTK0va6VQkQM6MNo7PqW77ByjAhoARA8EfrP1N4+KlKj8YS0ZUCtRT/YUuhyYDujIQ==} resolution: {integrity: sha512-0KpjqXRVvrYyCsX1swR/XTK0va6VQkQM6MNo7PqW77ByjAhoARA8EfrP1N4+KlKj8YS0ZUCtRT/YUuhyYDujIQ==}
lru-cache@10.4.3:
resolution: {integrity: sha512-JNAzZcXrCt42VGLuYz0zfAzDfAvJWW6AfYlDBQyDV5DClI2m5sAmK+OIO7s59XfsRsWHp02jAJrRadPRGTt6SQ==}
magic-string@0.30.18: magic-string@0.30.18:
resolution: {integrity: sha512-yi8swmWbO17qHhwIBNeeZxTceJMeBvWJaId6dyvTSOwTipqeHhMhOrz6513r1sOKnpvQ7zkhlG8tPrpilwTxHQ==} resolution: {integrity: sha512-yi8swmWbO17qHhwIBNeeZxTceJMeBvWJaId6dyvTSOwTipqeHhMhOrz6513r1sOKnpvQ7zkhlG8tPrpilwTxHQ==}
marked-highlight@2.2.2:
resolution: {integrity: sha512-KlHOP31DatbtPPXPaI8nx1KTrG3EW0Z5zewCwpUj65swbtKOTStteK3sNAjBqV75Pgo3fNEVNHeptg18mDuWgw==}
peerDependencies:
marked: '>=4 <17'
marked-katex-extension@5.1.5:
resolution: {integrity: sha512-AAlfnpi3ivBWt4Paqb8YW2R7eq2ESJ+AtrgJU1jImUwnxHJAhPkZ5MsR3POnWV9HKwlBLZTVGC5I5+I2xWRiFg==}
peerDependencies:
katex: '>=0.16 <0.17'
marked: '>=4 <17'
marked@16.2.1: marked@16.2.1:
resolution: {integrity: sha512-r3UrXED9lMlHF97jJByry90cwrZBBvZmjG1L68oYfuPMW+uDTnuMbyJDymCWwbTE+f+3LhpNDKfpR3a3saFyjA==} resolution: {integrity: sha512-r3UrXED9lMlHF97jJByry90cwrZBBvZmjG1L68oYfuPMW+uDTnuMbyJDymCWwbTE+f+3LhpNDKfpR3a3saFyjA==}
engines: {node: '>= 20'} engines: {node: '>= 20'}
@@ -1217,6 +1343,9 @@ packages:
natural-compare@1.4.0: natural-compare@1.4.0:
resolution: {integrity: sha512-OWND8ei3VtNC9h7V60qff3SVobHr996CTwgxubgyQYEpg290h9J0buyECNNJexkFm5sOajh5G116RYA1c8ZMSw==} resolution: {integrity: sha512-OWND8ei3VtNC9h7V60qff3SVobHr996CTwgxubgyQYEpg290h9J0buyECNNJexkFm5sOajh5G116RYA1c8ZMSw==}
nwsapi@2.2.22:
resolution: {integrity: sha512-ujSMe1OWVn55euT1ihwCI1ZcAaAU3nxUiDwfDQldc51ZXaB9m2AyOn6/jh1BLe2t/G8xd6uKG1UBF2aZJeg2SQ==}
optionator@0.9.4: optionator@0.9.4:
resolution: {integrity: sha512-6IpQ7mKUxRcZNLIObR0hz7lxsapSSIYNZJwXPGeF0mTVqGKFIXj1DQcMoT22S3ROcLyY/rz0PWaWZ9ayWmad9g==} resolution: {integrity: sha512-6IpQ7mKUxRcZNLIObR0hz7lxsapSSIYNZJwXPGeF0mTVqGKFIXj1DQcMoT22S3ROcLyY/rz0PWaWZ9ayWmad9g==}
engines: {node: '>= 0.8.0'} engines: {node: '>= 0.8.0'}
@@ -1236,6 +1365,9 @@ packages:
resolution: {integrity: sha512-GQ2EWRpQV8/o+Aw8YqtfZZPfNRWZYkbidE9k5rpl/hC3vtHHBfGm2Ifi6qWV+coDGkrUKZAxE3Lot5kcsRlh+g==} resolution: {integrity: sha512-GQ2EWRpQV8/o+Aw8YqtfZZPfNRWZYkbidE9k5rpl/hC3vtHHBfGm2Ifi6qWV+coDGkrUKZAxE3Lot5kcsRlh+g==}
engines: {node: '>=6'} engines: {node: '>=6'}
parse5@7.3.0:
resolution: {integrity: sha512-IInvU7fabl34qmi9gY8XOVxhYyMyuH2xUNpb2q8/Y+7552KlejkRvqvD19nMoUW/uQGGbqNpA6Tufu5FL5BZgw==}
path-exists@4.0.0: path-exists@4.0.0:
resolution: {integrity: sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==} resolution: {integrity: sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==}
engines: {node: '>=8'} engines: {node: '>=8'}
@@ -1336,6 +1468,9 @@ packages:
engines: {node: '>=18.0.0', npm: '>=8.0.0'} engines: {node: '>=18.0.0', npm: '>=8.0.0'}
hasBin: true hasBin: true
rrweb-cssom@0.8.0:
resolution: {integrity: sha512-guoltQEx+9aMf2gDZ0s62EcV8lsXR+0w8915TC3ITdn2YueuNjdAYh/levpU9nFaoChh9RUS5ZdQMrKfVEN9tw==}
run-parallel@1.2.0: run-parallel@1.2.0:
resolution: {integrity: sha512-5l4VyZR86LZ/lDxZTR6jqL8AFE2S0IFLMP26AbjsLVADxHdhB/c0GUsH+y39UfCi3dzz8OlQuPmnaJOMoDHQBA==} resolution: {integrity: sha512-5l4VyZR86LZ/lDxZTR6jqL8AFE2S0IFLMP26AbjsLVADxHdhB/c0GUsH+y39UfCi3dzz8OlQuPmnaJOMoDHQBA==}
@@ -1346,6 +1481,13 @@ packages:
safe-buffer@5.1.2: safe-buffer@5.1.2:
resolution: {integrity: sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==} resolution: {integrity: sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==}
safer-buffer@2.1.2:
resolution: {integrity: sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==}
saxes@6.0.0:
resolution: {integrity: sha512-xAg7SOnEhrm5zI3puOOKyy1OMcMlIJZYNJY7xLBwSze0UjhPLnWfj2GF2EpT0jmzaJKIWKHLsaSSajf35bcYnA==}
engines: {node: '>=v12.22.7'}
semver@7.7.2: semver@7.7.2:
resolution: {integrity: sha512-RF0Fw+rO5AMf9MAyaRXI4AV0Ulj5lMHqVxxdSgiVbixSCXoEmmX/jk0CuJw4+3SqroYO9VoUh+HcuJivvtJemA==} resolution: {integrity: sha512-RF0Fw+rO5AMf9MAyaRXI4AV0Ulj5lMHqVxxdSgiVbixSCXoEmmX/jk0CuJw4+3SqroYO9VoUh+HcuJivvtJemA==}
engines: {node: '>=10'} engines: {node: '>=10'}
@@ -1414,6 +1556,9 @@ packages:
resolution: {integrity: sha512-ldbPzKdjUy7IALMBn15jzBM/TNxdXMxKeQZ538zzdABUjLg7e7/OIwnlaMQ+OR6s91W7DbDmJYjxHThHH7r9xA==} resolution: {integrity: sha512-ldbPzKdjUy7IALMBn15jzBM/TNxdXMxKeQZ538zzdABUjLg7e7/OIwnlaMQ+OR6s91W7DbDmJYjxHThHH7r9xA==}
engines: {node: '>=18'} engines: {node: '>=18'}
symbol-tree@3.2.4:
resolution: {integrity: sha512-9QNk5KwDF+Bvz+PyObkmSYjI5ksVUYtjW7AU22r2NKcfLJcXp96hkDWU3+XndOsUb+AQ9QhfzfCT2O+CNWT5Tw==}
tailwindcss@4.1.12: tailwindcss@4.1.12:
resolution: {integrity: sha512-DzFtxOi+7NsFf7DBtI3BJsynR+0Yp6etH+nRPTbpWnS2pZBaSksv/JGctNwSWzbFjp0vxSqknaUylseZqMDGrA==} resolution: {integrity: sha512-DzFtxOi+7NsFf7DBtI3BJsynR+0Yp6etH+nRPTbpWnS2pZBaSksv/JGctNwSWzbFjp0vxSqknaUylseZqMDGrA==}
@@ -1436,6 +1581,13 @@ packages:
resolution: {integrity: sha512-tX5e7OM1HnYr2+a2C/4V0htOcSQcoSTH9KgJnVvNm5zm/cyEWKJ7j7YutsH9CxMdtOkkLFy2AHrMci9IM8IPZQ==} resolution: {integrity: sha512-tX5e7OM1HnYr2+a2C/4V0htOcSQcoSTH9KgJnVvNm5zm/cyEWKJ7j7YutsH9CxMdtOkkLFy2AHrMci9IM8IPZQ==}
engines: {node: '>=12.0.0'} engines: {node: '>=12.0.0'}
tldts-core@6.1.86:
resolution: {integrity: sha512-Je6p7pkk+KMzMv2XXKmAE3McmolOQFdxkKw0R8EYNr7sELW46JqnNeTX8ybPiQgvg1ymCoF8LXs5fzFaZvJPTA==}
tldts@6.1.86:
resolution: {integrity: sha512-WMi/OQ2axVTf/ykqCQgXiIct+mSQDFdH2fkwhPwgEwvJ1kSzZRiinb0zF2Xb8u4+OqPChmyI6MEu4EezNJz+FQ==}
hasBin: true
to-regex-range@5.0.1: to-regex-range@5.0.1:
resolution: {integrity: sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==} resolution: {integrity: sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==}
engines: {node: '>=8.0'} engines: {node: '>=8.0'}
@@ -1444,6 +1596,14 @@ packages:
resolution: {integrity: sha512-sf4i37nQ2LBx4m3wB74y+ubopq6W/dIzXg0FDGjsYnZHVa1Da8FH853wlL2gtUhg+xJXjfk3kUZS3BRoQeoQBQ==} resolution: {integrity: sha512-sf4i37nQ2LBx4m3wB74y+ubopq6W/dIzXg0FDGjsYnZHVa1Da8FH853wlL2gtUhg+xJXjfk3kUZS3BRoQeoQBQ==}
engines: {node: '>=6'} engines: {node: '>=6'}
tough-cookie@5.1.2:
resolution: {integrity: sha512-FVDYdxtnj0G6Qm/DhNPSb8Ju59ULcup3tuJxkFb5K8Bv2pUXILbf0xZWU8PX8Ov19OXljbUyveOFwRMwkXzO+A==}
engines: {node: '>=16'}
tr46@5.1.1:
resolution: {integrity: sha512-hdF5ZgjTqgAntKkklYw0R03MG2x/bSzTtkxmIRw/sTNV8YXsCJ1tfLAX23lhxhHJlEf3CRCOCGGWw3vI3GaSPw==}
engines: {node: '>=18'}
ts-api-utils@2.1.0: ts-api-utils@2.1.0:
resolution: {integrity: sha512-CUgTZL1irw8u29bzrOD/nH85jqyc74D6SshFgujOIA7osm2Rz7dYH77agkx7H4FBNxDq7Cjf+IjaX/8zwFW+ZQ==} resolution: {integrity: sha512-CUgTZL1irw8u29bzrOD/nH85jqyc74D6SshFgujOIA7osm2Rz7dYH77agkx7H4FBNxDq7Cjf+IjaX/8zwFW+ZQ==}
engines: {node: '>=18.12'} engines: {node: '>=18.12'}
@@ -1466,6 +1626,9 @@ packages:
engines: {node: '>=14.17'} engines: {node: '>=14.17'}
hasBin: true hasBin: true
undici-types@7.10.0:
resolution: {integrity: sha512-t5Fy/nfn+14LuOc2KNYg75vZqClpAiqscVvMygNnlsHBFpSXdJaYtXMcdNLpl/Qvc3P2cB3s6lOV51nqsFq4ag==}
uri-js@4.4.1: uri-js@4.4.1:
resolution: {integrity: sha512-7rKUyy33Q1yc98pQ1DAmLtwX109F7TIfWlW1Ydo8Wl1ii1SeHieeh0HHfPeL2fMXK6z0s8ecKs9frCuLJvndBg==} resolution: {integrity: sha512-7rKUyy33Q1yc98pQ1DAmLtwX109F7TIfWlW1Ydo8Wl1ii1SeHieeh0HHfPeL2fMXK6z0s8ecKs9frCuLJvndBg==}
@@ -1532,6 +1695,26 @@ packages:
w3c-keyname@2.2.8: w3c-keyname@2.2.8:
resolution: {integrity: sha512-dpojBhNsCNN7T82Tm7k26A6G9ML3NkhDsnw9n/eoxSRlVBB4CEtIQ/KTCLI2Fwf3ataSXRhYFkQi3SlnFwPvPQ==} resolution: {integrity: sha512-dpojBhNsCNN7T82Tm7k26A6G9ML3NkhDsnw9n/eoxSRlVBB4CEtIQ/KTCLI2Fwf3ataSXRhYFkQi3SlnFwPvPQ==}
w3c-xmlserializer@5.0.0:
resolution: {integrity: sha512-o8qghlI8NZHU1lLPrpi2+Uq7abh4GGPpYANlalzWxyWteJOCsr/P+oPBA49TOLu5FTZO4d3F9MnWJfiMo4BkmA==}
engines: {node: '>=18'}
webidl-conversions@7.0.0:
resolution: {integrity: sha512-VwddBukDzu71offAQR975unBIGqfKZpM+8ZX6ySk8nYhVoo5CYaZyzt3YBvYtRtO+aoGlqxPg/B87NGVZ/fu6g==}
engines: {node: '>=12'}
whatwg-encoding@3.1.1:
resolution: {integrity: sha512-6qN4hJdMwfYBtE3YBTTHhoeuUrDBPZmbQaxWAqSALV/MeEnR5z1xd8UKud2RAkFoPkmB+hli1TZSnyi84xz1vQ==}
engines: {node: '>=18'}
whatwg-mimetype@4.0.0:
resolution: {integrity: sha512-QaKxh0eNIi2mE9p2vEdzfagOKHCcj1pJ56EEHGQOVxp8r9/iszLUUV7v89x9O1p/T+NlTM5W7jW6+cz4Fq1YVg==}
engines: {node: '>=18'}
whatwg-url@14.2.0:
resolution: {integrity: sha512-De72GdQZzNTUBBChsXueQUnPKDkg/5A5zp7pFDuQAj5UFoENpiACU0wlCvzpAGnTkj++ihpKwKyYewn/XNUbKw==}
engines: {node: '>=18'}
which@2.0.2: which@2.0.2:
resolution: {integrity: sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==} resolution: {integrity: sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==}
engines: {node: '>= 8'} engines: {node: '>= 8'}
@@ -1541,6 +1724,25 @@ packages:
resolution: {integrity: sha512-BN22B5eaMMI9UMtjrGd5g5eCYPpCPDUy0FJXbYsaT5zYxjFOckS53SQDE3pWkVoWpHXVb3BrYcEN4Twa55B5cA==} resolution: {integrity: sha512-BN22B5eaMMI9UMtjrGd5g5eCYPpCPDUy0FJXbYsaT5zYxjFOckS53SQDE3pWkVoWpHXVb3BrYcEN4Twa55B5cA==}
engines: {node: '>=0.10.0'} engines: {node: '>=0.10.0'}
ws@8.18.3:
resolution: {integrity: sha512-PEIGCY5tSlUt50cqyMXfCzX+oOPqN0vuGqWzbcJ2xvnkzkq46oOpz7dQaTDBdfICb4N14+GARUDw2XV2N4tvzg==}
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
xml-name-validator@5.0.0:
resolution: {integrity: sha512-EvGK8EJ3DhaHfbRlETOWAS5pO9MZITeauHKJyb8wyajUfQUenkIg2MvLDTZ4T/TgIcm3HU0TFBgWWboAZ30UHg==}
engines: {node: '>=18'}
xmlchars@2.2.0:
resolution: {integrity: sha512-JZnDKK8B0RCDw84FNdDAIpZK+JuJw+s7Lz8nksI7SIuU3UXJJslUthsi+uWBUYOwPFwW7W7PRLRfUKpxjtjFCw==}
yallist@5.0.0: yallist@5.0.0:
resolution: {integrity: sha512-YgvUTfwqyc7UXVMrB+SImsVYSmTS8X/tSrtdNZMImM+n7+QTriRXyXim0mBrTXNeqzVF0KWGgHPeiyViFFrNDw==} resolution: {integrity: sha512-YgvUTfwqyc7UXVMrB+SImsVYSmTS8X/tSrtdNZMImM+n7+QTriRXyXim0mBrTXNeqzVF0KWGgHPeiyViFFrNDw==}
engines: {node: '>=18'} engines: {node: '>=18'}
@@ -1558,6 +1760,14 @@ packages:
snapshots: snapshots:
'@asamuzakjp/css-color@3.2.0':
dependencies:
'@csstools/css-calc': 2.1.4(@csstools/css-parser-algorithms@3.0.5(@csstools/css-tokenizer@3.0.4))(@csstools/css-tokenizer@3.0.4)
'@csstools/css-color-parser': 3.1.0(@csstools/css-parser-algorithms@3.0.5(@csstools/css-tokenizer@3.0.4))(@csstools/css-tokenizer@3.0.4)
'@csstools/css-parser-algorithms': 3.0.5(@csstools/css-tokenizer@3.0.4)
'@csstools/css-tokenizer': 3.0.4
lru-cache: 10.4.3
'@codemirror/autocomplete@6.18.6': '@codemirror/autocomplete@6.18.6':
dependencies: dependencies:
'@codemirror/language': 6.11.3 '@codemirror/language': 6.11.3
@@ -1644,6 +1854,26 @@ snapshots:
style-mod: 4.1.2 style-mod: 4.1.2
w3c-keyname: 2.2.8 w3c-keyname: 2.2.8
'@csstools/color-helpers@5.1.0': {}
'@csstools/css-calc@2.1.4(@csstools/css-parser-algorithms@3.0.5(@csstools/css-tokenizer@3.0.4))(@csstools/css-tokenizer@3.0.4)':
dependencies:
'@csstools/css-parser-algorithms': 3.0.5(@csstools/css-tokenizer@3.0.4)
'@csstools/css-tokenizer': 3.0.4
'@csstools/css-color-parser@3.1.0(@csstools/css-parser-algorithms@3.0.5(@csstools/css-tokenizer@3.0.4))(@csstools/css-tokenizer@3.0.4)':
dependencies:
'@csstools/color-helpers': 5.1.0
'@csstools/css-calc': 2.1.4(@csstools/css-parser-algorithms@3.0.5(@csstools/css-tokenizer@3.0.4))(@csstools/css-tokenizer@3.0.4)
'@csstools/css-parser-algorithms': 3.0.5(@csstools/css-tokenizer@3.0.4)
'@csstools/css-tokenizer': 3.0.4
'@csstools/css-parser-algorithms@3.0.5(@csstools/css-tokenizer@3.0.4)':
dependencies:
'@csstools/css-tokenizer': 3.0.4
'@csstools/css-tokenizer@3.0.4': {}
'@esbuild/aix-ppc64@0.25.9': '@esbuild/aix-ppc64@0.25.9':
optional: true optional: true
@@ -1921,15 +2151,15 @@ snapshots:
dependencies: dependencies:
acorn: 8.15.0 acorn: 8.15.0
'@sveltejs/adapter-auto@6.1.0(@sveltejs/kit@2.36.2(@sveltejs/vite-plugin-svelte@6.1.3(svelte@5.38.3)(vite@7.1.3(jiti@2.5.1)(lightningcss@1.30.1)))(svelte@5.38.3)(vite@7.1.3(jiti@2.5.1)(lightningcss@1.30.1)))': '@sveltejs/adapter-auto@6.1.0(@sveltejs/kit@2.36.2(@sveltejs/vite-plugin-svelte@6.1.3(svelte@5.38.3)(vite@7.1.3(@types/node@24.3.1)(jiti@2.5.1)(lightningcss@1.30.1)))(svelte@5.38.3)(vite@7.1.3(@types/node@24.3.1)(jiti@2.5.1)(lightningcss@1.30.1)))':
dependencies: dependencies:
'@sveltejs/kit': 2.36.2(@sveltejs/vite-plugin-svelte@6.1.3(svelte@5.38.3)(vite@7.1.3(jiti@2.5.1)(lightningcss@1.30.1)))(svelte@5.38.3)(vite@7.1.3(jiti@2.5.1)(lightningcss@1.30.1)) '@sveltejs/kit': 2.36.2(@sveltejs/vite-plugin-svelte@6.1.3(svelte@5.38.3)(vite@7.1.3(@types/node@24.3.1)(jiti@2.5.1)(lightningcss@1.30.1)))(svelte@5.38.3)(vite@7.1.3(@types/node@24.3.1)(jiti@2.5.1)(lightningcss@1.30.1))
'@sveltejs/kit@2.36.2(@sveltejs/vite-plugin-svelte@6.1.3(svelte@5.38.3)(vite@7.1.3(jiti@2.5.1)(lightningcss@1.30.1)))(svelte@5.38.3)(vite@7.1.3(jiti@2.5.1)(lightningcss@1.30.1))': '@sveltejs/kit@2.36.2(@sveltejs/vite-plugin-svelte@6.1.3(svelte@5.38.3)(vite@7.1.3(@types/node@24.3.1)(jiti@2.5.1)(lightningcss@1.30.1)))(svelte@5.38.3)(vite@7.1.3(@types/node@24.3.1)(jiti@2.5.1)(lightningcss@1.30.1))':
dependencies: dependencies:
'@standard-schema/spec': 1.0.0 '@standard-schema/spec': 1.0.0
'@sveltejs/acorn-typescript': 1.0.5(acorn@8.15.0) '@sveltejs/acorn-typescript': 1.0.5(acorn@8.15.0)
'@sveltejs/vite-plugin-svelte': 6.1.3(svelte@5.38.3)(vite@7.1.3(jiti@2.5.1)(lightningcss@1.30.1)) '@sveltejs/vite-plugin-svelte': 6.1.3(svelte@5.38.3)(vite@7.1.3(@types/node@24.3.1)(jiti@2.5.1)(lightningcss@1.30.1))
'@types/cookie': 0.6.0 '@types/cookie': 0.6.0
acorn: 8.15.0 acorn: 8.15.0
cookie: 0.6.0 cookie: 0.6.0
@@ -1942,27 +2172,27 @@ snapshots:
set-cookie-parser: 2.7.1 set-cookie-parser: 2.7.1
sirv: 3.0.1 sirv: 3.0.1
svelte: 5.38.3 svelte: 5.38.3
vite: 7.1.3(jiti@2.5.1)(lightningcss@1.30.1) vite: 7.1.3(@types/node@24.3.1)(jiti@2.5.1)(lightningcss@1.30.1)
'@sveltejs/vite-plugin-svelte-inspector@5.0.1(@sveltejs/vite-plugin-svelte@6.1.3(svelte@5.38.3)(vite@7.1.3(jiti@2.5.1)(lightningcss@1.30.1)))(svelte@5.38.3)(vite@7.1.3(jiti@2.5.1)(lightningcss@1.30.1))': '@sveltejs/vite-plugin-svelte-inspector@5.0.1(@sveltejs/vite-plugin-svelte@6.1.3(svelte@5.38.3)(vite@7.1.3(@types/node@24.3.1)(jiti@2.5.1)(lightningcss@1.30.1)))(svelte@5.38.3)(vite@7.1.3(@types/node@24.3.1)(jiti@2.5.1)(lightningcss@1.30.1))':
dependencies: dependencies:
'@sveltejs/vite-plugin-svelte': 6.1.3(svelte@5.38.3)(vite@7.1.3(jiti@2.5.1)(lightningcss@1.30.1)) '@sveltejs/vite-plugin-svelte': 6.1.3(svelte@5.38.3)(vite@7.1.3(@types/node@24.3.1)(jiti@2.5.1)(lightningcss@1.30.1))
debug: 4.4.1 debug: 4.4.1
svelte: 5.38.3 svelte: 5.38.3
vite: 7.1.3(jiti@2.5.1)(lightningcss@1.30.1) vite: 7.1.3(@types/node@24.3.1)(jiti@2.5.1)(lightningcss@1.30.1)
transitivePeerDependencies: transitivePeerDependencies:
- supports-color - supports-color
'@sveltejs/vite-plugin-svelte@6.1.3(svelte@5.38.3)(vite@7.1.3(jiti@2.5.1)(lightningcss@1.30.1))': '@sveltejs/vite-plugin-svelte@6.1.3(svelte@5.38.3)(vite@7.1.3(@types/node@24.3.1)(jiti@2.5.1)(lightningcss@1.30.1))':
dependencies: dependencies:
'@sveltejs/vite-plugin-svelte-inspector': 5.0.1(@sveltejs/vite-plugin-svelte@6.1.3(svelte@5.38.3)(vite@7.1.3(jiti@2.5.1)(lightningcss@1.30.1)))(svelte@5.38.3)(vite@7.1.3(jiti@2.5.1)(lightningcss@1.30.1)) '@sveltejs/vite-plugin-svelte-inspector': 5.0.1(@sveltejs/vite-plugin-svelte@6.1.3(svelte@5.38.3)(vite@7.1.3(@types/node@24.3.1)(jiti@2.5.1)(lightningcss@1.30.1)))(svelte@5.38.3)(vite@7.1.3(@types/node@24.3.1)(jiti@2.5.1)(lightningcss@1.30.1))
debug: 4.4.1 debug: 4.4.1
deepmerge: 4.3.1 deepmerge: 4.3.1
kleur: 4.1.5 kleur: 4.1.5
magic-string: 0.30.18 magic-string: 0.30.18
svelte: 5.38.3 svelte: 5.38.3
vite: 7.1.3(jiti@2.5.1)(lightningcss@1.30.1) vite: 7.1.3(@types/node@24.3.1)(jiti@2.5.1)(lightningcss@1.30.1)
vitefu: 1.1.1(vite@7.1.3(jiti@2.5.1)(lightningcss@1.30.1)) vitefu: 1.1.1(vite@7.1.3(@types/node@24.3.1)(jiti@2.5.1)(lightningcss@1.30.1))
transitivePeerDependencies: transitivePeerDependencies:
- supports-color - supports-color
@@ -2038,12 +2268,12 @@ snapshots:
postcss-selector-parser: 6.0.10 postcss-selector-parser: 6.0.10
tailwindcss: 4.1.12 tailwindcss: 4.1.12
'@tailwindcss/vite@4.1.12(vite@7.1.3(jiti@2.5.1)(lightningcss@1.30.1))': '@tailwindcss/vite@4.1.12(vite@7.1.3(@types/node@24.3.1)(jiti@2.5.1)(lightningcss@1.30.1))':
dependencies: dependencies:
'@tailwindcss/node': 4.1.12 '@tailwindcss/node': 4.1.12
'@tailwindcss/oxide': 4.1.12 '@tailwindcss/oxide': 4.1.12
tailwindcss: 4.1.12 tailwindcss: 4.1.12
vite: 7.1.3(jiti@2.5.1)(lightningcss@1.30.1) vite: 7.1.3(@types/node@24.3.1)(jiti@2.5.1)(lightningcss@1.30.1)
'@types/cookie@0.6.0': {} '@types/cookie@0.6.0': {}
@@ -2053,6 +2283,11 @@ snapshots:
'@types/json-schema@7.0.15': {} '@types/json-schema@7.0.15': {}
'@types/node@24.3.1':
dependencies:
undici-types: 7.10.0
optional: true
'@types/trusted-types@2.0.7': '@types/trusted-types@2.0.7':
optional: true optional: true
@@ -2155,6 +2390,8 @@ snapshots:
acorn@8.15.0: {} acorn@8.15.0: {}
agent-base@7.1.4: {}
ajv@6.12.6: ajv@6.12.6:
dependencies: dependencies:
fast-deep-equal: 3.1.3 fast-deep-equal: 3.1.3
@@ -2218,6 +2455,8 @@ snapshots:
color-name@1.1.4: {} color-name@1.1.4: {}
commander@8.3.0: {}
concat-map@0.0.1: {} concat-map@0.0.1: {}
cookie@0.6.0: {} cookie@0.6.0: {}
@@ -2234,10 +2473,22 @@ snapshots:
cssesc@3.0.0: {} cssesc@3.0.0: {}
cssstyle@4.6.0:
dependencies:
'@asamuzakjp/css-color': 3.2.0
rrweb-cssom: 0.8.0
data-urls@5.0.0:
dependencies:
whatwg-mimetype: 4.0.0
whatwg-url: 14.2.0
debug@4.4.1: debug@4.4.1:
dependencies: dependencies:
ms: 2.1.3 ms: 2.1.3
decimal.js@10.6.0: {}
deep-is@0.1.4: {} deep-is@0.1.4: {}
deepmerge@4.3.1: {} deepmerge@4.3.1: {}
@@ -2255,6 +2506,8 @@ snapshots:
graceful-fs: 4.2.11 graceful-fs: 4.2.11
tapable: 2.2.3 tapable: 2.2.3
entities@6.0.1: {}
esbuild@0.25.9: esbuild@0.25.9:
optionalDependencies: optionalDependencies:
'@esbuild/aix-ppc64': 0.25.9 '@esbuild/aix-ppc64': 0.25.9
@@ -2446,6 +2699,30 @@ snapshots:
has-flag@4.0.0: {} has-flag@4.0.0: {}
highlight.js@11.11.1: {}
html-encoding-sniffer@4.0.0:
dependencies:
whatwg-encoding: 3.1.1
http-proxy-agent@7.0.2:
dependencies:
agent-base: 7.1.4
debug: 4.4.1
transitivePeerDependencies:
- supports-color
https-proxy-agent@7.0.6:
dependencies:
agent-base: 7.1.4
debug: 4.4.1
transitivePeerDependencies:
- supports-color
iconv-lite@0.6.3:
dependencies:
safer-buffer: 2.1.2
ignore@5.3.2: {} ignore@5.3.2: {}
ignore@7.0.5: {} ignore@7.0.5: {}
@@ -2469,6 +2746,8 @@ snapshots:
is-number@7.0.0: {} is-number@7.0.0: {}
is-potential-custom-element-name@1.0.1: {}
is-reference@3.0.3: is-reference@3.0.3:
dependencies: dependencies:
'@types/estree': 1.0.8 '@types/estree': 1.0.8
@@ -2477,12 +2756,49 @@ snapshots:
isexe@2.0.0: {} isexe@2.0.0: {}
isomorphic-dompurify@2.26.0:
dependencies:
dompurify: 3.2.6
jsdom: 26.1.0
transitivePeerDependencies:
- bufferutil
- canvas
- supports-color
- utf-8-validate
jiti@2.5.1: {} jiti@2.5.1: {}
js-yaml@4.1.0: js-yaml@4.1.0:
dependencies: dependencies:
argparse: 2.0.1 argparse: 2.0.1
jsdom@26.1.0:
dependencies:
cssstyle: 4.6.0
data-urls: 5.0.0
decimal.js: 10.6.0
html-encoding-sniffer: 4.0.0
http-proxy-agent: 7.0.2
https-proxy-agent: 7.0.6
is-potential-custom-element-name: 1.0.1
nwsapi: 2.2.22
parse5: 7.3.0
rrweb-cssom: 0.8.0
saxes: 6.0.0
symbol-tree: 3.2.4
tough-cookie: 5.1.2
w3c-xmlserializer: 5.0.0
webidl-conversions: 7.0.0
whatwg-encoding: 3.1.1
whatwg-mimetype: 4.0.0
whatwg-url: 14.2.0
ws: 8.18.3
xml-name-validator: 5.0.0
transitivePeerDependencies:
- bufferutil
- supports-color
- utf-8-validate
json-buffer@3.0.1: {} json-buffer@3.0.1: {}
json-schema-traverse@0.4.1: {} json-schema-traverse@0.4.1: {}
@@ -2496,6 +2812,10 @@ snapshots:
readable-stream: 2.3.8 readable-stream: 2.3.8
setimmediate: 1.0.5 setimmediate: 1.0.5
katex@0.16.22:
dependencies:
commander: 8.3.0
keyv@4.5.4: keyv@4.5.4:
dependencies: dependencies:
json-buffer: 3.0.1 json-buffer: 3.0.1
@@ -2572,10 +2892,21 @@ snapshots:
lodash.merge@4.6.2: {} lodash.merge@4.6.2: {}
lru-cache@10.4.3: {}
magic-string@0.30.18: magic-string@0.30.18:
dependencies: dependencies:
'@jridgewell/sourcemap-codec': 1.5.5 '@jridgewell/sourcemap-codec': 1.5.5
marked-highlight@2.2.2(marked@16.2.1):
dependencies:
marked: 16.2.1
marked-katex-extension@5.1.5(katex@0.16.22)(marked@16.2.1):
dependencies:
katex: 0.16.22
marked: 16.2.1
marked@16.2.1: {} marked@16.2.1: {}
merge2@1.4.1: {} merge2@1.4.1: {}
@@ -2611,6 +2942,8 @@ snapshots:
natural-compare@1.4.0: {} natural-compare@1.4.0: {}
nwsapi@2.2.22: {}
optionator@0.9.4: optionator@0.9.4:
dependencies: dependencies:
deep-is: 0.1.4 deep-is: 0.1.4
@@ -2634,6 +2967,10 @@ snapshots:
dependencies: dependencies:
callsites: 3.1.0 callsites: 3.1.0
parse5@7.3.0:
dependencies:
entities: 6.0.1
path-exists@4.0.0: {} path-exists@4.0.0: {}
path-key@3.1.1: {} path-key@3.1.1: {}
@@ -2732,6 +3069,8 @@ snapshots:
'@rollup/rollup-win32-x64-msvc': 4.48.1 '@rollup/rollup-win32-x64-msvc': 4.48.1
fsevents: 2.3.3 fsevents: 2.3.3
rrweb-cssom@0.8.0: {}
run-parallel@1.2.0: run-parallel@1.2.0:
dependencies: dependencies:
queue-microtask: 1.2.3 queue-microtask: 1.2.3
@@ -2742,6 +3081,12 @@ snapshots:
safe-buffer@5.1.2: {} safe-buffer@5.1.2: {}
safer-buffer@2.1.2: {}
saxes@6.0.0:
dependencies:
xmlchars: 2.2.0
semver@7.7.2: {} semver@7.7.2: {}
set-cookie-parser@2.7.1: {} set-cookie-parser@2.7.1: {}
@@ -2819,6 +3164,8 @@ snapshots:
magic-string: 0.30.18 magic-string: 0.30.18
zimmerframe: 1.1.2 zimmerframe: 1.1.2
symbol-tree@3.2.4: {}
tailwindcss@4.1.12: {} tailwindcss@4.1.12: {}
tapable@2.2.3: {} tapable@2.2.3: {}
@@ -2843,12 +3190,26 @@ snapshots:
fdir: 6.5.0(picomatch@4.0.3) fdir: 6.5.0(picomatch@4.0.3)
picomatch: 4.0.3 picomatch: 4.0.3
tldts-core@6.1.86: {}
tldts@6.1.86:
dependencies:
tldts-core: 6.1.86
to-regex-range@5.0.1: to-regex-range@5.0.1:
dependencies: dependencies:
is-number: 7.0.0 is-number: 7.0.0
totalist@3.0.1: {} totalist@3.0.1: {}
tough-cookie@5.1.2:
dependencies:
tldts: 6.1.86
tr46@5.1.1:
dependencies:
punycode: 2.3.1
ts-api-utils@2.1.0(typescript@5.9.2): ts-api-utils@2.1.0(typescript@5.9.2):
dependencies: dependencies:
typescript: 5.9.2 typescript: 5.9.2
@@ -2870,6 +3231,9 @@ snapshots:
typescript@5.9.2: {} typescript@5.9.2: {}
undici-types@7.10.0:
optional: true
uri-js@4.4.1: uri-js@4.4.1:
dependencies: dependencies:
punycode: 2.3.1 punycode: 2.3.1
@@ -2878,12 +3242,12 @@ snapshots:
uuid@11.1.0: {} uuid@11.1.0: {}
vite-plugin-devtools-json@1.0.0(vite@7.1.3(jiti@2.5.1)(lightningcss@1.30.1)): vite-plugin-devtools-json@1.0.0(vite@7.1.3(@types/node@24.3.1)(jiti@2.5.1)(lightningcss@1.30.1)):
dependencies: dependencies:
uuid: 11.1.0 uuid: 11.1.0
vite: 7.1.3(jiti@2.5.1)(lightningcss@1.30.1) vite: 7.1.3(@types/node@24.3.1)(jiti@2.5.1)(lightningcss@1.30.1)
vite@7.1.3(jiti@2.5.1)(lightningcss@1.30.1): vite@7.1.3(@types/node@24.3.1)(jiti@2.5.1)(lightningcss@1.30.1):
dependencies: dependencies:
esbuild: 0.25.9 esbuild: 0.25.9
fdir: 6.5.0(picomatch@4.0.3) fdir: 6.5.0(picomatch@4.0.3)
@@ -2892,22 +3256,46 @@ snapshots:
rollup: 4.48.1 rollup: 4.48.1
tinyglobby: 0.2.14 tinyglobby: 0.2.14
optionalDependencies: optionalDependencies:
'@types/node': 24.3.1
fsevents: 2.3.3 fsevents: 2.3.3
jiti: 2.5.1 jiti: 2.5.1
lightningcss: 1.30.1 lightningcss: 1.30.1
vitefu@1.1.1(vite@7.1.3(jiti@2.5.1)(lightningcss@1.30.1)): vitefu@1.1.1(vite@7.1.3(@types/node@24.3.1)(jiti@2.5.1)(lightningcss@1.30.1)):
optionalDependencies: optionalDependencies:
vite: 7.1.3(jiti@2.5.1)(lightningcss@1.30.1) vite: 7.1.3(@types/node@24.3.1)(jiti@2.5.1)(lightningcss@1.30.1)
w3c-keyname@2.2.8: {} w3c-keyname@2.2.8: {}
w3c-xmlserializer@5.0.0:
dependencies:
xml-name-validator: 5.0.0
webidl-conversions@7.0.0: {}
whatwg-encoding@3.1.1:
dependencies:
iconv-lite: 0.6.3
whatwg-mimetype@4.0.0: {}
whatwg-url@14.2.0:
dependencies:
tr46: 5.1.1
webidl-conversions: 7.0.0
which@2.0.2: which@2.0.2:
dependencies: dependencies:
isexe: 2.0.0 isexe: 2.0.0
word-wrap@1.2.5: {} word-wrap@1.2.5: {}
ws@8.18.3: {}
xml-name-validator@5.0.0: {}
xmlchars@2.2.0: {}
yallist@5.0.0: {} yallist@5.0.0: {}
yaml@1.10.2: {} yaml@1.10.2: {}
+30 -1
View File
@@ -29,7 +29,7 @@ body {
color: var(--theme-foreground); color: var(--theme-foreground);
} }
.btn { @utility btn {
@apply px-2 py-1 cursor-pointer; @apply px-2 py-1 cursor-pointer;
} }
@@ -41,9 +41,38 @@ body {
@apply border py-1 px-2 rounded; @apply border py-1 px-2 rounded;
} }
.hljs {
@apply rounded;
}
.printarea { .printarea {
visibility: hidden; visibility: hidden;
display: none; display: none;
code {
text-wrap: wrap;
}
}
img {
display: inline-block;
margin: 0;
}
.top-menu-button {
@apply hover:bg-gray-300/40 dark:hover:bg-gray-500/40 btn rounded transition-colors flex-1 text-left;
}
.max-h-half {
max-height: calc(50vh - 32px);
}
@utility h-screen-trimmed {
height: calc(100vh - 64px);
}
@utility max-h-screen-trimmed {
max-height: calc(100vh - 64px);
} }
@media print { @media print {
+15
View File
@@ -0,0 +1,15 @@
<!-- Component for highlighting menu bar actions with their respective button when Alt is pressed -->
<script lang="ts">
import type { Snippet } from "svelte";
const {
altMenu,
children
}: {
altMenu: boolean;
children: Snippet;
} = $props();
</script>
<span class={altMenu ? "underline" : ""}>{@render children()}</span>
+55 -15
View File
@@ -1,46 +1,86 @@
<script lang="ts"> <script lang="ts">
import { download } from '$lib/download'; import { download } from "$lib/download";
import DOMPurify from 'dompurify'; import DOMPurify from "dompurify";
import { marked } from 'marked'; import hljs from "highlight.js";
import { Marked } from "marked";
import { markedHighlight } from "marked-highlight";
import markedKatex from "marked-katex-extension";
const { content }: { content: string } = $props(); const { content }: { content: string } = $props();
let exportDialog: HTMLDialogElement; let exportDialog: HTMLDialogElement;
let format = $state('html'); let format = $state("html");
let processing = $state(false); let processing = $state(false);
let printContent: string = $state(''); let printContent: string = $state("");
export function showModal() { export function showModal() {
exportDialog.showModal(); exportDialog.showModal();
} }
async function getHtml() {
const marked = new Marked(
markedHighlight({
emptyLangClass: "hljs",
langPrefix: "hljs language-",
highlight(code, lang, info) {
const language = hljs.getLanguage(lang) ? lang : "plaintext";
return hljs.highlight(code, { language }).value;
}
}),
markedKatex({
throwOnError: false
})
);
const htmlContent = await marked.parse(content);
return htmlContent;
}
async function exportHtml() { async function exportHtml() {
const html = await marked.parse(content); const html = `<!DOCTYPE html>
<html>
<head>
<link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/katex@0.16.22/dist/katex.min.css" integrity="sha384-5TcZemv2l/9On385z///+d7MSYlvIEw9FuZTIdZ14vJLqWphw7e7ZPuOiCHJcFCP" crossorigin="anonymous">
<link rel="stylesheet" href="https://cdn.jsdelivr.net/gh/highlightjs/cdn-release@11.11.1/build/styles/dark.min.css">
<style>
body {
margin: 1rem 1.5rem;
font-family: ui-sans-serif, system-ui, sans-serif, "Apple Color Emoji", "Segoe UI Emoji", "Segoe UI Symbol", "Noto Color Emoji";
}
.hljs {
border-radius: 0.25rem;
}
</style>
</head>
<body>
${await getHtml()}
</body>
</html>`; </html>`;
type: 'text/html' const blob = new Blob([html], {
type: "text/html" type: "text/html"
download(blob, 'litewriter-document.html'); });
download(blob, "litewriter-document.html"); download(blob, "litewriter-document.html");
} }
printContent = DOMPurify.sanitize(await marked.parse(content)); export async function exportPdf() {
printContent = DOMPurify.sanitize(await getHtml()); printContent = DOMPurify.sanitize(await getHtml());
setTimeout(() => { setTimeout(() => {
printContent = ''; window.print();
// printContent = ""; // printContent = "";
}, 1000); }, 1000);
} }
async function exportBtn() { async function exportBtn() {
case 'html': switch (format) {
case "html": case "html":
await exportHtml(); await exportHtml();
case 'pdf': break;
case "pdf": case "pdf":
await exportPdf(); await exportPdf();
case 'odt': break;
await (await import('../opendocument')).exportOdt(content); case "odt":
await (await import("../opendocument")).exportOdt(content); await (await import("../opendocument")).exportOdt(content);
break; break;
} }
@@ -48,7 +88,7 @@
exportDialog.close(); exportDialog.close();
} }
</script> </script>
<div class="printarea prose">
<div class="printarea prose bg-white max-w-[unset]"> <div class="printarea prose bg-white max-w-[unset]">
{@html printContent} {@html printContent}
</div> </div>
+18
View File
@@ -0,0 +1,18 @@
<script lang="ts">
import type { Snippet } from "svelte";
import type { MouseEventHandler } from "svelte/elements";
const {
onclick,
children
}: {
onclick: MouseEventHandler<HTMLButtonElement>;
children: Snippet;
} = $props();
</script>
<li class="flex">
<button class="top-menu-button" {onclick}>
{@render children()}
</button>
</li>
+22
View File
@@ -0,0 +1,22 @@
<script lang="ts">
import type { Snippet } from "svelte";
const {
shownMenu,
name,
children
}: {
shownMenu: string | null;
name: string;
children: Snippet;
} = $props();
</script>
<div
class={`z-10 w-44 absolute bg-theme-floating rounded ${shownMenu === name ? "" : "hidden"} shadow`}
role="menu"
>
<ul>
{@render children()}
</ul>
</div>
+127 -93
View File
@@ -1,24 +1,58 @@
<script lang="ts"> <script lang="ts">
import { Decoration, MatchDecorator, ViewPlugin, EditorView } from '@codemirror/view'; import type { PageTheme } from "$lib/themes/theme";
import { markdown } from '@codemirror/lang-markdown'; import MenuButton from "$lib/components/MenuButton.svelte";
import CodeMirror from 'svelte-codemirror-editor'; import MenuActionList from "$lib/components/MenuActionList.svelte";
import type { PageTheme } from '$lib/themes/theme'; import MenuAction from "$lib/components/MenuAction.svelte";
import { dracula } from '$lib/themes/dracula'; import markedKatex from "marked-katex-extension";
import { enhance } from '$app/forms'; import hljs from "highlight.js";
import { onMount } from 'svelte'; import ExportDialog from "$lib/components/ExportDialog.svelte";
import MenuButton from '$lib/components/MenuButton.svelte'; import DOMPurify from "isomorphic-dompurify";
import ExportDialog from '$lib/components/ExportDialog.svelte'; import CodeMirror from "svelte-codemirror-editor";
import '../css/codemirror.css'; import AltHighlight from "$lib/components/AltHighlight.svelte";
import { onMount } from "svelte";
import { markedHighlight } from "marked-highlight";
import { Marked } from "marked";
import { markdown } from "@codemirror/lang-markdown";
import { enhance } from "$app/forms";
import { dracula } from "$lib/themes/dracula";
import { Decoration, MatchDecorator, ViewPlugin, EditorView } from "@codemirror/view";
import "katex/dist/katex.min.css";
import "highlight.js/styles/dark.min.css";
import "../css/codemirror.css";
let { data } = $props(); let { data } = $props();
let { themeName } = data; let { themeName } = data;
let content = $state(`# Heading 1 let content = $state(``);
## Heading 2
### Heading 3
**bold** *italic*`); let unsaved = $derived(content !== "");
const marked = new Marked(
markedKatex({
throwOnError: false
}),
markedHighlight({
emptyLangClass: "hljs",
langPrefix: "hljs language-",
highlight(code, lang, info) {
const language = hljs.getLanguage(lang) ? lang : "plaintext";
return hljs.highlight(code, { language }).value;
}
})
);
let documentPreview = $derived.by(() => {
const sanitized = DOMPurify.sanitize(
marked
.parse(content, {
async: false
})
.replaceAll("<pre>", '<pre class="not-prose">')
);
return sanitized;
});
let changeThemeForm: HTMLFormElement; let changeThemeForm: HTMLFormElement;
@@ -27,21 +61,38 @@
Dracula: dracula Dracula: dracula
}; };
function beforeUnload(e: BeforeUnloadEvent) {
if (!unsaved || content.trim() === "") return;
e.preventDefault();
}
$effect(() => {
unsaved;
if (unsaved) {
window.addEventListener("beforeunload", beforeUnload);
return;
}
window.removeEventListener("beforeunload", beforeUnload);
});
let cmTheme = themeOptions[themeName]?.codemirror ?? null; let cmTheme = themeOptions[themeName]?.codemirror ?? null;
let headingDecorations: { [key: string]: Decoration } = { let headingDecorations: { [key: string]: Decoration } = {
'#': Decoration.mark({ class: 'h1' }), "#": Decoration.mark({ class: "h1" }),
'##': Decoration.mark({ class: 'h2' }), "##": Decoration.mark({ class: "h2" }),
'###': Decoration.mark({ class: 'h3' }), "###": Decoration.mark({ class: "h3" }),
'####': Decoration.mark({ class: 'h4' }), "####": Decoration.mark({ class: "h4" }),
'#####': Decoration.mark({ class: 'h5' }), "#####": Decoration.mark({ class: "h5" }),
'######': Decoration.mark({ class: 'h6' }) "######": Decoration.mark({ class: "h6" })
}; };
let decorator = new MatchDecorator({ let decorator = new MatchDecorator({
regexp: /^#{1,6} .*/gm, regexp: /^#{1,6} .*/gm,
decoration(match, view, pos) { decoration(match, view, pos) {
return headingDecorations[match[0].split(' ')[0]]; return headingDecorations[match[0].split(" ")[0]];
} }
}); });
@@ -61,13 +112,13 @@
const pageTheme = themeOptions[themeName]; const pageTheme = themeOptions[themeName];
if (pageTheme?.dark) { if (pageTheme?.dark) {
document.body.classList.add('dark'); document.body.classList.add("dark");
} }
document.body.style.setProperty('--theme-background', pageTheme?.background ?? null); document.body.style.setProperty("--theme-background", pageTheme?.background ?? null);
document.body.style.setProperty('--theme-background-darker', pageTheme?.backgroundDark ?? null); document.body.style.setProperty("--theme-background-darker", pageTheme?.backgroundDark ?? null);
document.body.style.setProperty('--theme-floating', pageTheme?.floatingControls ?? null); document.body.style.setProperty("--theme-floating", pageTheme?.floatingControls ?? null);
document.body.style.setProperty('--theme-foreground', pageTheme?.foreground ?? null); document.body.style.setProperty("--theme-foreground", pageTheme?.foreground ?? null);
}); });
let shownMenu = $state<string | null>(null); let shownMenu = $state<string | null>(null);
@@ -83,7 +134,7 @@
function menuButtonClicked(menuName: string) { function menuButtonClicked(menuName: string) {
if (shownMenu === menuName) { if (shownMenu === menuName) {
shownMenu = null; shownMenu = null;
menuTriggeredWithAlt = false; altMenu = false;
return; return;
} }
@@ -93,7 +144,7 @@
function menuButtonBlur(e: FocusEvent) { function menuButtonBlur(e: FocusEvent) {
if ( if (
e.relatedTarget !== null && e.relatedTarget !== null &&
(e.relatedTarget as HTMLElement).classList.contains('top-menu-button') (e.relatedTarget as HTMLElement).classList.contains("top-menu-button")
) { ) {
return; return;
} }
@@ -106,7 +157,7 @@
} }
function newFileBtn() { function newFileBtn() {
content = ''; content = "";
menuActionComplete(); menuActionComplete();
} }
@@ -126,10 +177,10 @@
} }
let altPressed = $state(false); let altPressed = $state(false);
let menuTriggeredWithAlt = $state(false); let altMenu = $state(false);
const altMenuMapping: { [key: string]: string } = { const altMenuMapping: { [key: string]: string } = {
f: 'file', f: "file",
h: 'help' h: "help"
}; };
const altSubmenuMapping: { [menu: string]: { [key: string]: Function } } = { const altSubmenuMapping: { [menu: string]: { [key: string]: Function } } = {
file: { file: {
@@ -143,12 +194,12 @@
}; };
function keyDownListener(ev: KeyboardEvent) { function keyDownListener(ev: KeyboardEvent) {
if (menuTriggeredWithAlt && shownMenu !== null && altSubmenuMapping[shownMenu]?.[ev.key]) { if (altMenu && shownMenu !== null && altSubmenuMapping[shownMenu]?.[ev.key]) {
ev.preventDefault(); ev.preventDefault();
ev.stopPropagation(); ev.stopPropagation();
altSubmenuMapping[shownMenu][ev.key](); altSubmenuMapping[shownMenu][ev.key]();
menuTriggeredWithAlt = false; altMenu = false;
shownMenu = null; shownMenu = null;
} }
@@ -159,29 +210,32 @@
ev.preventDefault(); ev.preventDefault();
ev.stopPropagation(); ev.stopPropagation();
menuTriggeredWithAlt = true; altMenu = true;
menuButtonClicked(menu); menuButtonClicked(menu);
return; return;
} }
if (ev.key === 'Alt') { if (ev.key === "Alt") {
ev.preventDefault();
ev.stopPropagation();
altPressed = true; altPressed = true;
menuTriggeredWithAlt = false; altMenu = false;
if (shownMenu !== null) {
shownMenu = null;
}
} }
return;
} }
} }
function keyUpListener(ev: KeyboardEvent) { function keyUpListener(ev: KeyboardEvent) {
if (ev.key === 'Alt') { if (ev.key === "Alt") {
ev.preventDefault();
ev.stopPropagation();
altPressed = false; altPressed = false;
} }
} }
const ariaAttribute = EditorView.contentAttributes.of({ const ariaAttribute = EditorView.contentAttributes.of({
'aria-label': 'The Markdown editor' "aria-label": "The Markdown editor"
}); });
</script> </script>
@@ -192,8 +246,8 @@
<svelte:body onkeydown={keyDownListener} onkeyup={keyUpListener} /> <svelte:body onkeydown={keyDownListener} onkeyup={keyUpListener} />
<header class="p-4 flex justify-between bg-theme-darker"> <header class="p-4 flex justify-between bg-theme-darker fixed h-auto w-full z-10">
<ul class="flex" role="list"> <ul class="flex">
<li> <li>
<MenuButton <MenuButton
{shownMenu} {shownMenu}
@@ -202,39 +256,19 @@
buttonClicked={menuButtonClicked} buttonClicked={menuButtonClicked}
blurEvent={menuButtonBlur} blurEvent={menuButtonBlur}
> >
<span class={altPressed ? 'underline' : ''}>F</span>ile <AltHighlight altMenu={altPressed}>F</AltHighlight>ile
</MenuButton> </MenuButton>
<div <MenuActionList {shownMenu} name="file">
class={`z-10 w-44 absolute bg-theme-floating rounded ${shownMenu === 'file' ? '' : 'hidden'} shadow`} <MenuAction onclick={newFileBtn}>
role="menu" <AltHighlight {altMenu}>N</AltHighlight>ew File
> </MenuAction>
<ul> <MenuAction onclick={exportFileBtn}>
<li class="flex"> <AltHighlight {altMenu}>E</AltHighlight>xport
<button </MenuAction>
class="hover:bg-gray-300/40 dark:hover:bg-gray-500/40 btn rounded transition-colors flex-1 text-left top-menu-button" <MenuAction onclick={printFileBtn}>
onclick={newFileBtn} <AltHighlight {altMenu}>P</AltHighlight>rint...
> </MenuAction>
<span class={menuTriggeredWithAlt ? 'underline' : ''}>N</span>ew File </MenuActionList>
</button>
</li>
<li class="flex">
<button
class="hover:bg-gray-300/40 dark:hover:bg-gray-500/40 btn rounded transition-colors flex-1 text-left top-menu-button"
onclick={exportFileBtn}
>
<span class={menuTriggeredWithAlt ? 'underline' : ''}>E</span>xport...
</button>
</li>
<li class="flex">
<button
class="hover:bg-gray-300/40 dark:hover:bg-gray-500/40 btn rounded transition-colors flex-1 text-left top-menu-button"
onclick={printFileBtn}
>
<span class={menuTriggeredWithAlt ? 'underline' : ''}>P</span>rint...
</button>
</li>
</ul>
</div>
</li> </li>
<li> <li>
<MenuButton <MenuButton
@@ -244,22 +278,13 @@
buttonClicked={menuButtonClicked} buttonClicked={menuButtonClicked}
blurEvent={menuButtonBlur} blurEvent={menuButtonBlur}
> >
<span class={altPressed ? 'underline' : ''}>H</span>elp <AltHighlight altMenu={altPressed}>H</AltHighlight>elp
</MenuButton> </MenuButton>
<div <MenuActionList {shownMenu} name="help">
class={`z-10 w-44 absolute bg-theme-floating rounded ${shownMenu === 'help' ? '' : 'hidden'} shadow`} <MenuAction onclick={aboutBtn}>
role="menu" <AltHighlight {altMenu}>A</AltHighlight>bout
> </MenuAction>
<ul> </MenuActionList>
<li class="flex">
<button
class="hover:bg-gray-300/40 dark:hover:bg-gray-500/40 btn rounded transition-colors flex-1 text-left top-menu-button"
>
<span class={menuTriggeredWithAlt ? 'underline' : ''}>A</span>bout
</button>
</li>
</ul>
</div>
</li> </li>
</ul> </ul>
<form method="POST" use:enhance action="?/changeTheme" bind:this={changeThemeForm}> <form method="POST" use:enhance action="?/changeTheme" bind:this={changeThemeForm}>
@@ -279,9 +304,18 @@
<ExportDialog bind:this={exportDialog} {content} /> <ExportDialog bind:this={exportDialog} {content} />
<CodeMirror <div class="grid grid-rows-2 lg:grid-cols-2 pt-16 max-h-screen">
<CodeMirror
bind:value={content} bind:value={content}
theme={cmTheme} theme={cmTheme}
lang={markdown()} lang={markdown()}
extensions={[headingPlugin, ariaAttribute]} extensions={[headingPlugin, ariaAttribute]}
/> class="overflow-auto lg:h-screen-trimmed! lg:max-w-[50vw]"
/>
<div
id="livePreview"
class="prose bg-white max-w-full p-4 lg:h-screen-trimmed! lg:max-w-[50vw] overflow-auto"
>
{@html documentPreview}
</div>
</div>
+4
View File
@@ -1,6 +1,10 @@
{ {
"name": "litewriter", "name": "litewriter",
"description": "A Markdown editor with a powerful exporter.", "description": "A Markdown editor with a powerful exporter.",
"categories": [
"productivity",
"utilities"
],
"start_url": "/", "start_url": "/",
"display": "standalone", "display": "standalone",
"prefer_related_applications": false, "prefer_related_applications": false,