From 482e13569ff0a3a289089e883d53007c2c5f063f Mon Sep 17 00:00:00 2001 From: Cory Dransfeldt Date: Fri, 18 Oct 2024 18:12:30 -0700 Subject: [PATCH] chore: use remote search --- .eleventy.js | 87 +- config/events/index.js | 59 + config/plugins/css-config.js | 33 + package-lock.json | 1810 ++++++++------- package.json | 13 +- scripts/install-components.mjs | 31 - src/assets/js/components/api-text.js | 69 - src/assets/js/components/mastodon-post.js | 115 - src/assets/js/components/mini-search.js | 2036 ----------------- src/assets/js/components/select-pagination.js | 48 - src/assets/js/components/theme-toggle.js | 45 - .../js/components/youtube-video-element.js | 547 ----- src/assets/{js => scripts}/index.js | 2 - src/assets/{css => styles}/base/index.css | 0 .../{css => styles}/components/badge-grid.css | 0 .../{css => styles}/components/banners.css | 0 .../{css => styles}/components/buttons.css | 0 .../{css => styles}/components/forms.css | 0 .../components/mastodon-post.css | 0 .../{css => styles}/components/media-grid.css | 0 .../{css => styles}/components/menu.css | 0 .../{css => styles}/components/modal.css | 0 .../components/music-chart.css | 0 .../{css => styles}/components/paginator.css | 0 .../components/progress-bar.css | 0 .../components/tab-buttons.css | 0 .../components/text-toggle.css | 0 .../components/theme-toggle.css | 0 src/assets/{css => styles}/defaults/fonts.css | 0 src/assets/{css => styles}/defaults/vars.css | 0 src/assets/{css => styles}/feed.xsl | 0 src/assets/{css => styles}/index.css | 0 src/assets/{css => styles}/pages/about.css | 0 src/assets/{css => styles}/pages/articles.css | 0 src/assets/{css => styles}/pages/blogroll.css | 0 src/assets/{css => styles}/pages/books.css | 0 src/assets/{css => styles}/pages/contact.css | 0 src/assets/{css => styles}/pages/feeds.css | 0 src/assets/{css => styles}/pages/links.css | 0 src/assets/{css => styles}/pages/music.css | 0 src/assets/{css => styles}/pages/watching.css | 0 src/assets/{css => styles}/pages/webrings.css | 0 src/assets/{css => styles}/plugins/prism.css | 0 src/assets/{css => styles}/reset.css | 0 src/assets/{css => styles}/styles.json | 0 src/includes/base.liquid | 16 +- .../partials/blocks/mastodon-post.liquid | 1 + .../partials/blocks/now-playing.liquid | 1 + .../partials/blocks/youtube-player.liquid | 1 + src/includes/partials/nav/paginator.liquid | 1 + src/includes/partials/nav/theme-toggle.liquid | 1 + src/pages/feeds/json/all.liquid | 1 + workers/dynamic-pages/utils/generators.js | 2 +- 53 files changed, 1184 insertions(+), 3735 deletions(-) create mode 100644 config/events/index.js create mode 100644 config/plugins/css-config.js delete mode 100644 scripts/install-components.mjs delete mode 100644 src/assets/js/components/api-text.js delete mode 100644 src/assets/js/components/mastodon-post.js delete mode 100644 src/assets/js/components/mini-search.js delete mode 100644 src/assets/js/components/select-pagination.js delete mode 100644 src/assets/js/components/theme-toggle.js delete mode 100644 src/assets/js/components/youtube-video-element.js rename src/assets/{js => scripts}/index.js (99%) rename src/assets/{css => styles}/base/index.css (100%) rename src/assets/{css => styles}/components/badge-grid.css (100%) rename src/assets/{css => styles}/components/banners.css (100%) rename src/assets/{css => styles}/components/buttons.css (100%) rename src/assets/{css => styles}/components/forms.css (100%) rename src/assets/{css => styles}/components/mastodon-post.css (100%) rename src/assets/{css => styles}/components/media-grid.css (100%) rename src/assets/{css => styles}/components/menu.css (100%) rename src/assets/{css => styles}/components/modal.css (100%) rename src/assets/{css => styles}/components/music-chart.css (100%) rename src/assets/{css => styles}/components/paginator.css (100%) rename src/assets/{css => styles}/components/progress-bar.css (100%) rename src/assets/{css => styles}/components/tab-buttons.css (100%) rename src/assets/{css => styles}/components/text-toggle.css (100%) rename src/assets/{css => styles}/components/theme-toggle.css (100%) rename src/assets/{css => styles}/defaults/fonts.css (100%) rename src/assets/{css => styles}/defaults/vars.css (100%) rename src/assets/{css => styles}/feed.xsl (100%) rename src/assets/{css => styles}/index.css (100%) rename src/assets/{css => styles}/pages/about.css (100%) rename src/assets/{css => styles}/pages/articles.css (100%) rename src/assets/{css => styles}/pages/blogroll.css (100%) rename src/assets/{css => styles}/pages/books.css (100%) rename src/assets/{css => styles}/pages/contact.css (100%) rename src/assets/{css => styles}/pages/feeds.css (100%) rename src/assets/{css => styles}/pages/links.css (100%) rename src/assets/{css => styles}/pages/music.css (100%) rename src/assets/{css => styles}/pages/watching.css (100%) rename src/assets/{css => styles}/pages/webrings.css (100%) rename src/assets/{css => styles}/plugins/prism.css (100%) rename src/assets/{css => styles}/reset.css (100%) rename src/assets/{css => styles}/styles.json (100%) diff --git a/.eleventy.js b/.eleventy.js index fff902cb..5e0c47d5 100644 --- a/.eleventy.js +++ b/.eleventy.js @@ -1,59 +1,49 @@ +import { createRequire } from 'module' import dotenvFlow from 'dotenv-flow' import filters from './config/filters/index.js' +import htmlmin from 'html-minifier-terser' import markdownIt from 'markdown-it' import markdownItAnchor from 'markdown-it-anchor' import markdownItFootnote from 'markdown-it-footnote' import markdownItPrism from 'markdown-it-prism' -import EleventyVitePlugin from '@11ty/eleventy-plugin-vite' -import { ViteMinifyPlugin } from 'vite-plugin-minify' -import { resolve } from 'path'; import syntaxHighlight from '@11ty/eleventy-plugin-syntaxhighlight' import tablerIcons from '@cdransf/eleventy-plugin-tabler-icons' +import { copyErrorPages, minifyJsComponents } from './config/events/index.js' import { albumReleasesCalendar } from './config/collections/index.js' +import { cssConfig } from './config/plugins/css-config.js' // load .env dotenvFlow.config() +// get app version +const require = createRequire(import.meta.url) +const appVersion = require('./package.json').version + export default async function (eleventyConfig) { eleventyConfig.addPlugin(syntaxHighlight) eleventyConfig.addPlugin(tablerIcons) - eleventyConfig.addPassthroughCopy('src/assets') - eleventyConfig.addPassthroughCopy('_redirects') - eleventyConfig.addPassthroughCopy('_headers') - - eleventyConfig.addPlugin(EleventyVitePlugin, { - tempFolderName: '.11ty-vite', - viteOptions: { - clearScreen: false, - appType: 'mpa', - server: { - middlewareMode: true, - }, - assetsInclude: ['src/assets/fonts/*.woff2'], - build: { - emptyOutDir: true, - rollupOptions: { - external: ['/js/script.js'], - input: { - main: resolve('./src/assets/index.js'), - }, - output: { - assetFileNames: 'assets/css/[name][extname]', - chunkFileNames: 'assets/js/[name].[hash].js', - entryFileNames: 'assets/js/[name].[hash].js' - }, - }, - }, - plugins: [ViteMinifyPlugin({})], - }, - }) + if (process.env.ELEVENTY_PRODUCTION) eleventyConfig.addPlugin(cssConfig) eleventyConfig.setServerOptions({ domdiff: false }) eleventyConfig.setWatchThrottleWaitTime(200) eleventyConfig.setQuietMode(true) eleventyConfig.configureErrorReporting({ allowMissingExtensions: true }) - eleventyConfig.setLiquidOptions({ jsTruthy: true }) + eleventyConfig.setLiquidOptions({ + jsTruthy: true, + }) + + eleventyConfig.addPassthroughCopy('src/assets') + eleventyConfig.addPassthroughCopy('_redirects') + eleventyConfig.addPassthroughCopy('_headers') + eleventyConfig.addPassthroughCopy({ + 'node_modules/@cdransf/api-text/api-text.js': 'assets/scripts/components/api-text.js', + 'node_modules/@cdransf/select-pagination/select-pagination.js': 'assets/scripts/components/select-pagination.js', + 'node_modules/@cdransf/theme-toggle/theme-toggle.js': 'assets/scripts/components/theme-toggle.js', + 'node_modules/@daviddarnes/mastodon-post/mastodon-post.js': 'assets/scripts/components/mastodon-post.js', + 'node_modules/minisearch/dist/umd/index.js': 'assets/scripts/components/minisearch.js', + 'node_modules/youtube-video-element/youtube-video-element.js': 'assets/scripts/components/youtube-video-element.js' + }) eleventyConfig.addCollection('albumReleasesCalendar', albumReleasesCalendar) @@ -66,7 +56,6 @@ export default async function (eleventyConfig) { }) md.use(markdownItFootnote) md.use(markdownItPrism) - eleventyConfig.setLibrary('md', md) eleventyConfig.addLiquidFilter('markdown', (content) => { @@ -78,6 +67,34 @@ export default async function (eleventyConfig) { eleventyConfig.addLiquidFilter(filterName, filters[filterName]) }) + eleventyConfig.addShortcode('appVersion', () => appVersion) + + // events + if (process.env.ELEVENTY_PRODUCTION) eleventyConfig.on('afterBuild', copyErrorPages) + if (process.env.ELEVENTY_PRODUCTION) eleventyConfig.on('afterBuild', minifyJsComponents) + + // transforms + if (process.env.ELEVENTY_PRODUCTION) eleventyConfig.addTransform('html-minify', (content, path) => { + if (path && path.endsWith('.html')) { + return htmlmin.minify(content, { + collapseBooleanAttributes: true, + collapseWhitespace: true, + decodeEntities: true, + includeAutoGeneratedTags: false, + minifyCSS: true, + minifyJS: true, + minifyURLs: true, + noNewlinesBeforeTagClose: true, + quoteCharacter: '"', + removeComments: true, + sortAttributes: true, + sortClassName: true, + useShortDoctype: true, + }) + } + return content + }) + return { passthroughFileCopy: true, dir: { diff --git a/config/events/index.js b/config/events/index.js new file mode 100644 index 00000000..df9378f2 --- /dev/null +++ b/config/events/index.js @@ -0,0 +1,59 @@ +import fs from 'fs' +import path from 'path' +import { minify } from 'terser' + +const errorPages = ['404', '500', '1000', 'broken', 'error', 'js-challenge', 'not-allowed', 'rate-limit'] + +export const copyErrorPages = () => { + errorPages.forEach((errorPage) => { + const sourcePath = path.join('_site', errorPage, 'index.html') + const destinationPath = path.join('_site', `${errorPage}.html`) + const directoryPath = path.join('_site', errorPage) + + fs.copyFile(sourcePath, destinationPath, (err) => { + if (err) { + console.error(`Error copying ${errorPage} page:`, err) + return + } + + fs.unlink(sourcePath, (unlinkErr) => { + if (unlinkErr) { + console.error(`Error deleting source file for ${errorPage} page:`, unlinkErr) + return + } + + fs.rmdir(directoryPath, (rmdirErr) => { + if (rmdirErr) console.error(`Error removing directory for ${errorPage} page:`, rmdirErr) + }) + }) + }) + }) +} + +export const minifyJsComponents = async () => { + const scriptsDir = '_site/assets/scripts' + + const minifyJsFilesInDir = async (dir) => { + const files = fs.readdirSync(dir) + for (const fileName of files) { + const filePath = path.join(dir, fileName) + const stat = fs.statSync(filePath) + + if (stat.isDirectory()) { + await minifyJsFilesInDir(filePath) + } else if (fileName.endsWith('.js')) { + const fileContent = fs.readFileSync(filePath, 'utf8') + const minified = await minify(fileContent) + if (minified.error) { + console.error(`Error minifying ${filePath}:`, minified.error) + } else { + fs.writeFileSync(filePath, minified.code) + } + } else { + console.log(`No .js files to minify in ${filePath}`) + } + } + } + + await minifyJsFilesInDir(scriptsDir) +} \ No newline at end of file diff --git a/config/plugins/css-config.js b/config/plugins/css-config.js new file mode 100644 index 00000000..fd55c8d8 --- /dev/null +++ b/config/plugins/css-config.js @@ -0,0 +1,33 @@ +import fs from 'node:fs/promises' +import path from 'node:path' +import postcss from 'postcss' +import postcssImport from 'postcss-import' +import postcssImportExtGlob from 'postcss-import-ext-glob' +import autoprefixer from 'autoprefixer' +import cssnano from 'cssnano' + +export const cssConfig = (eleventyConfig) => { + eleventyConfig.addTemplateFormats('css') + eleventyConfig.addExtension('css', { + outputFileExtension: 'css', + compile: async (inputContent, inputPath) => { + const outputPath = '_site/assets/css/index.css' + + if (inputPath.endsWith('index.css')) { + return async () => { + let result = await postcss([ + postcssImportExtGlob, + postcssImport, + autoprefixer, + cssnano, + ]).process(inputContent, { from: inputPath }) + + await fs.mkdir(path.dirname(outputPath), { recursive: true }) + await fs.writeFile(outputPath, result.css) + + return result.css + } + } + }, + }) +} \ No newline at end of file diff --git a/package-lock.json b/package-lock.json index 65fa8a5e..2330bd69 100644 --- a/package-lock.json +++ b/package-lock.json @@ -1,12 +1,12 @@ { "name": "coryd.dev", - "version": "1.4.0", + "version": "1.3.0", "lockfileVersion": 3, "requires": true, "packages": { "": { "name": "coryd.dev", - "version": "1.4.0", + "version": "1.3.0", "license": "MIT", "dependencies": { "@cdransf/api-text": "^1.5.0", @@ -20,12 +20,14 @@ "devDependencies": { "@11ty/eleventy": "v3.0.0", "@11ty/eleventy-plugin-syntaxhighlight": "^5.0.0", - "@11ty/eleventy-plugin-vite": "5.0.0", "@cdransf/eleventy-plugin-tabler-icons": "^2.0.3", "@supabase/supabase-js": "^2.45.5", + "autoprefixer": "^10.4.20", + "cssnano": "^7.0.6", "dotenv-flow": "^4.1.0", "express": "4.21.1", "fast-xml-parser": "^4.5.0", + "html-minifier-terser": "^7.2.0", "html-to-text": "^9.0.5", "ics": "^3.8.1", "linkedom": "0.18.5", @@ -34,10 +36,13 @@ "markdown-it-anchor": "^9.2.0", "markdown-it-footnote": "^4.0.0", "markdown-it-prism": "^2.3.0", + "postcss": "^8.4.47", + "postcss-import": "^16.1.0", + "postcss-import-ext-glob": "^2.1.1", "rimraf": "^6.0.1", "slugify": "^1.6.6", - "truncate-html": "^1.1.2", - "vite-plugin-minify": "2.0.0" + "terser": "^5.36.0", + "truncate-html": "^1.1.2" }, "engines": { "node": "20.x" @@ -197,24 +202,6 @@ "url": "https://opencollective.com/11ty" } }, - "node_modules/@11ty/eleventy-plugin-vite": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/@11ty/eleventy-plugin-vite/-/eleventy-plugin-vite-5.0.0.tgz", - "integrity": "sha512-4rpWzgtIzcc2m80LLz1osM0OsrVQ0GKQK4cQX4rIWrUpd64PeTxyRyJXHGglkysNHXMtQb9HV9E5dLUEfmFx/g==", - "dev": true, - "license": "MIT", - "dependencies": { - "@11ty/eleventy-utils": "^1.0.3", - "vite": "^5.4.8" - }, - "engines": { - "node": ">=18" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/11ty" - } - }, "node_modules/@11ty/eleventy-utils": { "version": "1.0.3", "resolved": "https://registry.npmjs.org/@11ty/eleventy-utils/-/eleventy-utils-1.0.3.tgz", @@ -415,397 +402,6 @@ "integrity": "sha512-6AMQ/tl6uI3wXknv8exYJguym/bPHxIW5XOYg7aWCQtMbP4XUDAsWp2pv4o9wtesIF8K7CssNPR93qFOh7D8lw==", "license": "MIT" }, - "node_modules/@esbuild/aix-ppc64": { - "version": "0.21.5", - "resolved": "https://registry.npmjs.org/@esbuild/aix-ppc64/-/aix-ppc64-0.21.5.tgz", - "integrity": "sha512-1SDgH6ZSPTlggy1yI6+Dbkiz8xzpHJEVAlF/AM1tHPLsf5STom9rwtjE4hKAF20FfXXNTFqEYXyJNWh1GiZedQ==", - "cpu": [ - "ppc64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "aix" - ], - "engines": { - "node": ">=12" - } - }, - "node_modules/@esbuild/android-arm": { - "version": "0.21.5", - "resolved": "https://registry.npmjs.org/@esbuild/android-arm/-/android-arm-0.21.5.tgz", - "integrity": "sha512-vCPvzSjpPHEi1siZdlvAlsPxXl7WbOVUBBAowWug4rJHb68Ox8KualB+1ocNvT5fjv6wpkX6o/iEpbDrf68zcg==", - "cpu": [ - "arm" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "android" - ], - "engines": { - "node": ">=12" - } - }, - "node_modules/@esbuild/android-arm64": { - "version": "0.21.5", - "resolved": "https://registry.npmjs.org/@esbuild/android-arm64/-/android-arm64-0.21.5.tgz", - "integrity": "sha512-c0uX9VAUBQ7dTDCjq+wdyGLowMdtR/GoC2U5IYk/7D1H1JYC0qseD7+11iMP2mRLN9RcCMRcjC4YMclCzGwS/A==", - "cpu": [ - "arm64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "android" - ], - "engines": { - "node": ">=12" - } - }, - "node_modules/@esbuild/android-x64": { - "version": "0.21.5", - "resolved": "https://registry.npmjs.org/@esbuild/android-x64/-/android-x64-0.21.5.tgz", - "integrity": "sha512-D7aPRUUNHRBwHxzxRvp856rjUHRFW1SdQATKXH2hqA0kAZb1hKmi02OpYRacl0TxIGz/ZmXWlbZgjwWYaCakTA==", - "cpu": [ - "x64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "android" - ], - "engines": { - "node": ">=12" - } - }, - "node_modules/@esbuild/darwin-arm64": { - "version": "0.21.5", - "resolved": "https://registry.npmjs.org/@esbuild/darwin-arm64/-/darwin-arm64-0.21.5.tgz", - "integrity": "sha512-DwqXqZyuk5AiWWf3UfLiRDJ5EDd49zg6O9wclZ7kUMv2WRFr4HKjXp/5t8JZ11QbQfUS6/cRCKGwYhtNAY88kQ==", - "cpu": [ - "arm64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "darwin" - ], - "engines": { - "node": ">=12" - } - }, - "node_modules/@esbuild/darwin-x64": { - "version": "0.21.5", - "resolved": "https://registry.npmjs.org/@esbuild/darwin-x64/-/darwin-x64-0.21.5.tgz", - "integrity": "sha512-se/JjF8NlmKVG4kNIuyWMV/22ZaerB+qaSi5MdrXtd6R08kvs2qCN4C09miupktDitvh8jRFflwGFBQcxZRjbw==", - "cpu": [ - "x64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "darwin" - ], - "engines": { - "node": ">=12" - } - }, - "node_modules/@esbuild/freebsd-arm64": { - "version": "0.21.5", - "resolved": "https://registry.npmjs.org/@esbuild/freebsd-arm64/-/freebsd-arm64-0.21.5.tgz", - "integrity": "sha512-5JcRxxRDUJLX8JXp/wcBCy3pENnCgBR9bN6JsY4OmhfUtIHe3ZW0mawA7+RDAcMLrMIZaf03NlQiX9DGyB8h4g==", - "cpu": [ - "arm64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "freebsd" - ], - "engines": { - "node": ">=12" - } - }, - "node_modules/@esbuild/freebsd-x64": { - "version": "0.21.5", - "resolved": "https://registry.npmjs.org/@esbuild/freebsd-x64/-/freebsd-x64-0.21.5.tgz", - "integrity": "sha512-J95kNBj1zkbMXtHVH29bBriQygMXqoVQOQYA+ISs0/2l3T9/kj42ow2mpqerRBxDJnmkUDCaQT/dfNXWX/ZZCQ==", - "cpu": [ - "x64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "freebsd" - ], - "engines": { - "node": ">=12" - } - }, - "node_modules/@esbuild/linux-arm": { - "version": "0.21.5", - "resolved": "https://registry.npmjs.org/@esbuild/linux-arm/-/linux-arm-0.21.5.tgz", - "integrity": "sha512-bPb5AHZtbeNGjCKVZ9UGqGwo8EUu4cLq68E95A53KlxAPRmUyYv2D6F0uUI65XisGOL1hBP5mTronbgo+0bFcA==", - "cpu": [ - "arm" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ], - "engines": { - "node": ">=12" - } - }, - "node_modules/@esbuild/linux-arm64": { - "version": "0.21.5", - "resolved": "https://registry.npmjs.org/@esbuild/linux-arm64/-/linux-arm64-0.21.5.tgz", - "integrity": "sha512-ibKvmyYzKsBeX8d8I7MH/TMfWDXBF3db4qM6sy+7re0YXya+K1cem3on9XgdT2EQGMu4hQyZhan7TeQ8XkGp4Q==", - "cpu": [ - "arm64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ], - "engines": { - "node": ">=12" - } - }, - "node_modules/@esbuild/linux-ia32": { - "version": "0.21.5", - "resolved": "https://registry.npmjs.org/@esbuild/linux-ia32/-/linux-ia32-0.21.5.tgz", - "integrity": "sha512-YvjXDqLRqPDl2dvRODYmmhz4rPeVKYvppfGYKSNGdyZkA01046pLWyRKKI3ax8fbJoK5QbxblURkwK/MWY18Tg==", - "cpu": [ - "ia32" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ], - "engines": { - "node": ">=12" - } - }, - "node_modules/@esbuild/linux-loong64": { - "version": "0.21.5", - "resolved": "https://registry.npmjs.org/@esbuild/linux-loong64/-/linux-loong64-0.21.5.tgz", - "integrity": "sha512-uHf1BmMG8qEvzdrzAqg2SIG/02+4/DHB6a9Kbya0XDvwDEKCoC8ZRWI5JJvNdUjtciBGFQ5PuBlpEOXQj+JQSg==", - "cpu": [ - "loong64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ], - "engines": { - "node": ">=12" - } - }, - "node_modules/@esbuild/linux-mips64el": { - "version": "0.21.5", - "resolved": "https://registry.npmjs.org/@esbuild/linux-mips64el/-/linux-mips64el-0.21.5.tgz", - "integrity": "sha512-IajOmO+KJK23bj52dFSNCMsz1QP1DqM6cwLUv3W1QwyxkyIWecfafnI555fvSGqEKwjMXVLokcV5ygHW5b3Jbg==", - "cpu": [ - "mips64el" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ], - "engines": { - "node": ">=12" - } - }, - "node_modules/@esbuild/linux-ppc64": { - "version": "0.21.5", - "resolved": "https://registry.npmjs.org/@esbuild/linux-ppc64/-/linux-ppc64-0.21.5.tgz", - "integrity": "sha512-1hHV/Z4OEfMwpLO8rp7CvlhBDnjsC3CttJXIhBi+5Aj5r+MBvy4egg7wCbe//hSsT+RvDAG7s81tAvpL2XAE4w==", - "cpu": [ - "ppc64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ], - "engines": { - "node": ">=12" - } - }, - "node_modules/@esbuild/linux-riscv64": { - "version": "0.21.5", - "resolved": "https://registry.npmjs.org/@esbuild/linux-riscv64/-/linux-riscv64-0.21.5.tgz", - "integrity": "sha512-2HdXDMd9GMgTGrPWnJzP2ALSokE/0O5HhTUvWIbD3YdjME8JwvSCnNGBnTThKGEB91OZhzrJ4qIIxk/SBmyDDA==", - "cpu": [ - "riscv64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ], - "engines": { - "node": ">=12" - } - }, - "node_modules/@esbuild/linux-s390x": { - "version": "0.21.5", - "resolved": "https://registry.npmjs.org/@esbuild/linux-s390x/-/linux-s390x-0.21.5.tgz", - "integrity": "sha512-zus5sxzqBJD3eXxwvjN1yQkRepANgxE9lgOW2qLnmr8ikMTphkjgXu1HR01K4FJg8h1kEEDAqDcZQtbrRnB41A==", - "cpu": [ - "s390x" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ], - "engines": { - "node": ">=12" - } - }, - "node_modules/@esbuild/linux-x64": { - "version": "0.21.5", - "resolved": "https://registry.npmjs.org/@esbuild/linux-x64/-/linux-x64-0.21.5.tgz", - "integrity": "sha512-1rYdTpyv03iycF1+BhzrzQJCdOuAOtaqHTWJZCWvijKD2N5Xu0TtVC8/+1faWqcP9iBCWOmjmhoH94dH82BxPQ==", - "cpu": [ - "x64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ], - "engines": { - "node": ">=12" - } - }, - "node_modules/@esbuild/netbsd-x64": { - "version": "0.21.5", - "resolved": "https://registry.npmjs.org/@esbuild/netbsd-x64/-/netbsd-x64-0.21.5.tgz", - "integrity": "sha512-Woi2MXzXjMULccIwMnLciyZH4nCIMpWQAs049KEeMvOcNADVxo0UBIQPfSmxB3CWKedngg7sWZdLvLczpe0tLg==", - "cpu": [ - "x64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "netbsd" - ], - "engines": { - "node": ">=12" - } - }, - "node_modules/@esbuild/openbsd-x64": { - "version": "0.21.5", - "resolved": "https://registry.npmjs.org/@esbuild/openbsd-x64/-/openbsd-x64-0.21.5.tgz", - "integrity": "sha512-HLNNw99xsvx12lFBUwoT8EVCsSvRNDVxNpjZ7bPn947b8gJPzeHWyNVhFsaerc0n3TsbOINvRP2byTZ5LKezow==", - "cpu": [ - "x64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "openbsd" - ], - "engines": { - "node": ">=12" - } - }, - "node_modules/@esbuild/sunos-x64": { - "version": "0.21.5", - "resolved": "https://registry.npmjs.org/@esbuild/sunos-x64/-/sunos-x64-0.21.5.tgz", - "integrity": "sha512-6+gjmFpfy0BHU5Tpptkuh8+uw3mnrvgs+dSPQXQOv3ekbordwnzTVEb4qnIvQcYXq6gzkyTnoZ9dZG+D4garKg==", - "cpu": [ - "x64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "sunos" - ], - "engines": { - "node": ">=12" - } - }, - "node_modules/@esbuild/win32-arm64": { - "version": "0.21.5", - "resolved": "https://registry.npmjs.org/@esbuild/win32-arm64/-/win32-arm64-0.21.5.tgz", - "integrity": "sha512-Z0gOTd75VvXqyq7nsl93zwahcTROgqvuAcYDUr+vOv8uHhNSKROyU961kgtCD1e95IqPKSQKH7tBTslnS3tA8A==", - "cpu": [ - "arm64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "win32" - ], - "engines": { - "node": ">=12" - } - }, - "node_modules/@esbuild/win32-ia32": { - "version": "0.21.5", - "resolved": "https://registry.npmjs.org/@esbuild/win32-ia32/-/win32-ia32-0.21.5.tgz", - "integrity": "sha512-SWXFF1CL2RVNMaVs+BBClwtfZSvDgtL//G/smwAc5oVK/UPu2Gu9tIaRgFmYFFKrmg3SyAjSrElf0TiJ1v8fYA==", - "cpu": [ - "ia32" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "win32" - ], - "engines": { - "node": ">=12" - } - }, - "node_modules/@esbuild/win32-x64": { - "version": "0.21.5", - "resolved": "https://registry.npmjs.org/@esbuild/win32-x64/-/win32-x64-0.21.5.tgz", - "integrity": "sha512-tQd/1efJuzPC6rCFwEvLtci/xNFcTZknmXs98FYDfGE4wP9ClFV98nyKrzJKVPMhdDnjzLhdUyMX4PsQAPjwIw==", - "cpu": [ - "x64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "win32" - ], - "engines": { - "node": ">=12" - } - }, "node_modules/@isaacs/cliui": { "version": "8.0.2", "resolved": "https://registry.npmjs.org/@isaacs/cliui/-/cliui-8.0.2.tgz", @@ -937,230 +533,6 @@ "node": ">=14" } }, - "node_modules/@rollup/rollup-android-arm-eabi": { - "version": "4.24.0", - "resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm-eabi/-/rollup-android-arm-eabi-4.24.0.tgz", - "integrity": "sha512-Q6HJd7Y6xdB48x8ZNVDOqsbh2uByBhgK8PiQgPhwkIw/HC/YX5Ghq2mQY5sRMZWHb3VsFkWooUVOZHKr7DmDIA==", - "cpu": [ - "arm" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "android" - ] - }, - "node_modules/@rollup/rollup-android-arm64": { - "version": "4.24.0", - "resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm64/-/rollup-android-arm64-4.24.0.tgz", - "integrity": "sha512-ijLnS1qFId8xhKjT81uBHuuJp2lU4x2yxa4ctFPtG+MqEE6+C5f/+X/bStmxapgmwLwiL3ih122xv8kVARNAZA==", - "cpu": [ - "arm64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "android" - ] - }, - "node_modules/@rollup/rollup-darwin-arm64": { - "version": "4.24.0", - "resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-arm64/-/rollup-darwin-arm64-4.24.0.tgz", - "integrity": "sha512-bIv+X9xeSs1XCk6DVvkO+S/z8/2AMt/2lMqdQbMrmVpgFvXlmde9mLcbQpztXm1tajC3raFDqegsH18HQPMYtA==", - "cpu": [ - "arm64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "darwin" - ] - }, - "node_modules/@rollup/rollup-darwin-x64": { - "version": "4.24.0", - "resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-x64/-/rollup-darwin-x64-4.24.0.tgz", - "integrity": "sha512-X6/nOwoFN7RT2svEQWUsW/5C/fYMBe4fnLK9DQk4SX4mgVBiTA9h64kjUYPvGQ0F/9xwJ5U5UfTbl6BEjaQdBQ==", - "cpu": [ - "x64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "darwin" - ] - }, - "node_modules/@rollup/rollup-linux-arm-gnueabihf": { - "version": "4.24.0", - "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-gnueabihf/-/rollup-linux-arm-gnueabihf-4.24.0.tgz", - "integrity": "sha512-0KXvIJQMOImLCVCz9uvvdPgfyWo93aHHp8ui3FrtOP57svqrF/roSSR5pjqL2hcMp0ljeGlU4q9o/rQaAQ3AYA==", - "cpu": [ - "arm" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ] - }, - "node_modules/@rollup/rollup-linux-arm-musleabihf": { - "version": "4.24.0", - "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-musleabihf/-/rollup-linux-arm-musleabihf-4.24.0.tgz", - "integrity": "sha512-it2BW6kKFVh8xk/BnHfakEeoLPv8STIISekpoF+nBgWM4d55CZKc7T4Dx1pEbTnYm/xEKMgy1MNtYuoA8RFIWw==", - "cpu": [ - "arm" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ] - }, - "node_modules/@rollup/rollup-linux-arm64-gnu": { - "version": "4.24.0", - "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-gnu/-/rollup-linux-arm64-gnu-4.24.0.tgz", - "integrity": "sha512-i0xTLXjqap2eRfulFVlSnM5dEbTVque/3Pi4g2y7cxrs7+a9De42z4XxKLYJ7+OhE3IgxvfQM7vQc43bwTgPwA==", - "cpu": [ - "arm64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ] - }, - "node_modules/@rollup/rollup-linux-arm64-musl": { - "version": "4.24.0", - "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-musl/-/rollup-linux-arm64-musl-4.24.0.tgz", - "integrity": "sha512-9E6MKUJhDuDh604Qco5yP/3qn3y7SLXYuiC0Rpr89aMScS2UAmK1wHP2b7KAa1nSjWJc/f/Lc0Wl1L47qjiyQw==", - "cpu": [ - "arm64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ] - }, - "node_modules/@rollup/rollup-linux-powerpc64le-gnu": { - "version": "4.24.0", - "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-powerpc64le-gnu/-/rollup-linux-powerpc64le-gnu-4.24.0.tgz", - "integrity": "sha512-2XFFPJ2XMEiF5Zi2EBf4h73oR1V/lycirxZxHZNc93SqDN/IWhYYSYj8I9381ikUFXZrz2v7r2tOVk2NBwxrWw==", - "cpu": [ - "ppc64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ] - }, - "node_modules/@rollup/rollup-linux-riscv64-gnu": { - "version": "4.24.0", - "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-riscv64-gnu/-/rollup-linux-riscv64-gnu-4.24.0.tgz", - "integrity": "sha512-M3Dg4hlwuntUCdzU7KjYqbbd+BLq3JMAOhCKdBE3TcMGMZbKkDdJ5ivNdehOssMCIokNHFOsv7DO4rlEOfyKpg==", - "cpu": [ - "riscv64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ] - }, - "node_modules/@rollup/rollup-linux-s390x-gnu": { - "version": "4.24.0", - "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-s390x-gnu/-/rollup-linux-s390x-gnu-4.24.0.tgz", - "integrity": "sha512-mjBaoo4ocxJppTorZVKWFpy1bfFj9FeCMJqzlMQGjpNPY9JwQi7OuS1axzNIk0nMX6jSgy6ZURDZ2w0QW6D56g==", - "cpu": [ - "s390x" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ] - }, - "node_modules/@rollup/rollup-linux-x64-gnu": { - "version": "4.24.0", - "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-gnu/-/rollup-linux-x64-gnu-4.24.0.tgz", - "integrity": "sha512-ZXFk7M72R0YYFN5q13niV0B7G8/5dcQ9JDp8keJSfr3GoZeXEoMHP/HlvqROA3OMbMdfr19IjCeNAnPUG93b6A==", - "cpu": [ - "x64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ] - }, - "node_modules/@rollup/rollup-linux-x64-musl": { - "version": "4.24.0", - "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-musl/-/rollup-linux-x64-musl-4.24.0.tgz", - "integrity": "sha512-w1i+L7kAXZNdYl+vFvzSZy8Y1arS7vMgIy8wusXJzRrPyof5LAb02KGr1PD2EkRcl73kHulIID0M501lN+vobQ==", - "cpu": [ - "x64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ] - }, - "node_modules/@rollup/rollup-win32-arm64-msvc": { - "version": "4.24.0", - "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-arm64-msvc/-/rollup-win32-arm64-msvc-4.24.0.tgz", - "integrity": "sha512-VXBrnPWgBpVDCVY6XF3LEW0pOU51KbaHhccHw6AS6vBWIC60eqsH19DAeeObl+g8nKAz04QFdl/Cefta0xQtUQ==", - "cpu": [ - "arm64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "win32" - ] - }, - "node_modules/@rollup/rollup-win32-ia32-msvc": { - "version": "4.24.0", - "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-ia32-msvc/-/rollup-win32-ia32-msvc-4.24.0.tgz", - "integrity": "sha512-xrNcGDU0OxVcPTH/8n/ShH4UevZxKIO6HJFK0e15XItZP2UcaiLFd5kiX7hJnqCbSztUF8Qot+JWBC/QXRPYWQ==", - "cpu": [ - "ia32" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "win32" - ] - }, - "node_modules/@rollup/rollup-win32-x64-msvc": { - "version": "4.24.0", - "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-x64-msvc/-/rollup-win32-x64-msvc-4.24.0.tgz", - "integrity": "sha512-fbMkAF7fufku0N2dE5TBXcNlg0pt0cJue4xBRE2Qc5Vqikxr4VCgKj/ht6SMdFcOacVA9rqF70APJ8RN/4vMJw==", - "cpu": [ - "x64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "win32" - ] - }, "node_modules/@selderee/plugin-htmlparser2": { "version": "0.11.0", "resolved": "https://registry.npmjs.org/@selderee/plugin-htmlparser2/-/plugin-htmlparser2-0.11.0.tgz", @@ -1289,19 +661,15 @@ "@supabase/storage-js": "2.7.1" } }, - "node_modules/@types/estree": { - "version": "1.0.6", - "resolved": "https://registry.npmjs.org/@types/estree/-/estree-1.0.6.tgz", - "integrity": "sha512-AYnb1nQyY49te+VRAVgmzfcgjYS91mY5P0TKUDCLEM+gNnA+3T6rWITXRLYCpahpqSQbN5cE+gHpnPyXjHWxcw==", + "node_modules/@trysound/sax": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/@trysound/sax/-/sax-0.2.0.tgz", + "integrity": "sha512-L7z9BgrNEcYyUYtF+HaEfiS5ebkh9jXqbszz7pC0hRBPaatV0XjSD3+eHrpqFemQfgwiFF0QPIarnIihIDn7OA==", "dev": true, - "license": "MIT" - }, - "node_modules/@types/html-minifier-terser": { - "version": "7.0.2", - "resolved": "https://registry.npmjs.org/@types/html-minifier-terser/-/html-minifier-terser-7.0.2.tgz", - "integrity": "sha512-mm2HqV22l8lFQh4r2oSsOEVea+m0qqxEmwpc9kC1p/XzmjLWrReR9D/GRs8Pex2NX/imyEH9c5IU/7tMBQCHOA==", - "dev": true, - "license": "MIT" + "license": "ISC", + "engines": { + "node": ">=10.13.0" + } }, "node_modules/@types/http-proxy": { "version": "1.17.15", @@ -1517,6 +885,44 @@ "dev": true, "license": "MIT" }, + "node_modules/autoprefixer": { + "version": "10.4.20", + "resolved": "https://registry.npmjs.org/autoprefixer/-/autoprefixer-10.4.20.tgz", + "integrity": "sha512-XY25y5xSv/wEoqzDyXXME4AFfkZI0P23z6Fs3YgymDnKJkCGOnkL0iTxCa85UTqaSgfcqyf3UA6+c7wUvx/16g==", + "dev": true, + "funding": [ + { + "type": "opencollective", + "url": "https://opencollective.com/postcss/" + }, + { + "type": "tidelift", + "url": "https://tidelift.com/funding/github/npm/autoprefixer" + }, + { + "type": "github", + "url": "https://github.com/sponsors/ai" + } + ], + "license": "MIT", + "dependencies": { + "browserslist": "^4.23.3", + "caniuse-lite": "^1.0.30001646", + "fraction.js": "^4.3.7", + "normalize-range": "^0.1.2", + "picocolors": "^1.0.1", + "postcss-value-parser": "^4.2.0" + }, + "bin": { + "autoprefixer": "bin/autoprefixer" + }, + "engines": { + "node": "^10 || ^12 || >=14" + }, + "peerDependencies": { + "postcss": "^8.1.0" + } + }, "node_modules/balanced-match": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz", @@ -1651,6 +1057,39 @@ "node": ">=8" } }, + "node_modules/browserslist": { + "version": "4.24.0", + "resolved": "https://registry.npmjs.org/browserslist/-/browserslist-4.24.0.tgz", + "integrity": "sha512-Rmb62sR1Zpjql25eSanFGEhAxcFwfA1K0GuQcLoaJBAcENegrQut3hYdhXFF1obQfiDyqIW/cLM5HSJ/9k884A==", + "dev": true, + "funding": [ + { + "type": "opencollective", + "url": "https://opencollective.com/browserslist" + }, + { + "type": "tidelift", + "url": "https://tidelift.com/funding/github/npm/browserslist" + }, + { + "type": "github", + "url": "https://github.com/sponsors/ai" + } + ], + "license": "MIT", + "dependencies": { + "caniuse-lite": "^1.0.30001663", + "electron-to-chromium": "^1.5.28", + "node-releases": "^2.0.18", + "update-browserslist-db": "^1.1.0" + }, + "bin": { + "browserslist": "cli.js" + }, + "engines": { + "node": "^6 || ^7 || ^8 || ^9 || ^10 || ^11 || ^12 || >=13.7" + } + }, "node_modules/buffer-from": { "version": "1.1.2", "resolved": "https://registry.npmjs.org/buffer-from/-/buffer-from-1.1.2.tgz", @@ -1699,6 +1138,40 @@ "tslib": "^2.0.3" } }, + "node_modules/caniuse-api": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/caniuse-api/-/caniuse-api-3.0.0.tgz", + "integrity": "sha512-bsTwuIg/BZZK/vreVTYYbSWoe2F+71P7K5QGEX+pT250DZbfU1MQ5prOKpPR+LL6uWKK3KMwMCAS74QB3Um1uw==", + "dev": true, + "license": "MIT", + "dependencies": { + "browserslist": "^4.0.0", + "caniuse-lite": "^1.0.0", + "lodash.memoize": "^4.1.2", + "lodash.uniq": "^4.5.0" + } + }, + "node_modules/caniuse-lite": { + "version": "1.0.30001669", + "resolved": "https://registry.npmjs.org/caniuse-lite/-/caniuse-lite-1.0.30001669.tgz", + "integrity": "sha512-DlWzFDJqstqtIVx1zeSpIMLjunf5SmwOw0N2Ck/QSQdS8PLS4+9HrLaYei4w8BIAL7IB/UEDu889d8vhCTPA0w==", + "dev": true, + "funding": [ + { + "type": "opencollective", + "url": "https://opencollective.com/browserslist" + }, + { + "type": "tidelift", + "url": "https://tidelift.com/funding/github/npm/caniuse-lite" + }, + { + "type": "github", + "url": "https://github.com/sponsors/ai" + } + ], + "license": "CC-BY-4.0" + }, "node_modules/chardet": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/chardet/-/chardet-2.0.0.tgz", @@ -1804,6 +1277,13 @@ "dev": true, "license": "MIT" }, + "node_modules/colord": { + "version": "2.9.3", + "resolved": "https://registry.npmjs.org/colord/-/colord-2.9.3.tgz", + "integrity": "sha512-jeC1axXpnb0/2nn/Y1LPuLdgXBLH7aDcHu4KEKfqw3CUhX7ZpfBSlPKyqXE6btIgEzfWtrX3/tyBCaCvXvMkOw==", + "dev": true, + "license": "MIT" + }, "node_modules/commander": { "version": "10.0.1", "resolved": "https://registry.npmjs.org/commander/-/commander-10.0.1.tgz", @@ -1876,6 +1356,19 @@ "node": ">= 8" } }, + "node_modules/css-declaration-sorter": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/css-declaration-sorter/-/css-declaration-sorter-7.2.0.tgz", + "integrity": "sha512-h70rUM+3PNFuaBDTLe8wF/cdWu+dOZmb7pJt8Z2sedYbAcQVQV/tEchueg3GWxwqS0cxtbxmaHEdkNACqcvsow==", + "dev": true, + "license": "ISC", + "engines": { + "node": "^14 || ^16 || >=18" + }, + "peerDependencies": { + "postcss": "^8.0.9" + } + }, "node_modules/css-select": { "version": "5.1.0", "resolved": "https://registry.npmjs.org/css-select/-/css-select-5.1.0.tgz", @@ -1893,6 +1386,20 @@ "url": "https://github.com/sponsors/fb55" } }, + "node_modules/css-tree": { + "version": "2.3.1", + "resolved": "https://registry.npmjs.org/css-tree/-/css-tree-2.3.1.tgz", + "integrity": "sha512-6Fv1DV/TYw//QF5IzQdqsNDjx/wc8TrMBZsqjL9eW01tWb7R7k/mq+/VXfJCl7SoD5emsJop9cOByJZfs8hYIw==", + "dev": true, + "license": "MIT", + "dependencies": { + "mdn-data": "2.0.30", + "source-map-js": "^1.0.1" + }, + "engines": { + "node": "^10 || ^12.20.0 || ^14.13.0 || >=15.0.0" + } + }, "node_modules/css-what": { "version": "6.1.0", "resolved": "https://registry.npmjs.org/css-what/-/css-what-6.1.0.tgz", @@ -1906,6 +1413,134 @@ "url": "https://github.com/sponsors/fb55" } }, + "node_modules/cssesc": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/cssesc/-/cssesc-3.0.0.tgz", + "integrity": "sha512-/Tb/JcjK111nNScGob5MNtsntNM1aCNUDipB/TkwZFhyDrrE47SOx/18wF2bbjgc3ZzCSKW1T5nt5EbFoAz/Vg==", + "dev": true, + "license": "MIT", + "bin": { + "cssesc": "bin/cssesc" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/cssnano": { + "version": "7.0.6", + "resolved": "https://registry.npmjs.org/cssnano/-/cssnano-7.0.6.tgz", + "integrity": "sha512-54woqx8SCbp8HwvNZYn68ZFAepuouZW4lTwiMVnBErM3VkO7/Sd4oTOt3Zz3bPx3kxQ36aISppyXj2Md4lg8bw==", + "dev": true, + "license": "MIT", + "dependencies": { + "cssnano-preset-default": "^7.0.6", + "lilconfig": "^3.1.2" + }, + "engines": { + "node": "^18.12.0 || ^20.9.0 || >=22.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/cssnano" + }, + "peerDependencies": { + "postcss": "^8.4.31" + } + }, + "node_modules/cssnano-preset-default": { + "version": "7.0.6", + "resolved": "https://registry.npmjs.org/cssnano-preset-default/-/cssnano-preset-default-7.0.6.tgz", + "integrity": "sha512-ZzrgYupYxEvdGGuqL+JKOY70s7+saoNlHSCK/OGn1vB2pQK8KSET8jvenzItcY+kA7NoWvfbb/YhlzuzNKjOhQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "browserslist": "^4.23.3", + "css-declaration-sorter": "^7.2.0", + "cssnano-utils": "^5.0.0", + "postcss-calc": "^10.0.2", + "postcss-colormin": "^7.0.2", + "postcss-convert-values": "^7.0.4", + "postcss-discard-comments": "^7.0.3", + "postcss-discard-duplicates": "^7.0.1", + "postcss-discard-empty": "^7.0.0", + "postcss-discard-overridden": "^7.0.0", + "postcss-merge-longhand": "^7.0.4", + "postcss-merge-rules": "^7.0.4", + "postcss-minify-font-values": "^7.0.0", + "postcss-minify-gradients": "^7.0.0", + "postcss-minify-params": "^7.0.2", + "postcss-minify-selectors": "^7.0.4", + "postcss-normalize-charset": "^7.0.0", + "postcss-normalize-display-values": "^7.0.0", + "postcss-normalize-positions": "^7.0.0", + "postcss-normalize-repeat-style": "^7.0.0", + "postcss-normalize-string": "^7.0.0", + "postcss-normalize-timing-functions": "^7.0.0", + "postcss-normalize-unicode": "^7.0.2", + "postcss-normalize-url": "^7.0.0", + "postcss-normalize-whitespace": "^7.0.0", + "postcss-ordered-values": "^7.0.1", + "postcss-reduce-initial": "^7.0.2", + "postcss-reduce-transforms": "^7.0.0", + "postcss-svgo": "^7.0.1", + "postcss-unique-selectors": "^7.0.3" + }, + "engines": { + "node": "^18.12.0 || ^20.9.0 || >=22.0" + }, + "peerDependencies": { + "postcss": "^8.4.31" + } + }, + "node_modules/cssnano-utils": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/cssnano-utils/-/cssnano-utils-5.0.0.tgz", + "integrity": "sha512-Uij0Xdxc24L6SirFr25MlwC2rCFX6scyUmuKpzI+JQ7cyqDEwD42fJ0xfB3yLfOnRDU5LKGgjQ9FA6LYh76GWQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": "^18.12.0 || ^20.9.0 || >=22.0" + }, + "peerDependencies": { + "postcss": "^8.4.31" + } + }, + "node_modules/csso": { + "version": "5.0.5", + "resolved": "https://registry.npmjs.org/csso/-/csso-5.0.5.tgz", + "integrity": "sha512-0LrrStPOdJj+SPCCrGhzryycLjwcgUSHBtxNA8aIDxf0GLsRh1cKYhB00Gd1lDOS4yGH69+SNn13+TWbVHETFQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "css-tree": "~2.2.0" + }, + "engines": { + "node": "^10 || ^12.20.0 || ^14.13.0 || >=15.0.0", + "npm": ">=7.0.0" + } + }, + "node_modules/csso/node_modules/css-tree": { + "version": "2.2.1", + "resolved": "https://registry.npmjs.org/css-tree/-/css-tree-2.2.1.tgz", + "integrity": "sha512-OA0mILzGc1kCOCSJerOeqDxDQ4HOh+G8NbOJFOTgOCzpw7fCBubk0fEyxp8AgOL/jvLgYA/uV0cMbe43ElF1JA==", + "dev": true, + "license": "MIT", + "dependencies": { + "mdn-data": "2.0.28", + "source-map-js": "^1.0.1" + }, + "engines": { + "node": "^10 || ^12.20.0 || ^14.13.0 || >=15.0.0", + "npm": ">=7.0.0" + } + }, + "node_modules/csso/node_modules/mdn-data": { + "version": "2.0.28", + "resolved": "https://registry.npmjs.org/mdn-data/-/mdn-data-2.0.28.tgz", + "integrity": "sha512-aylIc7Z9y4yzHYAJNuESG3hfhC+0Ibp/MAMiaOZgNv4pmEdFyfZhhhny4MNiAfWdBQ1RQ2mfDWmM1x8SvGyp8g==", + "dev": true, + "license": "CC0-1.0" + }, "node_modules/cssom": { "version": "0.5.0", "resolved": "https://registry.npmjs.org/cssom/-/cssom-0.5.0.tgz", @@ -2142,6 +1777,13 @@ "dev": true, "license": "MIT" }, + "node_modules/electron-to-chromium": { + "version": "1.5.41", + "resolved": "https://registry.npmjs.org/electron-to-chromium/-/electron-to-chromium-1.5.41.tgz", + "integrity": "sha512-dfdv/2xNjX0P8Vzme4cfzHqnPm5xsZXwsolTYr0eyW18IUmNyG08vL+fttvinTfhKfIKdRoqkDIC9e9iWQCNYQ==", + "dev": true, + "license": "ISC" + }, "node_modules/emoji-regex": { "version": "9.2.2", "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-9.2.2.tgz", @@ -2221,43 +1863,14 @@ "node": ">= 0.4" } }, - "node_modules/esbuild": { - "version": "0.21.5", - "resolved": "https://registry.npmjs.org/esbuild/-/esbuild-0.21.5.tgz", - "integrity": "sha512-mg3OPMV4hXywwpoDxu3Qda5xCKQi+vCTZq8S9J/EpkhB2HzKXq4SNFZE3+NK93JYxc8VMSep+lOUSC/RVKaBqw==", + "node_modules/escalade": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/escalade/-/escalade-3.2.0.tgz", + "integrity": "sha512-WUj2qlxaQtO4g6Pq5c29GTcWGDyd8itL8zTlipgECz3JesAiiOKotd8JU6otB3PACgG6xkJUyVhboMS+bje/jA==", "dev": true, - "hasInstallScript": true, "license": "MIT", - "bin": { - "esbuild": "bin/esbuild" - }, "engines": { - "node": ">=12" - }, - "optionalDependencies": { - "@esbuild/aix-ppc64": "0.21.5", - "@esbuild/android-arm": "0.21.5", - "@esbuild/android-arm64": "0.21.5", - "@esbuild/android-x64": "0.21.5", - "@esbuild/darwin-arm64": "0.21.5", - "@esbuild/darwin-x64": "0.21.5", - "@esbuild/freebsd-arm64": "0.21.5", - "@esbuild/freebsd-x64": "0.21.5", - "@esbuild/linux-arm": "0.21.5", - "@esbuild/linux-arm64": "0.21.5", - "@esbuild/linux-ia32": "0.21.5", - "@esbuild/linux-loong64": "0.21.5", - "@esbuild/linux-mips64el": "0.21.5", - "@esbuild/linux-ppc64": "0.21.5", - "@esbuild/linux-riscv64": "0.21.5", - "@esbuild/linux-s390x": "0.21.5", - "@esbuild/linux-x64": "0.21.5", - "@esbuild/netbsd-x64": "0.21.5", - "@esbuild/openbsd-x64": "0.21.5", - "@esbuild/sunos-x64": "0.21.5", - "@esbuild/win32-arm64": "0.21.5", - "@esbuild/win32-ia32": "0.21.5", - "@esbuild/win32-x64": "0.21.5" + "node": ">=6" } }, "node_modules/escape-html": { @@ -2468,6 +2081,13 @@ "node": ">=8.6.0" } }, + "node_modules/fast-sort": { + "version": "3.4.1", + "resolved": "https://registry.npmjs.org/fast-sort/-/fast-sort-3.4.1.tgz", + "integrity": "sha512-76uvGPsF6So53sZAqenP9UVT3p5l7cyTHkLWVCMinh41Y8NDrK1IYXJgaBMfc1gk7nJiSRZp676kddFG2Aa5+A==", + "dev": true, + "license": "MIT" + }, "node_modules/fast-xml-parser": { "version": "4.5.0", "resolved": "https://registry.npmjs.org/fast-xml-parser/-/fast-xml-parser-4.5.0.tgz", @@ -2606,6 +2226,20 @@ "node": ">= 0.6" } }, + "node_modules/fraction.js": { + "version": "4.3.7", + "resolved": "https://registry.npmjs.org/fraction.js/-/fraction.js-4.3.7.tgz", + "integrity": "sha512-ZsDfxO51wGAXREY55a7la9LScWpwv9RxIrYABrlvOFBlH/ShPnrtsXeuUIfXKKOVicNxQ+o8JTbJvjS4M89yew==", + "dev": true, + "license": "MIT", + "engines": { + "node": "*" + }, + "funding": { + "type": "patreon", + "url": "https://github.com/sponsors/rawify" + } + }, "node_modules/fresh": { "version": "0.5.2", "resolved": "https://registry.npmjs.org/fresh/-/fresh-0.5.2.tgz", @@ -3067,6 +2701,22 @@ "node": ">=8" } }, + "node_modules/is-core-module": { + "version": "2.15.1", + "resolved": "https://registry.npmjs.org/is-core-module/-/is-core-module-2.15.1.tgz", + "integrity": "sha512-z0vtXSwucUJtANQWldhbtbt7BnL0vxiFjIdDLAatwhDYty2bad6s+rijD6Ri4YuYJubLzIJLUidCh09e1djEVQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "hasown": "^2.0.2" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, "node_modules/is-decimal": { "version": "2.0.1", "resolved": "https://registry.npmjs.org/is-decimal/-/is-decimal-2.0.1.tgz", @@ -3230,6 +2880,19 @@ "url": "https://ko-fi.com/killymxi" } }, + "node_modules/lilconfig": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/lilconfig/-/lilconfig-3.1.2.tgz", + "integrity": "sha512-eop+wDAvpItUys0FWkHIKeC9ybYrTGbU41U5K7+bttZZeohvnY7M9dZ5kB21GNWiFT2q1OoPTvncPCgSOVO5ow==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=14" + }, + "funding": { + "url": "https://github.com/sponsors/antonk52" + } + }, "node_modules/linkedom": { "version": "0.18.5", "resolved": "https://registry.npmjs.org/linkedom/-/linkedom-0.18.5.tgz", @@ -3315,6 +2978,20 @@ "dev": true, "license": "MIT" }, + "node_modules/lodash.memoize": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/lodash.memoize/-/lodash.memoize-4.1.2.tgz", + "integrity": "sha512-t7j+NzmgnQzTAYXcsHYLgimltOV1MXHtlOWf6GjL9Kj8GK5FInw5JotxvbOs+IvV1/Dzo04/fCGfLVs7aXb4Ag==", + "dev": true, + "license": "MIT" + }, + "node_modules/lodash.uniq": { + "version": "4.5.0", + "resolved": "https://registry.npmjs.org/lodash.uniq/-/lodash.uniq-4.5.0.tgz", + "integrity": "sha512-xfBaXQd9ryd9dlSDvnvI0lvxfLJlYAZzXomUYzLKtUeOQvOP5piqAWuGtrhWeqaXK9hhoM/iyJc5AV+XfsX3HQ==", + "dev": true, + "license": "MIT" + }, "node_modules/lower-case": { "version": "2.0.2", "resolved": "https://registry.npmjs.org/lower-case/-/lower-case-2.0.2.tgz", @@ -3423,6 +3100,13 @@ "node": ">=0.10.0" } }, + "node_modules/mdn-data": { + "version": "2.0.30", + "resolved": "https://registry.npmjs.org/mdn-data/-/mdn-data-2.0.30.tgz", + "integrity": "sha512-GaqWWShW4kv/G9IEucWScBx9G1/vsFZZJUO+tD26M8J8z3Kw5RDQjaoZe03YAClgeS/SWPOcb4nkFBTEi5DUEA==", + "dev": true, + "license": "CC0-1.0" + }, "node_modules/mdurl": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/mdurl/-/mdurl-2.0.0.tgz", @@ -3634,6 +3318,13 @@ "tslib": "^2.0.3" } }, + "node_modules/node-releases": { + "version": "2.0.18", + "resolved": "https://registry.npmjs.org/node-releases/-/node-releases-2.0.18.tgz", + "integrity": "sha512-d9VeXT4SJ7ZeOqGX6R5EM022wpL+eWPooLI+5UpWn2jCT1aosUQEhQP214x33Wkwx3JQMvIm+tIoVOdodFS40g==", + "dev": true, + "license": "MIT" + }, "node_modules/node-retrieve-globals": { "version": "6.0.0", "resolved": "https://registry.npmjs.org/node-retrieve-globals/-/node-retrieve-globals-6.0.0.tgz", @@ -3656,6 +3347,16 @@ "node": ">=0.10.0" } }, + "node_modules/normalize-range": { + "version": "0.1.2", + "resolved": "https://registry.npmjs.org/normalize-range/-/normalize-range-0.1.2.tgz", + "integrity": "sha512-bdok/XvKII3nUpklnV6P2hxtMNrCboOjAcyBuQnWEhO665FwrSNRxU+AqpsyvO6LgGYPspN+lu5CLtw4jPRKNA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.10.0" + } + }, "node_modules/nth-check": { "version": "2.1.1", "resolved": "https://registry.npmjs.org/nth-check/-/nth-check-2.1.1.tgz", @@ -3866,6 +3567,13 @@ "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==", + "dev": true, + "license": "MIT" + }, "node_modules/path-scurry": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/path-scurry/-/path-scurry-2.0.0.tgz", @@ -3968,6 +3676,498 @@ "node": "^10 || ^12 || >=14" } }, + "node_modules/postcss-calc": { + "version": "10.0.2", + "resolved": "https://registry.npmjs.org/postcss-calc/-/postcss-calc-10.0.2.tgz", + "integrity": "sha512-DT/Wwm6fCKgpYVI7ZEWuPJ4az8hiEHtCUeYjZXqU7Ou4QqYh1Df2yCQ7Ca6N7xqKPFkxN3fhf+u9KSoOCJNAjg==", + "dev": true, + "license": "MIT", + "dependencies": { + "postcss-selector-parser": "^6.1.2", + "postcss-value-parser": "^4.2.0" + }, + "engines": { + "node": "^18.12 || ^20.9 || >=22.0" + }, + "peerDependencies": { + "postcss": "^8.4.38" + } + }, + "node_modules/postcss-colormin": { + "version": "7.0.2", + "resolved": "https://registry.npmjs.org/postcss-colormin/-/postcss-colormin-7.0.2.tgz", + "integrity": "sha512-YntRXNngcvEvDbEjTdRWGU606eZvB5prmHG4BF0yLmVpamXbpsRJzevyy6MZVyuecgzI2AWAlvFi8DAeCqwpvA==", + "dev": true, + "license": "MIT", + "dependencies": { + "browserslist": "^4.23.3", + "caniuse-api": "^3.0.0", + "colord": "^2.9.3", + "postcss-value-parser": "^4.2.0" + }, + "engines": { + "node": "^18.12.0 || ^20.9.0 || >=22.0" + }, + "peerDependencies": { + "postcss": "^8.4.31" + } + }, + "node_modules/postcss-convert-values": { + "version": "7.0.4", + "resolved": "https://registry.npmjs.org/postcss-convert-values/-/postcss-convert-values-7.0.4.tgz", + "integrity": "sha512-e2LSXPqEHVW6aoGbjV9RsSSNDO3A0rZLCBxN24zvxF25WknMPpX8Dm9UxxThyEbaytzggRuZxaGXqaOhxQ514Q==", + "dev": true, + "license": "MIT", + "dependencies": { + "browserslist": "^4.23.3", + "postcss-value-parser": "^4.2.0" + }, + "engines": { + "node": "^18.12.0 || ^20.9.0 || >=22.0" + }, + "peerDependencies": { + "postcss": "^8.4.31" + } + }, + "node_modules/postcss-discard-comments": { + "version": "7.0.3", + "resolved": "https://registry.npmjs.org/postcss-discard-comments/-/postcss-discard-comments-7.0.3.tgz", + "integrity": "sha512-q6fjd4WU4afNhWOA2WltHgCbkRhZPgQe7cXF74fuVB/ge4QbM9HEaOIzGSiMvM+g/cOsNAUGdf2JDzqA2F8iLA==", + "dev": true, + "license": "MIT", + "dependencies": { + "postcss-selector-parser": "^6.1.2" + }, + "engines": { + "node": "^18.12.0 || ^20.9.0 || >=22.0" + }, + "peerDependencies": { + "postcss": "^8.4.31" + } + }, + "node_modules/postcss-discard-duplicates": { + "version": "7.0.1", + "resolved": "https://registry.npmjs.org/postcss-discard-duplicates/-/postcss-discard-duplicates-7.0.1.tgz", + "integrity": "sha512-oZA+v8Jkpu1ct/xbbrntHRsfLGuzoP+cpt0nJe5ED2FQF8n8bJtn7Bo28jSmBYwqgqnqkuSXJfSUEE7if4nClQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": "^18.12.0 || ^20.9.0 || >=22.0" + }, + "peerDependencies": { + "postcss": "^8.4.31" + } + }, + "node_modules/postcss-discard-empty": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/postcss-discard-empty/-/postcss-discard-empty-7.0.0.tgz", + "integrity": "sha512-e+QzoReTZ8IAwhnSdp/++7gBZ/F+nBq9y6PomfwORfP7q9nBpK5AMP64kOt0bA+lShBFbBDcgpJ3X4etHg4lzA==", + "dev": true, + "license": "MIT", + "engines": { + "node": "^18.12.0 || ^20.9.0 || >=22.0" + }, + "peerDependencies": { + "postcss": "^8.4.31" + } + }, + "node_modules/postcss-discard-overridden": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/postcss-discard-overridden/-/postcss-discard-overridden-7.0.0.tgz", + "integrity": "sha512-GmNAzx88u3k2+sBTZrJSDauR0ccpE24omTQCVmaTTZFz1du6AasspjaUPMJ2ud4RslZpoFKyf+6MSPETLojc6w==", + "dev": true, + "license": "MIT", + "engines": { + "node": "^18.12.0 || ^20.9.0 || >=22.0" + }, + "peerDependencies": { + "postcss": "^8.4.31" + } + }, + "node_modules/postcss-import": { + "version": "16.1.0", + "resolved": "https://registry.npmjs.org/postcss-import/-/postcss-import-16.1.0.tgz", + "integrity": "sha512-7hsAZ4xGXl4MW+OKEWCnF6T5jqBw80/EE9aXg1r2yyn1RsVEU8EtKXbijEODa+rg7iih4bKf7vlvTGYR4CnPNg==", + "dev": true, + "license": "MIT", + "dependencies": { + "postcss-value-parser": "^4.0.0", + "read-cache": "^1.0.0", + "resolve": "^1.1.7" + }, + "engines": { + "node": ">=18.0.0" + }, + "peerDependencies": { + "postcss": "^8.0.0" + } + }, + "node_modules/postcss-import-ext-glob": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/postcss-import-ext-glob/-/postcss-import-ext-glob-2.1.1.tgz", + "integrity": "sha512-qd4ELOx2G0hyjgtmLnf/fSVJXXPhkcxcxhLT1y1mAnk53JYbMLoGg+AFtnJowOSvnv4CvjPAzpLpAcfWeofP5g==", + "dev": true, + "license": "MIT", + "dependencies": { + "fast-glob": "^3.2.12", + "fast-sort": "^3.2.0", + "postcss-value-parser": "^4.2.0" + }, + "peerDependencies": { + "postcss": "^8.2.0" + } + }, + "node_modules/postcss-merge-longhand": { + "version": "7.0.4", + "resolved": "https://registry.npmjs.org/postcss-merge-longhand/-/postcss-merge-longhand-7.0.4.tgz", + "integrity": "sha512-zer1KoZA54Q8RVHKOY5vMke0cCdNxMP3KBfDerjH/BYHh4nCIh+1Yy0t1pAEQF18ac/4z3OFclO+ZVH8azjR4A==", + "dev": true, + "license": "MIT", + "dependencies": { + "postcss-value-parser": "^4.2.0", + "stylehacks": "^7.0.4" + }, + "engines": { + "node": "^18.12.0 || ^20.9.0 || >=22.0" + }, + "peerDependencies": { + "postcss": "^8.4.31" + } + }, + "node_modules/postcss-merge-rules": { + "version": "7.0.4", + "resolved": "https://registry.npmjs.org/postcss-merge-rules/-/postcss-merge-rules-7.0.4.tgz", + "integrity": "sha512-ZsaamiMVu7uBYsIdGtKJ64PkcQt6Pcpep/uO90EpLS3dxJi6OXamIobTYcImyXGoW0Wpugh7DSD3XzxZS9JCPg==", + "dev": true, + "license": "MIT", + "dependencies": { + "browserslist": "^4.23.3", + "caniuse-api": "^3.0.0", + "cssnano-utils": "^5.0.0", + "postcss-selector-parser": "^6.1.2" + }, + "engines": { + "node": "^18.12.0 || ^20.9.0 || >=22.0" + }, + "peerDependencies": { + "postcss": "^8.4.31" + } + }, + "node_modules/postcss-minify-font-values": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/postcss-minify-font-values/-/postcss-minify-font-values-7.0.0.tgz", + "integrity": "sha512-2ckkZtgT0zG8SMc5aoNwtm5234eUx1GGFJKf2b1bSp8UflqaeFzR50lid4PfqVI9NtGqJ2J4Y7fwvnP/u1cQog==", + "dev": true, + "license": "MIT", + "dependencies": { + "postcss-value-parser": "^4.2.0" + }, + "engines": { + "node": "^18.12.0 || ^20.9.0 || >=22.0" + }, + "peerDependencies": { + "postcss": "^8.4.31" + } + }, + "node_modules/postcss-minify-gradients": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/postcss-minify-gradients/-/postcss-minify-gradients-7.0.0.tgz", + "integrity": "sha512-pdUIIdj/C93ryCHew0UgBnL2DtUS3hfFa5XtERrs4x+hmpMYGhbzo6l/Ir5de41O0GaKVpK1ZbDNXSY6GkXvtg==", + "dev": true, + "license": "MIT", + "dependencies": { + "colord": "^2.9.3", + "cssnano-utils": "^5.0.0", + "postcss-value-parser": "^4.2.0" + }, + "engines": { + "node": "^18.12.0 || ^20.9.0 || >=22.0" + }, + "peerDependencies": { + "postcss": "^8.4.31" + } + }, + "node_modules/postcss-minify-params": { + "version": "7.0.2", + "resolved": "https://registry.npmjs.org/postcss-minify-params/-/postcss-minify-params-7.0.2.tgz", + "integrity": "sha512-nyqVLu4MFl9df32zTsdcLqCFfE/z2+f8GE1KHPxWOAmegSo6lpV2GNy5XQvrzwbLmiU7d+fYay4cwto1oNdAaQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "browserslist": "^4.23.3", + "cssnano-utils": "^5.0.0", + "postcss-value-parser": "^4.2.0" + }, + "engines": { + "node": "^18.12.0 || ^20.9.0 || >=22.0" + }, + "peerDependencies": { + "postcss": "^8.4.31" + } + }, + "node_modules/postcss-minify-selectors": { + "version": "7.0.4", + "resolved": "https://registry.npmjs.org/postcss-minify-selectors/-/postcss-minify-selectors-7.0.4.tgz", + "integrity": "sha512-JG55VADcNb4xFCf75hXkzc1rNeURhlo7ugf6JjiiKRfMsKlDzN9CXHZDyiG6x/zGchpjQS+UAgb1d4nqXqOpmA==", + "dev": true, + "license": "MIT", + "dependencies": { + "cssesc": "^3.0.0", + "postcss-selector-parser": "^6.1.2" + }, + "engines": { + "node": "^18.12.0 || ^20.9.0 || >=22.0" + }, + "peerDependencies": { + "postcss": "^8.4.31" + } + }, + "node_modules/postcss-normalize-charset": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/postcss-normalize-charset/-/postcss-normalize-charset-7.0.0.tgz", + "integrity": "sha512-ABisNUXMeZeDNzCQxPxBCkXexvBrUHV+p7/BXOY+ulxkcjUZO0cp8ekGBwvIh2LbCwnWbyMPNJVtBSdyhM2zYQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": "^18.12.0 || ^20.9.0 || >=22.0" + }, + "peerDependencies": { + "postcss": "^8.4.31" + } + }, + "node_modules/postcss-normalize-display-values": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/postcss-normalize-display-values/-/postcss-normalize-display-values-7.0.0.tgz", + "integrity": "sha512-lnFZzNPeDf5uGMPYgGOw7v0BfB45+irSRz9gHQStdkkhiM0gTfvWkWB5BMxpn0OqgOQuZG/mRlZyJxp0EImr2Q==", + "dev": true, + "license": "MIT", + "dependencies": { + "postcss-value-parser": "^4.2.0" + }, + "engines": { + "node": "^18.12.0 || ^20.9.0 || >=22.0" + }, + "peerDependencies": { + "postcss": "^8.4.31" + } + }, + "node_modules/postcss-normalize-positions": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/postcss-normalize-positions/-/postcss-normalize-positions-7.0.0.tgz", + "integrity": "sha512-I0yt8wX529UKIGs2y/9Ybs2CelSvItfmvg/DBIjTnoUSrPxSV7Z0yZ8ShSVtKNaV/wAY+m7bgtyVQLhB00A1NQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "postcss-value-parser": "^4.2.0" + }, + "engines": { + "node": "^18.12.0 || ^20.9.0 || >=22.0" + }, + "peerDependencies": { + "postcss": "^8.4.31" + } + }, + "node_modules/postcss-normalize-repeat-style": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/postcss-normalize-repeat-style/-/postcss-normalize-repeat-style-7.0.0.tgz", + "integrity": "sha512-o3uSGYH+2q30ieM3ppu9GTjSXIzOrRdCUn8UOMGNw7Af61bmurHTWI87hRybrP6xDHvOe5WlAj3XzN6vEO8jLw==", + "dev": true, + "license": "MIT", + "dependencies": { + "postcss-value-parser": "^4.2.0" + }, + "engines": { + "node": "^18.12.0 || ^20.9.0 || >=22.0" + }, + "peerDependencies": { + "postcss": "^8.4.31" + } + }, + "node_modules/postcss-normalize-string": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/postcss-normalize-string/-/postcss-normalize-string-7.0.0.tgz", + "integrity": "sha512-w/qzL212DFVOpMy3UGyxrND+Kb0fvCiBBujiaONIihq7VvtC7bswjWgKQU/w4VcRyDD8gpfqUiBQ4DUOwEJ6Qg==", + "dev": true, + "license": "MIT", + "dependencies": { + "postcss-value-parser": "^4.2.0" + }, + "engines": { + "node": "^18.12.0 || ^20.9.0 || >=22.0" + }, + "peerDependencies": { + "postcss": "^8.4.31" + } + }, + "node_modules/postcss-normalize-timing-functions": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/postcss-normalize-timing-functions/-/postcss-normalize-timing-functions-7.0.0.tgz", + "integrity": "sha512-tNgw3YV0LYoRwg43N3lTe3AEWZ66W7Dh7lVEpJbHoKOuHc1sLrzMLMFjP8SNULHaykzsonUEDbKedv8C+7ej6g==", + "dev": true, + "license": "MIT", + "dependencies": { + "postcss-value-parser": "^4.2.0" + }, + "engines": { + "node": "^18.12.0 || ^20.9.0 || >=22.0" + }, + "peerDependencies": { + "postcss": "^8.4.31" + } + }, + "node_modules/postcss-normalize-unicode": { + "version": "7.0.2", + "resolved": "https://registry.npmjs.org/postcss-normalize-unicode/-/postcss-normalize-unicode-7.0.2.tgz", + "integrity": "sha512-ztisabK5C/+ZWBdYC+Y9JCkp3M9qBv/XFvDtSw0d/XwfT3UaKeW/YTm/MD/QrPNxuecia46vkfEhewjwcYFjkg==", + "dev": true, + "license": "MIT", + "dependencies": { + "browserslist": "^4.23.3", + "postcss-value-parser": "^4.2.0" + }, + "engines": { + "node": "^18.12.0 || ^20.9.0 || >=22.0" + }, + "peerDependencies": { + "postcss": "^8.4.31" + } + }, + "node_modules/postcss-normalize-url": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/postcss-normalize-url/-/postcss-normalize-url-7.0.0.tgz", + "integrity": "sha512-+d7+PpE+jyPX1hDQZYG+NaFD+Nd2ris6r8fPTBAjE8z/U41n/bib3vze8x7rKs5H1uEw5ppe9IojewouHk0klQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "postcss-value-parser": "^4.2.0" + }, + "engines": { + "node": "^18.12.0 || ^20.9.0 || >=22.0" + }, + "peerDependencies": { + "postcss": "^8.4.31" + } + }, + "node_modules/postcss-normalize-whitespace": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/postcss-normalize-whitespace/-/postcss-normalize-whitespace-7.0.0.tgz", + "integrity": "sha512-37/toN4wwZErqohedXYqWgvcHUGlT8O/m2jVkAfAe9Bd4MzRqlBmXrJRePH0e9Wgnz2X7KymTgTOaaFizQe3AQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "postcss-value-parser": "^4.2.0" + }, + "engines": { + "node": "^18.12.0 || ^20.9.0 || >=22.0" + }, + "peerDependencies": { + "postcss": "^8.4.31" + } + }, + "node_modules/postcss-ordered-values": { + "version": "7.0.1", + "resolved": "https://registry.npmjs.org/postcss-ordered-values/-/postcss-ordered-values-7.0.1.tgz", + "integrity": "sha512-irWScWRL6nRzYmBOXReIKch75RRhNS86UPUAxXdmW/l0FcAsg0lvAXQCby/1lymxn/o0gVa6Rv/0f03eJOwHxw==", + "dev": true, + "license": "MIT", + "dependencies": { + "cssnano-utils": "^5.0.0", + "postcss-value-parser": "^4.2.0" + }, + "engines": { + "node": "^18.12.0 || ^20.9.0 || >=22.0" + }, + "peerDependencies": { + "postcss": "^8.4.31" + } + }, + "node_modules/postcss-reduce-initial": { + "version": "7.0.2", + "resolved": "https://registry.npmjs.org/postcss-reduce-initial/-/postcss-reduce-initial-7.0.2.tgz", + "integrity": "sha512-pOnu9zqQww7dEKf62Nuju6JgsW2V0KRNBHxeKohU+JkHd/GAH5uvoObqFLqkeB2n20mr6yrlWDvo5UBU5GnkfA==", + "dev": true, + "license": "MIT", + "dependencies": { + "browserslist": "^4.23.3", + "caniuse-api": "^3.0.0" + }, + "engines": { + "node": "^18.12.0 || ^20.9.0 || >=22.0" + }, + "peerDependencies": { + "postcss": "^8.4.31" + } + }, + "node_modules/postcss-reduce-transforms": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/postcss-reduce-transforms/-/postcss-reduce-transforms-7.0.0.tgz", + "integrity": "sha512-pnt1HKKZ07/idH8cpATX/ujMbtOGhUfE+m8gbqwJE05aTaNw8gbo34a2e3if0xc0dlu75sUOiqvwCGY3fzOHew==", + "dev": true, + "license": "MIT", + "dependencies": { + "postcss-value-parser": "^4.2.0" + }, + "engines": { + "node": "^18.12.0 || ^20.9.0 || >=22.0" + }, + "peerDependencies": { + "postcss": "^8.4.31" + } + }, + "node_modules/postcss-selector-parser": { + "version": "6.1.2", + "resolved": "https://registry.npmjs.org/postcss-selector-parser/-/postcss-selector-parser-6.1.2.tgz", + "integrity": "sha512-Q8qQfPiZ+THO/3ZrOrO0cJJKfpYCagtMUkXbnEfmgUjwXg6z/WBeOyS9APBBPCTSiDV+s4SwQGu8yFsiMRIudg==", + "dev": true, + "license": "MIT", + "dependencies": { + "cssesc": "^3.0.0", + "util-deprecate": "^1.0.2" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/postcss-svgo": { + "version": "7.0.1", + "resolved": "https://registry.npmjs.org/postcss-svgo/-/postcss-svgo-7.0.1.tgz", + "integrity": "sha512-0WBUlSL4lhD9rA5k1e5D8EN5wCEyZD6HJk0jIvRxl+FDVOMlJ7DePHYWGGVc5QRqrJ3/06FTXM0bxjmJpmTPSA==", + "dev": true, + "license": "MIT", + "dependencies": { + "postcss-value-parser": "^4.2.0", + "svgo": "^3.3.2" + }, + "engines": { + "node": "^18.12.0 || ^20.9.0 || >= 18" + }, + "peerDependencies": { + "postcss": "^8.4.31" + } + }, + "node_modules/postcss-unique-selectors": { + "version": "7.0.3", + "resolved": "https://registry.npmjs.org/postcss-unique-selectors/-/postcss-unique-selectors-7.0.3.tgz", + "integrity": "sha512-J+58u5Ic5T1QjP/LDV9g3Cx4CNOgB5vz+kM6+OxHHhFACdcDeKhBXjQmB7fnIZM12YSTvsL0Opwco83DmacW2g==", + "dev": true, + "license": "MIT", + "dependencies": { + "postcss-selector-parser": "^6.1.2" + }, + "engines": { + "node": "^18.12.0 || ^20.9.0 || >=22.0" + }, + "peerDependencies": { + "postcss": "^8.4.31" + } + }, + "node_modules/postcss-value-parser": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/postcss-value-parser/-/postcss-value-parser-4.2.0.tgz", + "integrity": "sha512-1NNCs6uurfkVbeXG4S8JFT9t19m45ICnif8zWLd5oPSZ50QnwMfK+H3jv408d4jw/7Bttv5axS5IiHoLaVNHeQ==", + "dev": true, + "license": "MIT" + }, "node_modules/posthtml": { "version": "0.16.6", "resolved": "https://registry.npmjs.org/posthtml/-/posthtml-0.16.6.tgz", @@ -4231,6 +4431,16 @@ "node": ">= 0.8" } }, + "node_modules/read-cache": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/read-cache/-/read-cache-1.0.0.tgz", + "integrity": "sha512-Owdv/Ft7IjOgm/i0xvNDZ1LrRANRfew4b2prF3OWMQLxLfu3bS8FVhCsrSCMK4lR56Y9ya+AThoTpDCTxCmpRA==", + "dev": true, + "license": "MIT", + "dependencies": { + "pify": "^2.3.0" + } + }, "node_modules/readdirp": { "version": "3.6.0", "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-3.6.0.tgz", @@ -4260,6 +4470,24 @@ "integrity": "sha512-KigOCHcocU3XODJxsu8i/j8T9tzT4adHiecwORRQ0ZZFcp7ahwXuRU1m+yuO90C5ZUyGeGfocHDI14M3L3yDAQ==", "license": "MIT" }, + "node_modules/resolve": { + "version": "1.22.8", + "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.22.8.tgz", + "integrity": "sha512-oKWePCxqpd6FlLvGV1VU0x7bkPmmCNolxzjMf4NczoDnQcIWrAF+cPtZn5i6n+RfD2d9i0tzpKnG6Yk168yIyw==", + "dev": true, + "license": "MIT", + "dependencies": { + "is-core-module": "^2.13.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/reusify": { "version": "1.0.4", "resolved": "https://registry.npmjs.org/reusify/-/reusify-1.0.4.tgz", @@ -4291,42 +4519,6 @@ "url": "https://github.com/sponsors/isaacs" } }, - "node_modules/rollup": { - "version": "4.24.0", - "resolved": "https://registry.npmjs.org/rollup/-/rollup-4.24.0.tgz", - "integrity": "sha512-DOmrlGSXNk1DM0ljiQA+i+o0rSLhtii1je5wgk60j49d1jHT5YYttBv1iWOnYSTG+fZZESUOSNiAl89SIet+Cg==", - "dev": true, - "license": "MIT", - "dependencies": { - "@types/estree": "1.0.6" - }, - "bin": { - "rollup": "dist/bin/rollup" - }, - "engines": { - "node": ">=18.0.0", - "npm": ">=8.0.0" - }, - "optionalDependencies": { - "@rollup/rollup-android-arm-eabi": "4.24.0", - "@rollup/rollup-android-arm64": "4.24.0", - "@rollup/rollup-darwin-arm64": "4.24.0", - "@rollup/rollup-darwin-x64": "4.24.0", - "@rollup/rollup-linux-arm-gnueabihf": "4.24.0", - "@rollup/rollup-linux-arm-musleabihf": "4.24.0", - "@rollup/rollup-linux-arm64-gnu": "4.24.0", - "@rollup/rollup-linux-arm64-musl": "4.24.0", - "@rollup/rollup-linux-powerpc64le-gnu": "4.24.0", - "@rollup/rollup-linux-riscv64-gnu": "4.24.0", - "@rollup/rollup-linux-s390x-gnu": "4.24.0", - "@rollup/rollup-linux-x64-gnu": "4.24.0", - "@rollup/rollup-linux-x64-musl": "4.24.0", - "@rollup/rollup-win32-arm64-msvc": "4.24.0", - "@rollup/rollup-win32-ia32-msvc": "4.24.0", - "@rollup/rollup-win32-x64-msvc": "4.24.0", - "fsevents": "~2.3.2" - } - }, "node_modules/run-parallel": { "version": "1.2.0", "resolved": "https://registry.npmjs.org/run-parallel/-/run-parallel-1.2.0.tgz", @@ -4851,6 +5043,72 @@ "dev": true, "license": "MIT" }, + "node_modules/stylehacks": { + "version": "7.0.4", + "resolved": "https://registry.npmjs.org/stylehacks/-/stylehacks-7.0.4.tgz", + "integrity": "sha512-i4zfNrGMt9SB4xRK9L83rlsFCgdGANfeDAYacO1pkqcE7cRHPdWHwnKZVz7WY17Veq/FvyYsRAU++Ga+qDFIww==", + "dev": true, + "license": "MIT", + "dependencies": { + "browserslist": "^4.23.3", + "postcss-selector-parser": "^6.1.2" + }, + "engines": { + "node": "^18.12.0 || ^20.9.0 || >=22.0" + }, + "peerDependencies": { + "postcss": "^8.4.31" + } + }, + "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==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/svgo": { + "version": "3.3.2", + "resolved": "https://registry.npmjs.org/svgo/-/svgo-3.3.2.tgz", + "integrity": "sha512-OoohrmuUlBs8B8o6MB2Aevn+pRIH9zDALSR+6hhqVfa6fRwG/Qw9VUMSMW9VNg2CFc/MTIfabtdOVl9ODIJjpw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@trysound/sax": "0.2.0", + "commander": "^7.2.0", + "css-select": "^5.1.0", + "css-tree": "^2.3.1", + "css-what": "^6.1.0", + "csso": "^5.0.5", + "picocolors": "^1.0.0" + }, + "bin": { + "svgo": "bin/svgo" + }, + "engines": { + "node": ">=14.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/svgo" + } + }, + "node_modules/svgo/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==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">= 10" + } + }, "node_modules/terser": { "version": "5.36.0", "resolved": "https://registry.npmjs.org/terser/-/terser-5.36.0.tgz", @@ -4994,6 +5252,37 @@ "node": ">= 0.8" } }, + "node_modules/update-browserslist-db": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/update-browserslist-db/-/update-browserslist-db-1.1.1.tgz", + "integrity": "sha512-R8UzCaa9Az+38REPiJ1tXlImTJXlVfgHZsglwBD/k6nj76ctsH1E3q4doGrukiLQd3sGQYu56r5+lo5r94l29A==", + "dev": true, + "funding": [ + { + "type": "opencollective", + "url": "https://opencollective.com/browserslist" + }, + { + "type": "tidelift", + "url": "https://tidelift.com/funding/github/npm/browserslist" + }, + { + "type": "github", + "url": "https://github.com/sponsors/ai" + } + ], + "license": "MIT", + "dependencies": { + "escalade": "^3.2.0", + "picocolors": "^1.1.0" + }, + "bin": { + "update-browserslist-db": "cli.js" + }, + "peerDependencies": { + "browserslist": ">= 4.21.0" + } + }, "node_modules/urlpattern-polyfill": { "version": "10.0.0", "resolved": "https://registry.npmjs.org/urlpattern-polyfill/-/urlpattern-polyfill-10.0.0.tgz", @@ -5001,6 +5290,13 @@ "dev": true, "license": "MIT" }, + "node_modules/util-deprecate": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.2.tgz", + "integrity": "sha512-EPD5q1uXyFxJpCrLnCc1nHnq3gOa6DZBocAIiI2TaSCA7VCJ1UJDMagCzIkXNsUYfD1daK//LTEQ8xiIbrHtcw==", + "dev": true, + "license": "MIT" + }, "node_modules/utils-merge": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/utils-merge/-/utils-merge-1.0.1.tgz", @@ -5021,80 +5317,6 @@ "node": ">= 0.8" } }, - "node_modules/vite": { - "version": "5.4.9", - "resolved": "https://registry.npmjs.org/vite/-/vite-5.4.9.tgz", - "integrity": "sha512-20OVpJHh0PAM0oSOELa5GaZNWeDjcAvQjGXy2Uyr+Tp+/D2/Hdz6NLgpJLsarPTA2QJ6v8mX2P1ZfbsSKvdMkg==", - "dev": true, - "license": "MIT", - "dependencies": { - "esbuild": "^0.21.3", - "postcss": "^8.4.43", - "rollup": "^4.20.0" - }, - "bin": { - "vite": "bin/vite.js" - }, - "engines": { - "node": "^18.0.0 || >=20.0.0" - }, - "funding": { - "url": "https://github.com/vitejs/vite?sponsor=1" - }, - "optionalDependencies": { - "fsevents": "~2.3.3" - }, - "peerDependencies": { - "@types/node": "^18.0.0 || >=20.0.0", - "less": "*", - "lightningcss": "^1.21.0", - "sass": "*", - "sass-embedded": "*", - "stylus": "*", - "sugarss": "*", - "terser": "^5.4.0" - }, - "peerDependenciesMeta": { - "@types/node": { - "optional": true - }, - "less": { - "optional": true - }, - "lightningcss": { - "optional": true - }, - "sass": { - "optional": true - }, - "sass-embedded": { - "optional": true - }, - "stylus": { - "optional": true - }, - "sugarss": { - "optional": true - }, - "terser": { - "optional": true - } - } - }, - "node_modules/vite-plugin-minify": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/vite-plugin-minify/-/vite-plugin-minify-2.0.0.tgz", - "integrity": "sha512-xQWdXCip/CH3c5a0fftJtvpodOIZqp3gwfuSpGtik/W1YmZKe8WMTJrxvrjgrQ1NcP4EuqmiMCUaz8+If1CPMw==", - "dev": true, - "license": "MIT", - "dependencies": { - "@types/html-minifier-terser": "^7.0.2", - "html-minifier-terser": "^7.2.0" - }, - "peerDependencies": { - "vite": "^5.4.0" - } - }, "node_modules/webidl-conversions": { "version": "3.0.1", "resolved": "https://registry.npmjs.org/webidl-conversions/-/webidl-conversions-3.0.1.tgz", diff --git a/package.json b/package.json index 877d7b6e..a511b29f 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "coryd.dev", - "version": "1.4.0", + "version": "1.3.0", "description": "The source for my personal site. Built using 11ty (and other tools).", "type": "module", "engines": { @@ -38,12 +38,14 @@ "devDependencies": { "@11ty/eleventy": "v3.0.0", "@11ty/eleventy-plugin-syntaxhighlight": "^5.0.0", - "@11ty/eleventy-plugin-vite": "5.0.0", "@cdransf/eleventy-plugin-tabler-icons": "^2.0.3", "@supabase/supabase-js": "^2.45.5", + "autoprefixer": "^10.4.20", + "cssnano": "^7.0.6", "dotenv-flow": "^4.1.0", "express": "4.21.1", "fast-xml-parser": "^4.5.0", + "html-minifier-terser": "^7.2.0", "html-to-text": "^9.0.5", "ics": "^3.8.1", "linkedom": "0.18.5", @@ -52,9 +54,12 @@ "markdown-it-anchor": "^9.2.0", "markdown-it-footnote": "^4.0.0", "markdown-it-prism": "^2.3.0", + "postcss": "^8.4.47", + "postcss-import": "^16.1.0", + "postcss-import-ext-glob": "^2.1.1", "rimraf": "^6.0.1", "slugify": "^1.6.6", - "truncate-html": "^1.1.2", - "vite-plugin-minify": "2.0.0" + "terser": "^5.36.0", + "truncate-html": "^1.1.2" } } diff --git a/scripts/install-components.mjs b/scripts/install-components.mjs deleted file mode 100644 index 1c886da3..00000000 --- a/scripts/install-components.mjs +++ /dev/null @@ -1,31 +0,0 @@ -import fs from 'fs' -import path from 'path' -import { fileURLToPath } from 'url' - -const __dirname = path.dirname(fileURLToPath(import.meta.url)) - -const components = [ - { src: '@cdransf/api-text/api-text.js', dest: 'api-text.js' }, - { src: '@cdransf/select-pagination/select-pagination.js', dest: 'select-pagination.js' }, - { src: '@daviddarnes/mastodon-post/mastodon-post.js', dest: 'mastodon-post.js' }, - { src: 'minisearch/dist/es/index.js', dest: 'mini-search.js' }, - { src: '@cdransf/theme-toggle/theme-toggle.js', dest: 'theme-toggle.js' }, - { src: 'youtube-video-element/youtube-video-element.js', dest: 'youtube-video-element.js' } -] - -const destDir = path.resolve(__dirname, '../src/assets/js/components') - -if (!fs.existsSync(destDir)) { - fs.mkdirSync(destDir, { recursive: true }) - console.log(`Created directory: ${destDir}`) -} - -components.forEach(({ src, dest }) => { - const srcPath = path.resolve(__dirname, '../node_modules', src) - const destPath = path.join(destDir, dest) - - fs.copyFile(srcPath, destPath, err => { - if (err) console.error(`Failed to copy ${src}:`, err) - else console.log(`Copied ${src} to ${destPath}`) - }) -}) \ No newline at end of file diff --git a/src/assets/js/components/api-text.js b/src/assets/js/components/api-text.js deleted file mode 100644 index 60f5f025..00000000 --- a/src/assets/js/components/api-text.js +++ /dev/null @@ -1,69 +0,0 @@ -class ApiText extends HTMLElement { - static tagName = 'api-text' - - static register(tagName = this.tagName, registry = globalThis.customElements) { - registry.define(tagName, this) - } - - static attr = { - url: 'api-url', - } - - get url() { - return this.getAttribute(ApiText.attr.url) || '' - } - - async connectedCallback() { - if (this.shadowRoot) return - - this.attachShadow({ mode: 'open' }).appendChild(document.createElement('slot')) - - const loading = this.querySelector('.loading') - const content = this.querySelector('.content') - const cacheKey = this.url || 'api-text-cache' - const cache = sessionStorage?.getItem(cacheKey) - const noscriptContent = this.querySelector('noscript')?.innerHTML.trim() || '' - - const loadText = (string) => { - if (!string) { - if (noscriptContent) { - content.innerHTML = noscriptContent - loading.style.display = 'none' - content.style.display = 'block' - } else { - this.style.display = 'none' - } - return - } - loading.style.display = 'none' - content.style.display = 'block' - content.innerHTML = string - } - - if (cache) { - loadText(JSON.parse(cache)) - } else { - loading.style.display = 'block' - content.style.display = 'none' - } - - try { - const data = await this.data - const value = data.content - if (value) { - loadText(value) - sessionStorage?.setItem(cacheKey, JSON.stringify(value)) - } else { - loadText('') - } - } catch (error) { - loadText('') - } - } - - get data() { - return fetch(this.url).then(response => response.json()).catch(() => ({})) - } -} - -ApiText.register() \ No newline at end of file diff --git a/src/assets/js/components/mastodon-post.js b/src/assets/js/components/mastodon-post.js deleted file mode 100644 index c9f3a00f..00000000 --- a/src/assets/js/components/mastodon-post.js +++ /dev/null @@ -1,115 +0,0 @@ -const mastodonPostTemplate = document.createElement("template"); - -mastodonPostTemplate.innerHTML = ` -
-
-
- - @ - -
-
Reposts
-
Replies
-
Favourites
-
-
-
-`; - -mastodonPostTemplate.id = "mastodon-post-template"; - -if (!document.getElementById(mastodonPostTemplate.id)) { - document.body.appendChild(mastodonPostTemplate); -} - -class MastodonPost extends HTMLElement { - static register(tagName) { - if ("customElements" in window) { - customElements.define(tagName || "mastodon-post", MastodonPost); - } - } - - async connectedCallback() { - this.append(this.template); - - const data = { ...(await this.data), ...this.linkData }; - - this.slots.forEach((slot) => { - slot.dataset.key.split(",").forEach((keyItem) => { - const value = this.getValue(keyItem, data); - if (keyItem === "content") { - slot.innerHTML = value; - } else { - this.populateSlot(slot, value); - } - }); - }); - } - - populateSlot(slot, value) { - if (typeof value == "string" && value.startsWith("http")) { - if (slot.localName === "img") slot.src = value; - if (slot.localName === "a") slot.href = value; - } else { - slot.textContent = value; - } - } - - handleKey(object, key) { - const parsedKeyInt = parseFloat(key); - - if (Number.isNaN(parsedKeyInt)) { - return object[key]; - } - - return object[parsedKeyInt]; - } - - getValue(string, data) { - let keys = string.trim().split(/\.|\[|\]/g); - keys = keys.filter((string) => string.length); - - const value = keys.reduce( - (object, key) => this.handleKey(object, key), - data - ); - return value; - } - - get template() { - return document - .getElementById( - this.getAttribute("template") || `${this.localName}-template` - ) - .content.cloneNode(true); - } - - get slots() { - return this.querySelectorAll("[data-key]"); - } - - get link() { - return this.querySelector("a").href; - } - - get linkData() { - const url = new URL(this.link); - const paths = url.pathname.split("/").filter((string) => string.length); - return { - url: this.link, - hostname: url.hostname, - username: paths.find((path) => path.startsWith("@")), - postId: paths.find((path) => !path.startsWith("@")) - }; - } - - get endpoint() { - return `https://${this.linkData.hostname}/api/v1/statuses/${this.linkData.postId}`; - } - - get data() { - return fetch(this.endpoint).then((response) => response.json()); - } -} - -MastodonPost.register(); diff --git a/src/assets/js/components/mini-search.js b/src/assets/js/components/mini-search.js deleted file mode 100644 index 73536919..00000000 --- a/src/assets/js/components/mini-search.js +++ /dev/null @@ -1,2036 +0,0 @@ -/****************************************************************************** -Copyright (c) Microsoft Corporation. - -Permission to use, copy, modify, and/or distribute this software for any -purpose with or without fee is hereby granted. - -THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH -REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY -AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, -INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM -LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR -OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR -PERFORMANCE OF THIS SOFTWARE. -***************************************************************************** */ -/* global Reflect, Promise, SuppressedError, Symbol */ - - -function __awaiter(thisArg, _arguments, P, generator) { - function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); } - return new (P || (P = Promise))(function (resolve, reject) { - function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } } - function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } } - function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); } - step((generator = generator.apply(thisArg, _arguments || [])).next()); - }); -} - -typeof SuppressedError === "function" ? SuppressedError : function (error, suppressed, message) { - var e = new Error(message); - return e.name = "SuppressedError", e.error = error, e.suppressed = suppressed, e; -}; - -/** @ignore */ -const ENTRIES = 'ENTRIES'; -/** @ignore */ -const KEYS = 'KEYS'; -/** @ignore */ -const VALUES = 'VALUES'; -/** @ignore */ -const LEAF = ''; -/** - * @private - */ -class TreeIterator { - constructor(set, type) { - const node = set._tree; - const keys = Array.from(node.keys()); - this.set = set; - this._type = type; - this._path = keys.length > 0 ? [{ node, keys }] : []; - } - next() { - const value = this.dive(); - this.backtrack(); - return value; - } - dive() { - if (this._path.length === 0) { - return { done: true, value: undefined }; - } - const { node, keys } = last$1(this._path); - if (last$1(keys) === LEAF) { - return { done: false, value: this.result() }; - } - const child = node.get(last$1(keys)); - this._path.push({ node: child, keys: Array.from(child.keys()) }); - return this.dive(); - } - backtrack() { - if (this._path.length === 0) { - return; - } - const keys = last$1(this._path).keys; - keys.pop(); - if (keys.length > 0) { - return; - } - this._path.pop(); - this.backtrack(); - } - key() { - return this.set._prefix + this._path - .map(({ keys }) => last$1(keys)) - .filter(key => key !== LEAF) - .join(''); - } - value() { - return last$1(this._path).node.get(LEAF); - } - result() { - switch (this._type) { - case VALUES: return this.value(); - case KEYS: return this.key(); - default: return [this.key(), this.value()]; - } - } - [Symbol.iterator]() { - return this; - } -} -const last$1 = (array) => { - return array[array.length - 1]; -}; - -/* eslint-disable no-labels */ -/** - * @ignore - */ -const fuzzySearch = (node, query, maxDistance) => { - const results = new Map(); - if (query === undefined) - return results; - // Number of columns in the Levenshtein matrix. - const n = query.length + 1; - // Matching terms can never be longer than N + maxDistance. - const m = n + maxDistance; - // Fill first matrix row and column with numbers: 0 1 2 3 ... - const matrix = new Uint8Array(m * n).fill(maxDistance + 1); - for (let j = 0; j < n; ++j) - matrix[j] = j; - for (let i = 1; i < m; ++i) - matrix[i * n] = i; - recurse(node, query, maxDistance, results, matrix, 1, n, ''); - return results; -}; -// Modified version of http://stevehanov.ca/blog/?id=114 -// This builds a Levenshtein matrix for a given query and continuously updates -// it for nodes in the radix tree that fall within the given maximum edit -// distance. Keeping the same matrix around is beneficial especially for larger -// edit distances. -// -// k a t e <-- query -// 0 1 2 3 4 -// c 1 1 2 3 4 -// a 2 2 1 2 3 -// t 3 3 2 1 [2] <-- edit distance -// ^ -// ^ term in radix tree, rows are added and removed as needed -const recurse = (node, query, maxDistance, results, matrix, m, n, prefix) => { - const offset = m * n; - key: for (const key of node.keys()) { - if (key === LEAF) { - // We've reached a leaf node. Check if the edit distance acceptable and - // store the result if it is. - const distance = matrix[offset - 1]; - if (distance <= maxDistance) { - results.set(prefix, [node.get(key), distance]); - } - } - else { - // Iterate over all characters in the key. Update the Levenshtein matrix - // and check if the minimum distance in the last row is still within the - // maximum edit distance. If it is, we can recurse over all child nodes. - let i = m; - for (let pos = 0; pos < key.length; ++pos, ++i) { - const char = key[pos]; - const thisRowOffset = n * i; - const prevRowOffset = thisRowOffset - n; - // Set the first column based on the previous row, and initialize the - // minimum distance in the current row. - let minDistance = matrix[thisRowOffset]; - const jmin = Math.max(0, i - maxDistance - 1); - const jmax = Math.min(n - 1, i + maxDistance); - // Iterate over remaining columns (characters in the query). - for (let j = jmin; j < jmax; ++j) { - const different = char !== query[j]; - // It might make sense to only read the matrix positions used for - // deletion/insertion if the characters are different. But we want to - // avoid conditional reads for performance reasons. - const rpl = matrix[prevRowOffset + j] + +different; - const del = matrix[prevRowOffset + j + 1] + 1; - const ins = matrix[thisRowOffset + j] + 1; - const dist = matrix[thisRowOffset + j + 1] = Math.min(rpl, del, ins); - if (dist < minDistance) - minDistance = dist; - } - // Because distance will never decrease, we can stop. There will be no - // matching child nodes. - if (minDistance > maxDistance) { - continue key; - } - } - recurse(node.get(key), query, maxDistance, results, matrix, i, n, prefix + key); - } - } -}; - -/* eslint-disable no-labels */ -/** - * A class implementing the same interface as a standard JavaScript - * [`Map`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Map) - * with string keys, but adding support for efficiently searching entries with - * prefix or fuzzy search. This class is used internally by {@link MiniSearch} - * as the inverted index data structure. The implementation is a radix tree - * (compressed prefix tree). - * - * Since this class can be of general utility beyond _MiniSearch_, it is - * exported by the `minisearch` package and can be imported (or required) as - * `minisearch/SearchableMap`. - * - * @typeParam T The type of the values stored in the map. - */ -class SearchableMap { - /** - * The constructor is normally called without arguments, creating an empty - * map. In order to create a {@link SearchableMap} from an iterable or from an - * object, check {@link SearchableMap.from} and {@link - * SearchableMap.fromObject}. - * - * The constructor arguments are for internal use, when creating derived - * mutable views of a map at a prefix. - */ - constructor(tree = new Map(), prefix = '') { - this._size = undefined; - this._tree = tree; - this._prefix = prefix; - } - /** - * Creates and returns a mutable view of this {@link SearchableMap}, - * containing only entries that share the given prefix. - * - * ### Usage: - * - * ```javascript - * let map = new SearchableMap() - * map.set("unicorn", 1) - * map.set("universe", 2) - * map.set("university", 3) - * map.set("unique", 4) - * map.set("hello", 5) - * - * let uni = map.atPrefix("uni") - * uni.get("unique") // => 4 - * uni.get("unicorn") // => 1 - * uni.get("hello") // => undefined - * - * let univer = map.atPrefix("univer") - * univer.get("unique") // => undefined - * univer.get("universe") // => 2 - * univer.get("university") // => 3 - * ``` - * - * @param prefix The prefix - * @return A {@link SearchableMap} representing a mutable view of the original - * Map at the given prefix - */ - atPrefix(prefix) { - if (!prefix.startsWith(this._prefix)) { - throw new Error('Mismatched prefix'); - } - const [node, path] = trackDown(this._tree, prefix.slice(this._prefix.length)); - if (node === undefined) { - const [parentNode, key] = last(path); - for (const k of parentNode.keys()) { - if (k !== LEAF && k.startsWith(key)) { - const node = new Map(); - node.set(k.slice(key.length), parentNode.get(k)); - return new SearchableMap(node, prefix); - } - } - } - return new SearchableMap(node, prefix); - } - /** - * @see https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Map/clear - */ - clear() { - this._size = undefined; - this._tree.clear(); - } - /** - * @see https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Map/delete - * @param key Key to delete - */ - delete(key) { - this._size = undefined; - return remove(this._tree, key); - } - /** - * @see https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Map/entries - * @return An iterator iterating through `[key, value]` entries. - */ - entries() { - return new TreeIterator(this, ENTRIES); - } - /** - * @see https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Map/forEach - * @param fn Iteration function - */ - forEach(fn) { - for (const [key, value] of this) { - fn(key, value, this); - } - } - /** - * Returns a Map of all the entries that have a key within the given edit - * distance from the search key. The keys of the returned Map are the matching - * keys, while the values are two-element arrays where the first element is - * the value associated to the key, and the second is the edit distance of the - * key to the search key. - * - * ### Usage: - * - * ```javascript - * let map = new SearchableMap() - * map.set('hello', 'world') - * map.set('hell', 'yeah') - * map.set('ciao', 'mondo') - * - * // Get all entries that match the key 'hallo' with a maximum edit distance of 2 - * map.fuzzyGet('hallo', 2) - * // => Map(2) { 'hello' => ['world', 1], 'hell' => ['yeah', 2] } - * - * // In the example, the "hello" key has value "world" and edit distance of 1 - * // (change "e" to "a"), the key "hell" has value "yeah" and edit distance of 2 - * // (change "e" to "a", delete "o") - * ``` - * - * @param key The search key - * @param maxEditDistance The maximum edit distance (Levenshtein) - * @return A Map of the matching keys to their value and edit distance - */ - fuzzyGet(key, maxEditDistance) { - return fuzzySearch(this._tree, key, maxEditDistance); - } - /** - * @see https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Map/get - * @param key Key to get - * @return Value associated to the key, or `undefined` if the key is not - * found. - */ - get(key) { - const node = lookup(this._tree, key); - return node !== undefined ? node.get(LEAF) : undefined; - } - /** - * @see https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Map/has - * @param key Key - * @return True if the key is in the map, false otherwise - */ - has(key) { - const node = lookup(this._tree, key); - return node !== undefined && node.has(LEAF); - } - /** - * @see https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Map/keys - * @return An `Iterable` iterating through keys - */ - keys() { - return new TreeIterator(this, KEYS); - } - /** - * @see https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Map/set - * @param key Key to set - * @param value Value to associate to the key - * @return The {@link SearchableMap} itself, to allow chaining - */ - set(key, value) { - if (typeof key !== 'string') { - throw new Error('key must be a string'); - } - this._size = undefined; - const node = createPath(this._tree, key); - node.set(LEAF, value); - return this; - } - /** - * @see https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Map/size - */ - get size() { - if (this._size) { - return this._size; - } - /** @ignore */ - this._size = 0; - const iter = this.entries(); - while (!iter.next().done) - this._size += 1; - return this._size; - } - /** - * Updates the value at the given key using the provided function. The function - * is called with the current value at the key, and its return value is used as - * the new value to be set. - * - * ### Example: - * - * ```javascript - * // Increment the current value by one - * searchableMap.update('somekey', (currentValue) => currentValue == null ? 0 : currentValue + 1) - * ``` - * - * If the value at the given key is or will be an object, it might not require - * re-assignment. In that case it is better to use `fetch()`, because it is - * faster. - * - * @param key The key to update - * @param fn The function used to compute the new value from the current one - * @return The {@link SearchableMap} itself, to allow chaining - */ - update(key, fn) { - if (typeof key !== 'string') { - throw new Error('key must be a string'); - } - this._size = undefined; - const node = createPath(this._tree, key); - node.set(LEAF, fn(node.get(LEAF))); - return this; - } - /** - * Fetches the value of the given key. If the value does not exist, calls the - * given function to create a new value, which is inserted at the given key - * and subsequently returned. - * - * ### Example: - * - * ```javascript - * const map = searchableMap.fetch('somekey', () => new Map()) - * map.set('foo', 'bar') - * ``` - * - * @param key The key to update - * @param defaultValue A function that creates a new value if the key does not exist - * @return The existing or new value at the given key - */ - fetch(key, initial) { - if (typeof key !== 'string') { - throw new Error('key must be a string'); - } - this._size = undefined; - const node = createPath(this._tree, key); - let value = node.get(LEAF); - if (value === undefined) { - node.set(LEAF, value = initial()); - } - return value; - } - /** - * @see https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Map/values - * @return An `Iterable` iterating through values. - */ - values() { - return new TreeIterator(this, VALUES); - } - /** - * @see https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Map/@@iterator - */ - [Symbol.iterator]() { - return this.entries(); - } - /** - * Creates a {@link SearchableMap} from an `Iterable` of entries - * - * @param entries Entries to be inserted in the {@link SearchableMap} - * @return A new {@link SearchableMap} with the given entries - */ - static from(entries) { - const tree = new SearchableMap(); - for (const [key, value] of entries) { - tree.set(key, value); - } - return tree; - } - /** - * Creates a {@link SearchableMap} from the iterable properties of a JavaScript object - * - * @param object Object of entries for the {@link SearchableMap} - * @return A new {@link SearchableMap} with the given entries - */ - static fromObject(object) { - return SearchableMap.from(Object.entries(object)); - } -} -const trackDown = (tree, key, path = []) => { - if (key.length === 0 || tree == null) { - return [tree, path]; - } - for (const k of tree.keys()) { - if (k !== LEAF && key.startsWith(k)) { - path.push([tree, k]); // performance: update in place - return trackDown(tree.get(k), key.slice(k.length), path); - } - } - path.push([tree, key]); // performance: update in place - return trackDown(undefined, '', path); -}; -const lookup = (tree, key) => { - if (key.length === 0 || tree == null) { - return tree; - } - for (const k of tree.keys()) { - if (k !== LEAF && key.startsWith(k)) { - return lookup(tree.get(k), key.slice(k.length)); - } - } -}; -// Create a path in the radix tree for the given key, and returns the deepest -// node. This function is in the hot path for indexing. It avoids unnecessary -// string operations and recursion for performance. -const createPath = (node, key) => { - const keyLength = key.length; - outer: for (let pos = 0; node && pos < keyLength;) { - for (const k of node.keys()) { - // Check whether this key is a candidate: the first characters must match. - if (k !== LEAF && key[pos] === k[0]) { - const len = Math.min(keyLength - pos, k.length); - // Advance offset to the point where key and k no longer match. - let offset = 1; - while (offset < len && key[pos + offset] === k[offset]) - ++offset; - const child = node.get(k); - if (offset === k.length) { - // The existing key is shorter than the key we need to create. - node = child; - } - else { - // Partial match: we need to insert an intermediate node to contain - // both the existing subtree and the new node. - const intermediate = new Map(); - intermediate.set(k.slice(offset), child); - node.set(key.slice(pos, pos + offset), intermediate); - node.delete(k); - node = intermediate; - } - pos += offset; - continue outer; - } - } - // Create a final child node to contain the final suffix of the key. - const child = new Map(); - node.set(key.slice(pos), child); - return child; - } - return node; -}; -const remove = (tree, key) => { - const [node, path] = trackDown(tree, key); - if (node === undefined) { - return; - } - node.delete(LEAF); - if (node.size === 0) { - cleanup(path); - } - else if (node.size === 1) { - const [key, value] = node.entries().next().value; - merge(path, key, value); - } -}; -const cleanup = (path) => { - if (path.length === 0) { - return; - } - const [node, key] = last(path); - node.delete(key); - if (node.size === 0) { - cleanup(path.slice(0, -1)); - } - else if (node.size === 1) { - const [key, value] = node.entries().next().value; - if (key !== LEAF) { - merge(path.slice(0, -1), key, value); - } - } -}; -const merge = (path, key, value) => { - if (path.length === 0) { - return; - } - const [node, nodeKey] = last(path); - node.set(nodeKey + key, value); - node.delete(nodeKey); -}; -const last = (array) => { - return array[array.length - 1]; -}; - -const OR = 'or'; -const AND = 'and'; -const AND_NOT = 'and_not'; -/** - * {@link MiniSearch} is the main entrypoint class, implementing a full-text - * search engine in memory. - * - * @typeParam T The type of the documents being indexed. - * - * ### Basic example: - * - * ```javascript - * const documents = [ - * { - * id: 1, - * title: 'Moby Dick', - * text: 'Call me Ishmael. Some years ago...', - * category: 'fiction' - * }, - * { - * id: 2, - * title: 'Zen and the Art of Motorcycle Maintenance', - * text: 'I can see by my watch...', - * category: 'fiction' - * }, - * { - * id: 3, - * title: 'Neuromancer', - * text: 'The sky above the port was...', - * category: 'fiction' - * }, - * { - * id: 4, - * title: 'Zen and the Art of Archery', - * text: 'At first sight it must seem...', - * category: 'non-fiction' - * }, - * // ...and more - * ] - * - * // Create a search engine that indexes the 'title' and 'text' fields for - * // full-text search. Search results will include 'title' and 'category' (plus the - * // id field, that is always stored and returned) - * const miniSearch = new MiniSearch({ - * fields: ['title', 'text'], - * storeFields: ['title', 'category'] - * }) - * - * // Add documents to the index - * miniSearch.addAll(documents) - * - * // Search for documents: - * let results = miniSearch.search('zen art motorcycle') - * // => [ - * // { id: 2, title: 'Zen and the Art of Motorcycle Maintenance', category: 'fiction', score: 2.77258 }, - * // { id: 4, title: 'Zen and the Art of Archery', category: 'non-fiction', score: 1.38629 } - * // ] - * ``` - */ -class MiniSearch { - /** - * @param options Configuration options - * - * ### Examples: - * - * ```javascript - * // Create a search engine that indexes the 'title' and 'text' fields of your - * // documents: - * const miniSearch = new MiniSearch({ fields: ['title', 'text'] }) - * ``` - * - * ### ID Field: - * - * ```javascript - * // Your documents are assumed to include a unique 'id' field, but if you want - * // to use a different field for document identification, you can set the - * // 'idField' option: - * const miniSearch = new MiniSearch({ idField: 'key', fields: ['title', 'text'] }) - * ``` - * - * ### Options and defaults: - * - * ```javascript - * // The full set of options (here with their default value) is: - * const miniSearch = new MiniSearch({ - * // idField: field that uniquely identifies a document - * idField: 'id', - * - * // extractField: function used to get the value of a field in a document. - * // By default, it assumes the document is a flat object with field names as - * // property keys and field values as string property values, but custom logic - * // can be implemented by setting this option to a custom extractor function. - * extractField: (document, fieldName) => document[fieldName], - * - * // tokenize: function used to split fields into individual terms. By - * // default, it is also used to tokenize search queries, unless a specific - * // `tokenize` search option is supplied. When tokenizing an indexed field, - * // the field name is passed as the second argument. - * tokenize: (string, _fieldName) => string.split(SPACE_OR_PUNCTUATION), - * - * // processTerm: function used to process each tokenized term before - * // indexing. It can be used for stemming and normalization. Return a falsy - * // value in order to discard a term. By default, it is also used to process - * // search queries, unless a specific `processTerm` option is supplied as a - * // search option. When processing a term from a indexed field, the field - * // name is passed as the second argument. - * processTerm: (term, _fieldName) => term.toLowerCase(), - * - * // searchOptions: default search options, see the `search` method for - * // details - * searchOptions: undefined, - * - * // fields: document fields to be indexed. Mandatory, but not set by default - * fields: undefined - * - * // storeFields: document fields to be stored and returned as part of the - * // search results. - * storeFields: [] - * }) - * ``` - */ - constructor(options) { - if ((options === null || options === void 0 ? void 0 : options.fields) == null) { - throw new Error('MiniSearch: option "fields" must be provided'); - } - const autoVacuum = (options.autoVacuum == null || options.autoVacuum === true) ? defaultAutoVacuumOptions : options.autoVacuum; - this._options = Object.assign(Object.assign(Object.assign({}, defaultOptions), options), { autoVacuum, searchOptions: Object.assign(Object.assign({}, defaultSearchOptions), (options.searchOptions || {})), autoSuggestOptions: Object.assign(Object.assign({}, defaultAutoSuggestOptions), (options.autoSuggestOptions || {})) }); - this._index = new SearchableMap(); - this._documentCount = 0; - this._documentIds = new Map(); - this._idToShortId = new Map(); - // Fields are defined during initialization, don't change, are few in - // number, rarely need iterating over, and have string keys. Therefore in - // this case an object is a better candidate than a Map to store the mapping - // from field key to ID. - this._fieldIds = {}; - this._fieldLength = new Map(); - this._avgFieldLength = []; - this._nextId = 0; - this._storedFields = new Map(); - this._dirtCount = 0; - this._currentVacuum = null; - this._enqueuedVacuum = null; - this._enqueuedVacuumConditions = defaultVacuumConditions; - this.addFields(this._options.fields); - } - /** - * Adds a document to the index - * - * @param document The document to be indexed - */ - add(document) { - const { extractField, tokenize, processTerm, fields, idField } = this._options; - const id = extractField(document, idField); - if (id == null) { - throw new Error(`MiniSearch: document does not have ID field "${idField}"`); - } - if (this._idToShortId.has(id)) { - throw new Error(`MiniSearch: duplicate ID ${id}`); - } - const shortDocumentId = this.addDocumentId(id); - this.saveStoredFields(shortDocumentId, document); - for (const field of fields) { - const fieldValue = extractField(document, field); - if (fieldValue == null) - continue; - const tokens = tokenize(fieldValue.toString(), field); - const fieldId = this._fieldIds[field]; - const uniqueTerms = new Set(tokens).size; - this.addFieldLength(shortDocumentId, fieldId, this._documentCount - 1, uniqueTerms); - for (const term of tokens) { - const processedTerm = processTerm(term, field); - if (Array.isArray(processedTerm)) { - for (const t of processedTerm) { - this.addTerm(fieldId, shortDocumentId, t); - } - } - else if (processedTerm) { - this.addTerm(fieldId, shortDocumentId, processedTerm); - } - } - } - } - /** - * Adds all the given documents to the index - * - * @param documents An array of documents to be indexed - */ - addAll(documents) { - for (const document of documents) - this.add(document); - } - /** - * Adds all the given documents to the index asynchronously. - * - * Returns a promise that resolves (to `undefined`) when the indexing is done. - * This method is useful when index many documents, to avoid blocking the main - * thread. The indexing is performed asynchronously and in chunks. - * - * @param documents An array of documents to be indexed - * @param options Configuration options - * @return A promise resolving to `undefined` when the indexing is done - */ - addAllAsync(documents, options = {}) { - const { chunkSize = 10 } = options; - const acc = { chunk: [], promise: Promise.resolve() }; - const { chunk, promise } = documents.reduce(({ chunk, promise }, document, i) => { - chunk.push(document); - if ((i + 1) % chunkSize === 0) { - return { - chunk: [], - promise: promise - .then(() => new Promise(resolve => setTimeout(resolve, 0))) - .then(() => this.addAll(chunk)) - }; - } - else { - return { chunk, promise }; - } - }, acc); - return promise.then(() => this.addAll(chunk)); - } - /** - * Removes the given document from the index. - * - * The document to remove must NOT have changed between indexing and removal, - * otherwise the index will be corrupted. - * - * This method requires passing the full document to be removed (not just the - * ID), and immediately removes the document from the inverted index, allowing - * memory to be released. A convenient alternative is {@link - * MiniSearch#discard}, which needs only the document ID, and has the same - * visible effect, but delays cleaning up the index until the next vacuuming. - * - * @param document The document to be removed - */ - remove(document) { - const { tokenize, processTerm, extractField, fields, idField } = this._options; - const id = extractField(document, idField); - if (id == null) { - throw new Error(`MiniSearch: document does not have ID field "${idField}"`); - } - const shortId = this._idToShortId.get(id); - if (shortId == null) { - throw new Error(`MiniSearch: cannot remove document with ID ${id}: it is not in the index`); - } - for (const field of fields) { - const fieldValue = extractField(document, field); - if (fieldValue == null) - continue; - const tokens = tokenize(fieldValue.toString(), field); - const fieldId = this._fieldIds[field]; - const uniqueTerms = new Set(tokens).size; - this.removeFieldLength(shortId, fieldId, this._documentCount, uniqueTerms); - for (const term of tokens) { - const processedTerm = processTerm(term, field); - if (Array.isArray(processedTerm)) { - for (const t of processedTerm) { - this.removeTerm(fieldId, shortId, t); - } - } - else if (processedTerm) { - this.removeTerm(fieldId, shortId, processedTerm); - } - } - } - this._storedFields.delete(shortId); - this._documentIds.delete(shortId); - this._idToShortId.delete(id); - this._fieldLength.delete(shortId); - this._documentCount -= 1; - } - /** - * Removes all the given documents from the index. If called with no arguments, - * it removes _all_ documents from the index. - * - * @param documents The documents to be removed. If this argument is omitted, - * all documents are removed. Note that, for removing all documents, it is - * more efficient to call this method with no arguments than to pass all - * documents. - */ - removeAll(documents) { - if (documents) { - for (const document of documents) - this.remove(document); - } - else if (arguments.length > 0) { - throw new Error('Expected documents to be present. Omit the argument to remove all documents.'); - } - else { - this._index = new SearchableMap(); - this._documentCount = 0; - this._documentIds = new Map(); - this._idToShortId = new Map(); - this._fieldLength = new Map(); - this._avgFieldLength = []; - this._storedFields = new Map(); - this._nextId = 0; - } - } - /** - * Discards the document with the given ID, so it won't appear in search results - * - * It has the same visible effect of {@link MiniSearch.remove} (both cause the - * document to stop appearing in searches), but a different effect on the - * internal data structures: - * - * - {@link MiniSearch#remove} requires passing the full document to be - * removed as argument, and removes it from the inverted index immediately. - * - * - {@link MiniSearch#discard} instead only needs the document ID, and - * works by marking the current version of the document as discarded, so it - * is immediately ignored by searches. This is faster and more convenient - * than {@link MiniSearch#remove}, but the index is not immediately - * modified. To take care of that, vacuuming is performed after a certain - * number of documents are discarded, cleaning up the index and allowing - * memory to be released. - * - * After discarding a document, it is possible to re-add a new version, and - * only the new version will appear in searches. In other words, discarding - * and re-adding a document works exactly like removing and re-adding it. The - * {@link MiniSearch.replace} method can also be used to replace a document - * with a new version. - * - * #### Details about vacuuming - * - * Repetite calls to this method would leave obsolete document references in - * the index, invisible to searches. Two mechanisms take care of cleaning up: - * clean up during search, and vacuuming. - * - * - Upon search, whenever a discarded ID is found (and ignored for the - * results), references to the discarded document are removed from the - * inverted index entries for the search terms. This ensures that subsequent - * searches for the same terms do not need to skip these obsolete references - * again. - * - * - In addition, vacuuming is performed automatically by default (see the - * `autoVacuum` field in {@link Options}) after a certain number of - * documents are discarded. Vacuuming traverses all terms in the index, - * cleaning up all references to discarded documents. Vacuuming can also be - * triggered manually by calling {@link MiniSearch#vacuum}. - * - * @param id The ID of the document to be discarded - */ - discard(id) { - const shortId = this._idToShortId.get(id); - if (shortId == null) { - throw new Error(`MiniSearch: cannot discard document with ID ${id}: it is not in the index`); - } - this._idToShortId.delete(id); - this._documentIds.delete(shortId); - this._storedFields.delete(shortId); - (this._fieldLength.get(shortId) || []).forEach((fieldLength, fieldId) => { - this.removeFieldLength(shortId, fieldId, this._documentCount, fieldLength); - }); - this._fieldLength.delete(shortId); - this._documentCount -= 1; - this._dirtCount += 1; - this.maybeAutoVacuum(); - } - maybeAutoVacuum() { - if (this._options.autoVacuum === false) { - return; - } - const { minDirtFactor, minDirtCount, batchSize, batchWait } = this._options.autoVacuum; - this.conditionalVacuum({ batchSize, batchWait }, { minDirtCount, minDirtFactor }); - } - /** - * Discards the documents with the given IDs, so they won't appear in search - * results - * - * It is equivalent to calling {@link MiniSearch#discard} for all the given - * IDs, but with the optimization of triggering at most one automatic - * vacuuming at the end. - * - * Note: to remove all documents from the index, it is faster and more - * convenient to call {@link MiniSearch.removeAll} with no argument, instead - * of passing all IDs to this method. - */ - discardAll(ids) { - const autoVacuum = this._options.autoVacuum; - try { - this._options.autoVacuum = false; - for (const id of ids) { - this.discard(id); - } - } - finally { - this._options.autoVacuum = autoVacuum; - } - this.maybeAutoVacuum(); - } - /** - * It replaces an existing document with the given updated version - * - * It works by discarding the current version and adding the updated one, so - * it is functionally equivalent to calling {@link MiniSearch#discard} - * followed by {@link MiniSearch#add}. The ID of the updated document should - * be the same as the original one. - * - * Since it uses {@link MiniSearch#discard} internally, this method relies on - * vacuuming to clean up obsolete document references from the index, allowing - * memory to be released (see {@link MiniSearch#discard}). - * - * @param updatedDocument The updated document to replace the old version - * with - */ - replace(updatedDocument) { - const { idField, extractField } = this._options; - const id = extractField(updatedDocument, idField); - this.discard(id); - this.add(updatedDocument); - } - /** - * Triggers a manual vacuuming, cleaning up references to discarded documents - * from the inverted index - * - * Vacuuming is only useful for applications that use the {@link - * MiniSearch#discard} or {@link MiniSearch#replace} methods. - * - * By default, vacuuming is performed automatically when needed (controlled by - * the `autoVacuum` field in {@link Options}), so there is usually no need to - * call this method, unless one wants to make sure to perform vacuuming at a - * specific moment. - * - * Vacuuming traverses all terms in the inverted index in batches, and cleans - * up references to discarded documents from the posting list, allowing memory - * to be released. - * - * The method takes an optional object as argument with the following keys: - * - * - `batchSize`: the size of each batch (1000 by default) - * - * - `batchWait`: the number of milliseconds to wait between batches (10 by - * default) - * - * On large indexes, vacuuming could have a non-negligible cost: batching - * avoids blocking the thread for long, diluting this cost so that it is not - * negatively affecting the application. Nonetheless, this method should only - * be called when necessary, and relying on automatic vacuuming is usually - * better. - * - * It returns a promise that resolves (to undefined) when the clean up is - * completed. If vacuuming is already ongoing at the time this method is - * called, a new one is enqueued immediately after the ongoing one, and a - * corresponding promise is returned. However, no more than one vacuuming is - * enqueued on top of the ongoing one, even if this method is called more - * times (enqueuing multiple ones would be useless). - * - * @param options Configuration options for the batch size and delay. See - * {@link VacuumOptions}. - */ - vacuum(options = {}) { - return this.conditionalVacuum(options); - } - conditionalVacuum(options, conditions) { - // If a vacuum is already ongoing, schedule another as soon as it finishes, - // unless there's already one enqueued. If one was already enqueued, do not - // enqueue another on top, but make sure that the conditions are the - // broadest. - if (this._currentVacuum) { - this._enqueuedVacuumConditions = this._enqueuedVacuumConditions && conditions; - if (this._enqueuedVacuum != null) { - return this._enqueuedVacuum; - } - this._enqueuedVacuum = this._currentVacuum.then(() => { - const conditions = this._enqueuedVacuumConditions; - this._enqueuedVacuumConditions = defaultVacuumConditions; - return this.performVacuuming(options, conditions); - }); - return this._enqueuedVacuum; - } - if (this.vacuumConditionsMet(conditions) === false) { - return Promise.resolve(); - } - this._currentVacuum = this.performVacuuming(options); - return this._currentVacuum; - } - performVacuuming(options, conditions) { - return __awaiter(this, void 0, void 0, function* () { - const initialDirtCount = this._dirtCount; - if (this.vacuumConditionsMet(conditions)) { - const batchSize = options.batchSize || defaultVacuumOptions.batchSize; - const batchWait = options.batchWait || defaultVacuumOptions.batchWait; - let i = 1; - for (const [term, fieldsData] of this._index) { - for (const [fieldId, fieldIndex] of fieldsData) { - for (const [shortId] of fieldIndex) { - if (this._documentIds.has(shortId)) { - continue; - } - if (fieldIndex.size <= 1) { - fieldsData.delete(fieldId); - } - else { - fieldIndex.delete(shortId); - } - } - } - if (this._index.get(term).size === 0) { - this._index.delete(term); - } - if (i % batchSize === 0) { - yield new Promise((resolve) => setTimeout(resolve, batchWait)); - } - i += 1; - } - this._dirtCount -= initialDirtCount; - } - // Make the next lines always async, so they execute after this function returns - yield null; - this._currentVacuum = this._enqueuedVacuum; - this._enqueuedVacuum = null; - }); - } - vacuumConditionsMet(conditions) { - if (conditions == null) { - return true; - } - let { minDirtCount, minDirtFactor } = conditions; - minDirtCount = minDirtCount || defaultAutoVacuumOptions.minDirtCount; - minDirtFactor = minDirtFactor || defaultAutoVacuumOptions.minDirtFactor; - return this.dirtCount >= minDirtCount && this.dirtFactor >= minDirtFactor; - } - /** - * Is `true` if a vacuuming operation is ongoing, `false` otherwise - */ - get isVacuuming() { - return this._currentVacuum != null; - } - /** - * The number of documents discarded since the most recent vacuuming - */ - get dirtCount() { - return this._dirtCount; - } - /** - * A number between 0 and 1 giving an indication about the proportion of - * documents that are discarded, and can therefore be cleaned up by vacuuming. - * A value close to 0 means that the index is relatively clean, while a higher - * value means that the index is relatively dirty, and vacuuming could release - * memory. - */ - get dirtFactor() { - return this._dirtCount / (1 + this._documentCount + this._dirtCount); - } - /** - * Returns `true` if a document with the given ID is present in the index and - * available for search, `false` otherwise - * - * @param id The document ID - */ - has(id) { - return this._idToShortId.has(id); - } - /** - * Returns the stored fields (as configured in the `storeFields` constructor - * option) for the given document ID. Returns `undefined` if the document is - * not present in the index. - * - * @param id The document ID - */ - getStoredFields(id) { - const shortId = this._idToShortId.get(id); - if (shortId == null) { - return undefined; - } - return this._storedFields.get(shortId); - } - /** - * Search for documents matching the given search query. - * - * The result is a list of scored document IDs matching the query, sorted by - * descending score, and each including data about which terms were matched and - * in which fields. - * - * ### Basic usage: - * - * ```javascript - * // Search for "zen art motorcycle" with default options: terms have to match - * // exactly, and individual terms are joined with OR - * miniSearch.search('zen art motorcycle') - * // => [ { id: 2, score: 2.77258, match: { ... } }, { id: 4, score: 1.38629, match: { ... } } ] - * ``` - * - * ### Restrict search to specific fields: - * - * ```javascript - * // Search only in the 'title' field - * miniSearch.search('zen', { fields: ['title'] }) - * ``` - * - * ### Field boosting: - * - * ```javascript - * // Boost a field - * miniSearch.search('zen', { boost: { title: 2 } }) - * ``` - * - * ### Prefix search: - * - * ```javascript - * // Search for "moto" with prefix search (it will match documents - * // containing terms that start with "moto" or "neuro") - * miniSearch.search('moto neuro', { prefix: true }) - * ``` - * - * ### Fuzzy search: - * - * ```javascript - * // Search for "ismael" with fuzzy search (it will match documents containing - * // terms similar to "ismael", with a maximum edit distance of 0.2 term.length - * // (rounded to nearest integer) - * miniSearch.search('ismael', { fuzzy: 0.2 }) - * ``` - * - * ### Combining strategies: - * - * ```javascript - * // Mix of exact match, prefix search, and fuzzy search - * miniSearch.search('ismael mob', { - * prefix: true, - * fuzzy: 0.2 - * }) - * ``` - * - * ### Advanced prefix and fuzzy search: - * - * ```javascript - * // Perform fuzzy and prefix search depending on the search term. Here - * // performing prefix and fuzzy search only on terms longer than 3 characters - * miniSearch.search('ismael mob', { - * prefix: term => term.length > 3 - * fuzzy: term => term.length > 3 ? 0.2 : null - * }) - * ``` - * - * ### Combine with AND: - * - * ```javascript - * // Combine search terms with AND (to match only documents that contain both - * // "motorcycle" and "art") - * miniSearch.search('motorcycle art', { combineWith: 'AND' }) - * ``` - * - * ### Combine with AND_NOT: - * - * There is also an AND_NOT combinator, that finds documents that match the - * first term, but do not match any of the other terms. This combinator is - * rarely useful with simple queries, and is meant to be used with advanced - * query combinations (see later for more details). - * - * ### Filtering results: - * - * ```javascript - * // Filter only results in the 'fiction' category (assuming that 'category' - * // is a stored field) - * miniSearch.search('motorcycle art', { - * filter: (result) => result.category === 'fiction' - * }) - * ``` - * - * ### Wildcard query - * - * Searching for an empty string (assuming the default tokenizer) returns no - * results. Sometimes though, one needs to match all documents, like in a - * "wildcard" search. This is possible by passing the special value - * {@link MiniSearch.wildcard} as the query: - * - * ```javascript - * // Return search results for all documents - * miniSearch.search(MiniSearch.wildcard) - * ``` - * - * Note that search options such as `filter` and `boostDocument` are still - * applied, influencing which results are returned, and their order: - * - * ```javascript - * // Return search results for all documents in the 'fiction' category - * miniSearch.search(MiniSearch.wildcard, { - * filter: (result) => result.category === 'fiction' - * }) - * ``` - * - * ### Advanced combination of queries: - * - * It is possible to combine different subqueries with OR, AND, and AND_NOT, - * and even with different search options, by passing a query expression - * tree object as the first argument, instead of a string. - * - * ```javascript - * // Search for documents that contain "zen" and ("motorcycle" or "archery") - * miniSearch.search({ - * combineWith: 'AND', - * queries: [ - * 'zen', - * { - * combineWith: 'OR', - * queries: ['motorcycle', 'archery'] - * } - * ] - * }) - * - * // Search for documents that contain ("apple" or "pear") but not "juice" and - * // not "tree" - * miniSearch.search({ - * combineWith: 'AND_NOT', - * queries: [ - * { - * combineWith: 'OR', - * queries: ['apple', 'pear'] - * }, - * 'juice', - * 'tree' - * ] - * }) - * ``` - * - * Each node in the expression tree can be either a string, or an object that - * supports all {@link SearchOptions} fields, plus a `queries` array field for - * subqueries. - * - * Note that, while this can become complicated to do by hand for complex or - * deeply nested queries, it provides a formalized expression tree API for - * external libraries that implement a parser for custom query languages. - * - * @param query Search query - * @param options Search options. Each option, if not given, defaults to the corresponding value of `searchOptions` given to the constructor, or to the library default. - */ - search(query, searchOptions = {}) { - const rawResults = this.executeQuery(query, searchOptions); - const results = []; - for (const [docId, { score, terms, match }] of rawResults) { - // terms are the matched query terms, which will be returned to the user - // as queryTerms. The quality is calculated based on them, as opposed to - // the matched terms in the document (which can be different due to - // prefix and fuzzy match) - const quality = terms.length || 1; - const result = { - id: this._documentIds.get(docId), - score: score * quality, - terms: Object.keys(match), - queryTerms: terms, - match - }; - Object.assign(result, this._storedFields.get(docId)); - if (searchOptions.filter == null || searchOptions.filter(result)) { - results.push(result); - } - } - // If it's a wildcard query, and no document boost is applied, skip sorting - // the results, as all results have the same score of 1 - if (query === MiniSearch.wildcard && - searchOptions.boostDocument == null && - this._options.searchOptions.boostDocument == null) { - return results; - } - results.sort(byScore); - return results; - } - /** - * Provide suggestions for the given search query - * - * The result is a list of suggested modified search queries, derived from the - * given search query, each with a relevance score, sorted by descending score. - * - * By default, it uses the same options used for search, except that by - * default it performs prefix search on the last term of the query, and - * combine terms with `'AND'` (requiring all query terms to match). Custom - * options can be passed as a second argument. Defaults can be changed upon - * calling the {@link MiniSearch} constructor, by passing a - * `autoSuggestOptions` option. - * - * ### Basic usage: - * - * ```javascript - * // Get suggestions for 'neuro': - * miniSearch.autoSuggest('neuro') - * // => [ { suggestion: 'neuromancer', terms: [ 'neuromancer' ], score: 0.46240 } ] - * ``` - * - * ### Multiple words: - * - * ```javascript - * // Get suggestions for 'zen ar': - * miniSearch.autoSuggest('zen ar') - * // => [ - * // { suggestion: 'zen archery art', terms: [ 'zen', 'archery', 'art' ], score: 1.73332 }, - * // { suggestion: 'zen art', terms: [ 'zen', 'art' ], score: 1.21313 } - * // ] - * ``` - * - * ### Fuzzy suggestions: - * - * ```javascript - * // Correct spelling mistakes using fuzzy search: - * miniSearch.autoSuggest('neromancer', { fuzzy: 0.2 }) - * // => [ { suggestion: 'neuromancer', terms: [ 'neuromancer' ], score: 1.03998 } ] - * ``` - * - * ### Filtering: - * - * ```javascript - * // Get suggestions for 'zen ar', but only within the 'fiction' category - * // (assuming that 'category' is a stored field): - * miniSearch.autoSuggest('zen ar', { - * filter: (result) => result.category === 'fiction' - * }) - * // => [ - * // { suggestion: 'zen archery art', terms: [ 'zen', 'archery', 'art' ], score: 1.73332 }, - * // { suggestion: 'zen art', terms: [ 'zen', 'art' ], score: 1.21313 } - * // ] - * ``` - * - * @param queryString Query string to be expanded into suggestions - * @param options Search options. The supported options and default values - * are the same as for the {@link MiniSearch#search} method, except that by - * default prefix search is performed on the last term in the query, and terms - * are combined with `'AND'`. - * @return A sorted array of suggestions sorted by relevance score. - */ - autoSuggest(queryString, options = {}) { - options = Object.assign(Object.assign({}, this._options.autoSuggestOptions), options); - const suggestions = new Map(); - for (const { score, terms } of this.search(queryString, options)) { - const phrase = terms.join(' '); - const suggestion = suggestions.get(phrase); - if (suggestion != null) { - suggestion.score += score; - suggestion.count += 1; - } - else { - suggestions.set(phrase, { score, terms, count: 1 }); - } - } - const results = []; - for (const [suggestion, { score, terms, count }] of suggestions) { - results.push({ suggestion, terms, score: score / count }); - } - results.sort(byScore); - return results; - } - /** - * Total number of documents available to search - */ - get documentCount() { - return this._documentCount; - } - /** - * Number of terms in the index - */ - get termCount() { - return this._index.size; - } - /** - * Deserializes a JSON index (serialized with `JSON.stringify(miniSearch)`) - * and instantiates a MiniSearch instance. It should be given the same options - * originally used when serializing the index. - * - * ### Usage: - * - * ```javascript - * // If the index was serialized with: - * let miniSearch = new MiniSearch({ fields: ['title', 'text'] }) - * miniSearch.addAll(documents) - * - * const json = JSON.stringify(miniSearch) - * // It can later be deserialized like this: - * miniSearch = MiniSearch.loadJSON(json, { fields: ['title', 'text'] }) - * ``` - * - * @param json JSON-serialized index - * @param options configuration options, same as the constructor - * @return An instance of MiniSearch deserialized from the given JSON. - */ - static loadJSON(json, options) { - if (options == null) { - throw new Error('MiniSearch: loadJSON should be given the same options used when serializing the index'); - } - return this.loadJS(JSON.parse(json), options); - } - /** - * Async equivalent of {@link MiniSearch.loadJSON} - * - * This function is an alternative to {@link MiniSearch.loadJSON} that returns - * a promise, and loads the index in batches, leaving pauses between them to avoid - * blocking the main thread. It tends to be slower than the synchronous - * version, but does not block the main thread, so it can be a better choice - * when deserializing very large indexes. - * - * @param json JSON-serialized index - * @param options configuration options, same as the constructor - * @return A Promise that will resolve to an instance of MiniSearch deserialized from the given JSON. - */ - static loadJSONAsync(json, options) { - return __awaiter(this, void 0, void 0, function* () { - if (options == null) { - throw new Error('MiniSearch: loadJSON should be given the same options used when serializing the index'); - } - return this.loadJSAsync(JSON.parse(json), options); - }); - } - /** - * Returns the default value of an option. It will throw an error if no option - * with the given name exists. - * - * @param optionName Name of the option - * @return The default value of the given option - * - * ### Usage: - * - * ```javascript - * // Get default tokenizer - * MiniSearch.getDefault('tokenize') - * - * // Get default term processor - * MiniSearch.getDefault('processTerm') - * - * // Unknown options will throw an error - * MiniSearch.getDefault('notExisting') - * // => throws 'MiniSearch: unknown option "notExisting"' - * ``` - */ - static getDefault(optionName) { - if (defaultOptions.hasOwnProperty(optionName)) { - return getOwnProperty(defaultOptions, optionName); - } - else { - throw new Error(`MiniSearch: unknown option "${optionName}"`); - } - } - /** - * @ignore - */ - static loadJS(js, options) { - const { index, documentIds, fieldLength, storedFields, serializationVersion } = js; - const miniSearch = this.instantiateMiniSearch(js, options); - miniSearch._documentIds = objectToNumericMap(documentIds); - miniSearch._fieldLength = objectToNumericMap(fieldLength); - miniSearch._storedFields = objectToNumericMap(storedFields); - for (const [shortId, id] of miniSearch._documentIds) { - miniSearch._idToShortId.set(id, shortId); - } - for (const [term, data] of index) { - const dataMap = new Map(); - for (const fieldId of Object.keys(data)) { - let indexEntry = data[fieldId]; - // Version 1 used to nest the index entry inside a field called ds - if (serializationVersion === 1) { - indexEntry = indexEntry.ds; - } - dataMap.set(parseInt(fieldId, 10), objectToNumericMap(indexEntry)); - } - miniSearch._index.set(term, dataMap); - } - return miniSearch; - } - /** - * @ignore - */ - static loadJSAsync(js, options) { - return __awaiter(this, void 0, void 0, function* () { - const { index, documentIds, fieldLength, storedFields, serializationVersion } = js; - const miniSearch = this.instantiateMiniSearch(js, options); - miniSearch._documentIds = yield objectToNumericMapAsync(documentIds); - miniSearch._fieldLength = yield objectToNumericMapAsync(fieldLength); - miniSearch._storedFields = yield objectToNumericMapAsync(storedFields); - for (const [shortId, id] of miniSearch._documentIds) { - miniSearch._idToShortId.set(id, shortId); - } - let count = 0; - for (const [term, data] of index) { - const dataMap = new Map(); - for (const fieldId of Object.keys(data)) { - let indexEntry = data[fieldId]; - // Version 1 used to nest the index entry inside a field called ds - if (serializationVersion === 1) { - indexEntry = indexEntry.ds; - } - dataMap.set(parseInt(fieldId, 10), yield objectToNumericMapAsync(indexEntry)); - } - if (++count % 1000 === 0) - yield wait(0); - miniSearch._index.set(term, dataMap); - } - return miniSearch; - }); - } - /** - * @ignore - */ - static instantiateMiniSearch(js, options) { - const { documentCount, nextId, fieldIds, averageFieldLength, dirtCount, serializationVersion } = js; - if (serializationVersion !== 1 && serializationVersion !== 2) { - throw new Error('MiniSearch: cannot deserialize an index created with an incompatible version'); - } - const miniSearch = new MiniSearch(options); - miniSearch._documentCount = documentCount; - miniSearch._nextId = nextId; - miniSearch._idToShortId = new Map(); - miniSearch._fieldIds = fieldIds; - miniSearch._avgFieldLength = averageFieldLength; - miniSearch._dirtCount = dirtCount || 0; - miniSearch._index = new SearchableMap(); - return miniSearch; - } - /** - * @ignore - */ - executeQuery(query, searchOptions = {}) { - if (query === MiniSearch.wildcard) { - return this.executeWildcardQuery(searchOptions); - } - if (typeof query !== 'string') { - const options = Object.assign(Object.assign(Object.assign({}, searchOptions), query), { queries: undefined }); - const results = query.queries.map((subquery) => this.executeQuery(subquery, options)); - return this.combineResults(results, options.combineWith); - } - const { tokenize, processTerm, searchOptions: globalSearchOptions } = this._options; - const options = Object.assign(Object.assign({ tokenize, processTerm }, globalSearchOptions), searchOptions); - const { tokenize: searchTokenize, processTerm: searchProcessTerm } = options; - const terms = searchTokenize(query) - .flatMap((term) => searchProcessTerm(term)) - .filter((term) => !!term); - const queries = terms.map(termToQuerySpec(options)); - const results = queries.map(query => this.executeQuerySpec(query, options)); - return this.combineResults(results, options.combineWith); - } - /** - * @ignore - */ - executeQuerySpec(query, searchOptions) { - const options = Object.assign(Object.assign({}, this._options.searchOptions), searchOptions); - const boosts = (options.fields || this._options.fields).reduce((boosts, field) => (Object.assign(Object.assign({}, boosts), { [field]: getOwnProperty(options.boost, field) || 1 })), {}); - const { boostDocument, weights, maxFuzzy, bm25: bm25params } = options; - const { fuzzy: fuzzyWeight, prefix: prefixWeight } = Object.assign(Object.assign({}, defaultSearchOptions.weights), weights); - const data = this._index.get(query.term); - const results = this.termResults(query.term, query.term, 1, query.termBoost, data, boosts, boostDocument, bm25params); - let prefixMatches; - let fuzzyMatches; - if (query.prefix) { - prefixMatches = this._index.atPrefix(query.term); - } - if (query.fuzzy) { - const fuzzy = (query.fuzzy === true) ? 0.2 : query.fuzzy; - const maxDistance = fuzzy < 1 ? Math.min(maxFuzzy, Math.round(query.term.length * fuzzy)) : fuzzy; - if (maxDistance) - fuzzyMatches = this._index.fuzzyGet(query.term, maxDistance); - } - if (prefixMatches) { - for (const [term, data] of prefixMatches) { - const distance = term.length - query.term.length; - if (!distance) { - continue; - } // Skip exact match. - // Delete the term from fuzzy results (if present) if it is also a - // prefix result. This entry will always be scored as a prefix result. - fuzzyMatches === null || fuzzyMatches === void 0 ? void 0 : fuzzyMatches.delete(term); - // Weight gradually approaches 0 as distance goes to infinity, with the - // weight for the hypothetical distance 0 being equal to prefixWeight. - // The rate of change is much lower than that of fuzzy matches to - // account for the fact that prefix matches stay more relevant than - // fuzzy matches for longer distances. - const weight = prefixWeight * term.length / (term.length + 0.3 * distance); - this.termResults(query.term, term, weight, query.termBoost, data, boosts, boostDocument, bm25params, results); - } - } - if (fuzzyMatches) { - for (const term of fuzzyMatches.keys()) { - const [data, distance] = fuzzyMatches.get(term); - if (!distance) { - continue; - } // Skip exact match. - // Weight gradually approaches 0 as distance goes to infinity, with the - // weight for the hypothetical distance 0 being equal to fuzzyWeight. - const weight = fuzzyWeight * term.length / (term.length + distance); - this.termResults(query.term, term, weight, query.termBoost, data, boosts, boostDocument, bm25params, results); - } - } - return results; - } - /** - * @ignore - */ - executeWildcardQuery(searchOptions) { - const results = new Map(); - const options = Object.assign(Object.assign({}, this._options.searchOptions), searchOptions); - for (const [shortId, id] of this._documentIds) { - const score = options.boostDocument ? options.boostDocument(id, '', this._storedFields.get(shortId)) : 1; - results.set(shortId, { - score, - terms: [], - match: {} - }); - } - return results; - } - /** - * @ignore - */ - combineResults(results, combineWith = OR) { - if (results.length === 0) { - return new Map(); - } - const operator = combineWith.toLowerCase(); - const combinator = combinators[operator]; - if (!combinator) { - throw new Error(`Invalid combination operator: ${combineWith}`); - } - return results.reduce(combinator) || new Map(); - } - /** - * Allows serialization of the index to JSON, to possibly store it and later - * deserialize it with {@link MiniSearch.loadJSON}. - * - * Normally one does not directly call this method, but rather call the - * standard JavaScript `JSON.stringify()` passing the {@link MiniSearch} - * instance, and JavaScript will internally call this method. Upon - * deserialization, one must pass to {@link MiniSearch.loadJSON} the same - * options used to create the original instance that was serialized. - * - * ### Usage: - * - * ```javascript - * // Serialize the index: - * let miniSearch = new MiniSearch({ fields: ['title', 'text'] }) - * miniSearch.addAll(documents) - * const json = JSON.stringify(miniSearch) - * - * // Later, to deserialize it: - * miniSearch = MiniSearch.loadJSON(json, { fields: ['title', 'text'] }) - * ``` - * - * @return A plain-object serializable representation of the search index. - */ - toJSON() { - const index = []; - for (const [term, fieldIndex] of this._index) { - const data = {}; - for (const [fieldId, freqs] of fieldIndex) { - data[fieldId] = Object.fromEntries(freqs); - } - index.push([term, data]); - } - return { - documentCount: this._documentCount, - nextId: this._nextId, - documentIds: Object.fromEntries(this._documentIds), - fieldIds: this._fieldIds, - fieldLength: Object.fromEntries(this._fieldLength), - averageFieldLength: this._avgFieldLength, - storedFields: Object.fromEntries(this._storedFields), - dirtCount: this._dirtCount, - index, - serializationVersion: 2 - }; - } - /** - * @ignore - */ - termResults(sourceTerm, derivedTerm, termWeight, termBoost, fieldTermData, fieldBoosts, boostDocumentFn, bm25params, results = new Map()) { - if (fieldTermData == null) - return results; - for (const field of Object.keys(fieldBoosts)) { - const fieldBoost = fieldBoosts[field]; - const fieldId = this._fieldIds[field]; - const fieldTermFreqs = fieldTermData.get(fieldId); - if (fieldTermFreqs == null) - continue; - let matchingFields = fieldTermFreqs.size; - const avgFieldLength = this._avgFieldLength[fieldId]; - for (const docId of fieldTermFreqs.keys()) { - if (!this._documentIds.has(docId)) { - this.removeTerm(fieldId, docId, derivedTerm); - matchingFields -= 1; - continue; - } - const docBoost = boostDocumentFn ? boostDocumentFn(this._documentIds.get(docId), derivedTerm, this._storedFields.get(docId)) : 1; - if (!docBoost) - continue; - const termFreq = fieldTermFreqs.get(docId); - const fieldLength = this._fieldLength.get(docId)[fieldId]; - // NOTE: The total number of fields is set to the number of documents - // `this._documentCount`. It could also make sense to use the number of - // documents where the current field is non-blank as a normalization - // factor. This will make a difference in scoring if the field is rarely - // present. This is currently not supported, and may require further - // analysis to see if it is a valid use case. - const rawScore = calcBM25Score(termFreq, matchingFields, this._documentCount, fieldLength, avgFieldLength, bm25params); - const weightedScore = termWeight * termBoost * fieldBoost * docBoost * rawScore; - const result = results.get(docId); - if (result) { - result.score += weightedScore; - assignUniqueTerm(result.terms, sourceTerm); - const match = getOwnProperty(result.match, derivedTerm); - if (match) { - match.push(field); - } - else { - result.match[derivedTerm] = [field]; - } - } - else { - results.set(docId, { - score: weightedScore, - terms: [sourceTerm], - match: { [derivedTerm]: [field] } - }); - } - } - } - return results; - } - /** - * @ignore - */ - addTerm(fieldId, documentId, term) { - const indexData = this._index.fetch(term, createMap); - let fieldIndex = indexData.get(fieldId); - if (fieldIndex == null) { - fieldIndex = new Map(); - fieldIndex.set(documentId, 1); - indexData.set(fieldId, fieldIndex); - } - else { - const docs = fieldIndex.get(documentId); - fieldIndex.set(documentId, (docs || 0) + 1); - } - } - /** - * @ignore - */ - removeTerm(fieldId, documentId, term) { - if (!this._index.has(term)) { - this.warnDocumentChanged(documentId, fieldId, term); - return; - } - const indexData = this._index.fetch(term, createMap); - const fieldIndex = indexData.get(fieldId); - if (fieldIndex == null || fieldIndex.get(documentId) == null) { - this.warnDocumentChanged(documentId, fieldId, term); - } - else if (fieldIndex.get(documentId) <= 1) { - if (fieldIndex.size <= 1) { - indexData.delete(fieldId); - } - else { - fieldIndex.delete(documentId); - } - } - else { - fieldIndex.set(documentId, fieldIndex.get(documentId) - 1); - } - if (this._index.get(term).size === 0) { - this._index.delete(term); - } - } - /** - * @ignore - */ - warnDocumentChanged(shortDocumentId, fieldId, term) { - for (const fieldName of Object.keys(this._fieldIds)) { - if (this._fieldIds[fieldName] === fieldId) { - this._options.logger('warn', `MiniSearch: document with ID ${this._documentIds.get(shortDocumentId)} has changed before removal: term "${term}" was not present in field "${fieldName}". Removing a document after it has changed can corrupt the index!`, 'version_conflict'); - return; - } - } - } - /** - * @ignore - */ - addDocumentId(documentId) { - const shortDocumentId = this._nextId; - this._idToShortId.set(documentId, shortDocumentId); - this._documentIds.set(shortDocumentId, documentId); - this._documentCount += 1; - this._nextId += 1; - return shortDocumentId; - } - /** - * @ignore - */ - addFields(fields) { - for (let i = 0; i < fields.length; i++) { - this._fieldIds[fields[i]] = i; - } - } - /** - * @ignore - */ - addFieldLength(documentId, fieldId, count, length) { - let fieldLengths = this._fieldLength.get(documentId); - if (fieldLengths == null) - this._fieldLength.set(documentId, fieldLengths = []); - fieldLengths[fieldId] = length; - const averageFieldLength = this._avgFieldLength[fieldId] || 0; - const totalFieldLength = (averageFieldLength * count) + length; - this._avgFieldLength[fieldId] = totalFieldLength / (count + 1); - } - /** - * @ignore - */ - removeFieldLength(documentId, fieldId, count, length) { - if (count === 1) { - this._avgFieldLength[fieldId] = 0; - return; - } - const totalFieldLength = (this._avgFieldLength[fieldId] * count) - length; - this._avgFieldLength[fieldId] = totalFieldLength / (count - 1); - } - /** - * @ignore - */ - saveStoredFields(documentId, doc) { - const { storeFields, extractField } = this._options; - if (storeFields == null || storeFields.length === 0) { - return; - } - let documentFields = this._storedFields.get(documentId); - if (documentFields == null) - this._storedFields.set(documentId, documentFields = {}); - for (const fieldName of storeFields) { - const fieldValue = extractField(doc, fieldName); - if (fieldValue !== undefined) - documentFields[fieldName] = fieldValue; - } - } -} -/** - * The special wildcard symbol that can be passed to {@link MiniSearch#search} - * to match all documents - */ -MiniSearch.wildcard = Symbol('*'); -const getOwnProperty = (object, property) => Object.prototype.hasOwnProperty.call(object, property) ? object[property] : undefined; -const combinators = { - [OR]: (a, b) => { - for (const docId of b.keys()) { - const existing = a.get(docId); - if (existing == null) { - a.set(docId, b.get(docId)); - } - else { - const { score, terms, match } = b.get(docId); - existing.score = existing.score + score; - existing.match = Object.assign(existing.match, match); - assignUniqueTerms(existing.terms, terms); - } - } - return a; - }, - [AND]: (a, b) => { - const combined = new Map(); - for (const docId of b.keys()) { - const existing = a.get(docId); - if (existing == null) - continue; - const { score, terms, match } = b.get(docId); - assignUniqueTerms(existing.terms, terms); - combined.set(docId, { - score: existing.score + score, - terms: existing.terms, - match: Object.assign(existing.match, match) - }); - } - return combined; - }, - [AND_NOT]: (a, b) => { - for (const docId of b.keys()) - a.delete(docId); - return a; - } -}; -const defaultBM25params = { k: 1.2, b: 0.7, d: 0.5 }; -const calcBM25Score = (termFreq, matchingCount, totalCount, fieldLength, avgFieldLength, bm25params) => { - const { k, b, d } = bm25params; - const invDocFreq = Math.log(1 + (totalCount - matchingCount + 0.5) / (matchingCount + 0.5)); - return invDocFreq * (d + termFreq * (k + 1) / (termFreq + k * (1 - b + b * fieldLength / avgFieldLength))); -}; -const termToQuerySpec = (options) => (term, i, terms) => { - const fuzzy = (typeof options.fuzzy === 'function') - ? options.fuzzy(term, i, terms) - : (options.fuzzy || false); - const prefix = (typeof options.prefix === 'function') - ? options.prefix(term, i, terms) - : (options.prefix === true); - const termBoost = (typeof options.boostTerm === 'function') - ? options.boostTerm(term, i, terms) - : 1; - return { term, fuzzy, prefix, termBoost }; -}; -const defaultOptions = { - idField: 'id', - extractField: (document, fieldName) => document[fieldName], - tokenize: (text) => text.split(SPACE_OR_PUNCTUATION), - processTerm: (term) => term.toLowerCase(), - fields: undefined, - searchOptions: undefined, - storeFields: [], - logger: (level, message) => { - if (typeof (console === null || console === void 0 ? void 0 : console[level]) === 'function') - console[level](message); - }, - autoVacuum: true -}; -const defaultSearchOptions = { - combineWith: OR, - prefix: false, - fuzzy: false, - maxFuzzy: 6, - boost: {}, - weights: { fuzzy: 0.45, prefix: 0.375 }, - bm25: defaultBM25params -}; -const defaultAutoSuggestOptions = { - combineWith: AND, - prefix: (term, i, terms) => i === terms.length - 1 -}; -const defaultVacuumOptions = { batchSize: 1000, batchWait: 10 }; -const defaultVacuumConditions = { minDirtFactor: 0.1, minDirtCount: 20 }; -const defaultAutoVacuumOptions = Object.assign(Object.assign({}, defaultVacuumOptions), defaultVacuumConditions); -const assignUniqueTerm = (target, term) => { - // Avoid adding duplicate terms. - if (!target.includes(term)) - target.push(term); -}; -const assignUniqueTerms = (target, source) => { - for (const term of source) { - // Avoid adding duplicate terms. - if (!target.includes(term)) - target.push(term); - } -}; -const byScore = ({ score: a }, { score: b }) => b - a; -const createMap = () => new Map(); -const objectToNumericMap = (object) => { - const map = new Map(); - for (const key of Object.keys(object)) { - map.set(parseInt(key, 10), object[key]); - } - return map; -}; -const objectToNumericMapAsync = (object) => __awaiter(void 0, void 0, void 0, function* () { - const map = new Map(); - let count = 0; - for (const key of Object.keys(object)) { - map.set(parseInt(key, 10), object[key]); - if (++count % 1000 === 0) { - yield wait(0); - } - } - return map; -}); -const wait = (ms) => new Promise((resolve) => setTimeout(resolve, ms)); -// This regular expression matches any Unicode space, newline, or punctuation -// character -const SPACE_OR_PUNCTUATION = /[\n\r\p{Z}\p{P}]+/u; - -export { MiniSearch as default }; -//# sourceMappingURL=index.js.map diff --git a/src/assets/js/components/select-pagination.js b/src/assets/js/components/select-pagination.js deleted file mode 100644 index f514f85d..00000000 --- a/src/assets/js/components/select-pagination.js +++ /dev/null @@ -1,48 +0,0 @@ -class SelectPagination extends HTMLElement { - static register(tagName = 'select-pagination') { - if ("customElements" in window) customElements.define(tagName, this) - } - - static get observedAttributes() { - return ['data-base-index'] - } - - get baseIndex() { - return this.getAttribute('data-base-index') || 0 - } - - connectedCallback() { - if (this.shadowRoot) return - - this.attachShadow({ mode: 'open' }).appendChild(document.createElement('slot')) - - const uriSegments = window.location.pathname.split('/').filter(Boolean) - let pageNumber = this.extractPageNumber(uriSegments) || 0 - - this.control = this.querySelector('select') - this.control.value = pageNumber - this.control.addEventListener('change', (event) => { - pageNumber = parseInt(event.target.value) - const updatedUrlSegments = this.updateUrlSegments(uriSegments, pageNumber) - window.location.href = `${window.location.origin}/${updatedUrlSegments.join('/')}` - }) - } - - extractPageNumber(segments) { - const lastSegment = segments[segments.length - 1] - return !isNaN(lastSegment) ? parseInt(lastSegment) : null - } - - updateUrlSegments(segments, pageNumber) { - if (!isNaN(segments[segments.length - 1])) { - segments[segments.length - 1] = pageNumber.toString() - } else { - segments.push(pageNumber.toString()) - } - - if (pageNumber === parseInt(this.baseIndex)) segments.pop() - return segments - } -} - -SelectPagination.register() \ No newline at end of file diff --git a/src/assets/js/components/theme-toggle.js b/src/assets/js/components/theme-toggle.js deleted file mode 100644 index 6a667edf..00000000 --- a/src/assets/js/components/theme-toggle.js +++ /dev/null @@ -1,45 +0,0 @@ -class ThemeToggle extends HTMLElement { - static tagName = 'theme-toggle' - - static register(tagName = this.tagName, registry = globalThis.customElements) { - registry.define(tagName, this) - } - - connectedCallback() { - if (this.shadowRoot) return - this.attachShadow({ mode: 'open' }).appendChild(document.createElement('slot')) - this.root = document.documentElement - this.button = this.querySelector('button') - this.prefersDarkScheme = window.matchMedia('(prefers-color-scheme: dark)') - this.currentTheme = sessionStorage.getItem('theme') - - this.setTheme() - - this.button.addEventListener('click', () => this.toggleTheme()) - this.prefersDarkScheme.addEventListener('change', (event) => this.onPreferredColorSchemeChange(event)) - } - - setTheme() { - if (!this.currentTheme) { - this.currentTheme = this.prefersDarkScheme.matches ? 'dark' : 'light' - } - - this.theme = this.currentTheme - this.root.setAttribute('data-theme', this.theme) - } - - toggleTheme() { - this.currentTheme = this.currentTheme === 'dark' ? 'light' : 'dark' - sessionStorage.setItem('theme', this.currentTheme) - this.setTheme() - } - - onPreferredColorSchemeChange(event) { - if (!sessionStorage.getItem('theme')) { - this.currentTheme = event.matches ? 'dark' : 'light' - this.setTheme() - } - } -} - -ThemeToggle.register() \ No newline at end of file diff --git a/src/assets/js/components/youtube-video-element.js b/src/assets/js/components/youtube-video-element.js deleted file mode 100644 index b2ad092f..00000000 --- a/src/assets/js/components/youtube-video-element.js +++ /dev/null @@ -1,547 +0,0 @@ -// https://developers.google.com/youtube/iframe_api_reference - -const EMBED_BASE = 'https://www.youtube.com/embed'; -const API_URL = 'https://www.youtube.com/iframe_api'; -const API_GLOBAL = 'YT'; -const API_GLOBAL_READY = 'onYouTubeIframeAPIReady'; -const MATCH_SRC = - /(?:youtu\.be\/|youtube\.com\/(?:shorts\/|embed\/|v\/|watch\?v=|watch\?.+&v=))((\w|-){11})/; - -function getTemplateHTML(attrs) { - const iframeAttrs = { - src: serializeIframeUrl(attrs), - frameborder: 0, - width: '100%', - height: '100%', - allow: 'accelerometer; fullscreen; autoplay; encrypted-media; gyroscope; picture-in-picture', - }; - - return /*html*/` - - - `; -} - -function serializeIframeUrl(attrs) { - if (!attrs.src) return; - - const matches = attrs.src.match(MATCH_SRC); - const srcId = matches && matches[1]; - - const params = { - // ?controls=true is enabled by default in the iframe - controls: attrs.controls === '' ? null : 0, - autoplay: attrs.autoplay, - loop: attrs.loop, - mute: attrs.muted, - playsinline: attrs.playsinline, - preload: attrs.preload ?? 'metadata', - // origin: globalThis.location?.origin, - enablejsapi: 1, - showinfo: 0, - rel: 0, - iv_load_policy: 3, - modestbranding: 1, - }; - - return `${EMBED_BASE}/${srcId}?${serialize(params)}`; -} - -class YoutubeVideoElement extends (globalThis.HTMLElement ?? class {}) { - static getTemplateHTML = getTemplateHTML; - static shadowRootOptions = { mode: 'open' }; - static observedAttributes = [ - 'autoplay', - 'controls', - 'crossorigin', - 'loop', - 'muted', - 'playsinline', - 'poster', - 'preload', - 'src', - ]; - - loadComplete = new PublicPromise(); - #loadRequested; - #hasLoaded; - #readyState = 0; - #seeking = false; - #seekComplete; - isLoaded = false; - - async load() { - if (this.#loadRequested) return; - - if (!this.shadowRoot) { - this.attachShadow({ mode: 'open' }); - } - - if (this.#hasLoaded) { - this.loadComplete = new PublicPromise(); - this.isLoaded = false; - } - this.#hasLoaded = true; - - // Wait 1 tick to allow other attributes to be set. - await (this.#loadRequested = Promise.resolve()); - this.#loadRequested = null; - - this.#readyState = 0; - this.dispatchEvent(new Event('emptied')); - - let oldApi = this.api; - this.api = null; - - if (!this.src) { - // Removes the