diff --git a/package-lock.json b/package-lock.json index e091273..a006c2b 100644 --- a/package-lock.json +++ b/package-lock.json @@ -28,7 +28,6 @@ "version": "5.2.0", "resolved": "https://registry.npmjs.org/@alloc/quick-lru/-/quick-lru-5.2.0.tgz", "integrity": "sha512-UrcABB+4bUrFABwbluTIBErXwvbsU/V7TZWfmbgJfbkwiBuziS9gxdODUyuiecfdGQ85jglMW6juS3+z5TsKLw==", - "dev": true, "license": "MIT", "engines": { "node": ">=10" @@ -954,6 +953,14 @@ "node": "^18.18.0 || ^20.9.0 || >=21.1.0" } }, + "node_modules/@heroicons/react": { + "version": "2.1.5", + "resolved": "https://registry.npmjs.org/@heroicons/react/-/react-2.1.5.tgz", + "integrity": "sha512-FuzFN+BsHa+7OxbvAERtgBTNeZpUjgM/MIizfVkSCL2/edriN0Hx/DWRCR//aPYwO5QX/YlgLGXk+E3PcfZwjA==", + "peerDependencies": { + "react": ">= 16" + } + }, "node_modules/@humanwhocodes/module-importer": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/@humanwhocodes/module-importer/-/module-importer-1.0.1.tgz", @@ -986,7 +993,6 @@ "version": "8.0.2", "resolved": "https://registry.npmjs.org/@isaacs/cliui/-/cliui-8.0.2.tgz", "integrity": "sha512-O8jcjabXaleOG9DQ0+ARXWZBTfnP4WNAqzuiJK7ll44AmxGKv/J2M4TPjxjY3znBCfvBXFzucm1twdyFybFqEA==", - "dev": true, "license": "ISC", "dependencies": { "string-width": "^5.1.2", @@ -1004,7 +1010,6 @@ "version": "6.1.0", "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-6.1.0.tgz", "integrity": "sha512-7HSX4QQb4CspciLpVFwyRe79O3xsIZDDLER21kERQ71oaPodF8jL725AgJMFAYbooIqolJoRLuM81SpeUkpkvA==", - "dev": true, "license": "MIT", "engines": { "node": ">=12" @@ -1017,7 +1022,6 @@ "version": "7.1.0", "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-7.1.0.tgz", "integrity": "sha512-iq6eVVI64nQQTRYq2KtEg2d2uU7LElhTJwsH4YzIHZshxlgZms/wIc4VoDQTlG/IvVIrBKG06CrZnp0qv7hkcQ==", - "dev": true, "license": "MIT", "dependencies": { "ansi-regex": "^6.0.1" @@ -1110,7 +1114,6 @@ "version": "0.3.5", "resolved": "https://registry.npmjs.org/@jridgewell/gen-mapping/-/gen-mapping-0.3.5.tgz", "integrity": "sha512-IzL8ZoEDIBRWEzlCcRhOaCupYyN5gdIK+Q6fbFdPDg6HqX6jpkItn7DFIpW9LQzXG6Df9sA7+OKnq0qlz/GaQg==", - "dev": true, "license": "MIT", "dependencies": { "@jridgewell/set-array": "^1.2.1", @@ -1125,7 +1128,6 @@ "version": "3.1.2", "resolved": "https://registry.npmjs.org/@jridgewell/resolve-uri/-/resolve-uri-3.1.2.tgz", "integrity": "sha512-bRISgCIjP20/tbWSPWMEi54QVPRZExkuD9lJL+UIxUKtwVJA8wW1Trb1jMs1RFXo1CBTNZ/5hpC9QvmKWdopKw==", - "dev": true, "license": "MIT", "engines": { "node": ">=6.0.0" @@ -1135,7 +1137,6 @@ "version": "1.2.1", "resolved": "https://registry.npmjs.org/@jridgewell/set-array/-/set-array-1.2.1.tgz", "integrity": "sha512-R8gLRTZeyp03ymzP/6Lil/28tGeGEzhx1q2k703KGWRAI1VdvPIXdG70VJc2pAMw3NA6JKL5hhFu1sJX0Mnn/A==", - "dev": true, "license": "MIT", "engines": { "node": ">=6.0.0" @@ -1145,14 +1146,12 @@ "version": "1.5.0", "resolved": "https://registry.npmjs.org/@jridgewell/sourcemap-codec/-/sourcemap-codec-1.5.0.tgz", "integrity": "sha512-gv3ZRaISU3fjPAgNsriBRqGWQL6quFx04YMPW/zD8XMLsU32mhCCbfbO6KZFLjvYpCZ8zyDEgqsgf+PwPaM7GQ==", - "dev": true, "license": "MIT" }, "node_modules/@jridgewell/trace-mapping": { "version": "0.3.25", "resolved": "https://registry.npmjs.org/@jridgewell/trace-mapping/-/trace-mapping-0.3.25.tgz", "integrity": "sha512-vNk6aEwybGtawWmy/PzwnGDOjCkLWSD2wqvjGGAgOAwCGWySYXfYoxt00IJkTF+8Lb57DwOb3Aa0o9CApepiYQ==", - "dev": true, "license": "MIT", "dependencies": { "@jridgewell/resolve-uri": "^3.1.0", @@ -1191,11 +1190,150 @@ "react": ">=16" } }, + "node_modules/@next/env": { + "version": "14.2.11", + "resolved": "https://registry.npmjs.org/@next/env/-/env-14.2.11.tgz", + "integrity": "sha512-HYsQRSIXwiNqvzzYThrBwq6RhXo3E0n8j8nQnAs8i4fCEo2Zf/3eS0IiRA8XnRg9Ha0YnpkyJZIZg1qEwemrHw==" + }, + "node_modules/@next/swc-darwin-arm64": { + "version": "14.2.11", + "resolved": "https://registry.npmjs.org/@next/swc-darwin-arm64/-/swc-darwin-arm64-14.2.11.tgz", + "integrity": "sha512-eiY9u7wEJZWp/Pga07Qy3ZmNEfALmmSS1HtsJF3y1QEyaExu7boENz11fWqDmZ3uvcyAxCMhTrA1jfVxITQW8g==", + "cpu": [ + "arm64" + ], + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": ">= 10" + } + }, + "node_modules/@next/swc-darwin-x64": { + "version": "14.2.11", + "resolved": "https://registry.npmjs.org/@next/swc-darwin-x64/-/swc-darwin-x64-14.2.11.tgz", + "integrity": "sha512-lnB0zYCld4yE0IX3ANrVMmtAbziBb7MYekcmR6iE9bujmgERl6+FK+b0MBq0pl304lYe7zO4yxJus9H/Af8jbg==", + "cpu": [ + "x64" + ], + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": ">= 10" + } + }, + "node_modules/@next/swc-linux-arm64-gnu": { + "version": "14.2.11", + "resolved": "https://registry.npmjs.org/@next/swc-linux-arm64-gnu/-/swc-linux-arm64-gnu-14.2.11.tgz", + "integrity": "sha512-Ulo9TZVocYmUAtzvZ7FfldtwUoQY0+9z3BiXZCLSUwU2bp7GqHA7/bqrfsArDlUb2xeGwn3ZuBbKtNK8TR0A8w==", + "cpu": [ + "arm64" + ], + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">= 10" + } + }, + "node_modules/@next/swc-linux-arm64-musl": { + "version": "14.2.11", + "resolved": "https://registry.npmjs.org/@next/swc-linux-arm64-musl/-/swc-linux-arm64-musl-14.2.11.tgz", + "integrity": "sha512-fH377DnKGyUnkWlmUpFF1T90m0dADBfK11dF8sOQkiELF9M+YwDRCGe8ZyDzvQcUd20Rr5U7vpZRrAxKwd3Rzg==", + "cpu": [ + "arm64" + ], + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">= 10" + } + }, + "node_modules/@next/swc-linux-x64-gnu": { + "version": "14.2.11", + "resolved": "https://registry.npmjs.org/@next/swc-linux-x64-gnu/-/swc-linux-x64-gnu-14.2.11.tgz", + "integrity": "sha512-a0TH4ZZp4NS0LgXP/488kgvWelNpwfgGTUCDXVhPGH6pInb7yIYNgM4kmNWOxBFt+TIuOH6Pi9NnGG4XWFUyXQ==", + "cpu": [ + "x64" + ], + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">= 10" + } + }, + "node_modules/@next/swc-linux-x64-musl": { + "version": "14.2.11", + "resolved": "https://registry.npmjs.org/@next/swc-linux-x64-musl/-/swc-linux-x64-musl-14.2.11.tgz", + "integrity": "sha512-DYYZcO4Uir2gZxA4D2JcOAKVs8ZxbOFYPpXSVIgeoQbREbeEHxysVsg3nY4FrQy51e5opxt5mOHl/LzIyZBoKA==", + "cpu": [ + "x64" + ], + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">= 10" + } + }, + "node_modules/@next/swc-win32-arm64-msvc": { + "version": "14.2.11", + "resolved": "https://registry.npmjs.org/@next/swc-win32-arm64-msvc/-/swc-win32-arm64-msvc-14.2.11.tgz", + "integrity": "sha512-PwqHeKG3/kKfPpM6of1B9UJ+Er6ySUy59PeFu0Un0LBzJTRKKAg2V6J60Yqzp99m55mLa+YTbU6xj61ImTv9mg==", + "cpu": [ + "arm64" + ], + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">= 10" + } + }, + "node_modules/@next/swc-win32-ia32-msvc": { + "version": "14.2.11", + "resolved": "https://registry.npmjs.org/@next/swc-win32-ia32-msvc/-/swc-win32-ia32-msvc-14.2.11.tgz", + "integrity": "sha512-0U7PWMnOYIvM74GY6rbH6w7v+vNPDVH1gUhlwHpfInJnNe5LkmUZqhp7FNWeNa5wbVgRcRi1F1cyxp4dmeLLvA==", + "cpu": [ + "ia32" + ], + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">= 10" + } + }, + "node_modules/@next/swc-win32-x64-msvc": { + "version": "14.2.11", + "resolved": "https://registry.npmjs.org/@next/swc-win32-x64-msvc/-/swc-win32-x64-msvc-14.2.11.tgz", + "integrity": "sha512-gQpS7mcgovWoaTG1FbS5/ojF7CGfql1Q0ZLsMrhcsi2Sr9HEqsUZ70MPJyaYBXbk6iEAP7UXMD9HC8KY1qNwvA==", + "cpu": [ + "x64" + ], + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">= 10" + } + }, "node_modules/@nodelib/fs.scandir": { "version": "2.1.5", "resolved": "https://registry.npmjs.org/@nodelib/fs.scandir/-/fs.scandir-2.1.5.tgz", "integrity": "sha512-vq24Bq3ym5HEQm2NKCr3yXDwjc7vTsEThRDnkp2DK9p1uqLR+DHurm/NOTo0KG7HYHU7eppKZj3MyqYuMBf62g==", - "dev": true, "license": "MIT", "dependencies": { "@nodelib/fs.stat": "2.0.5", @@ -1209,7 +1347,6 @@ "version": "2.0.5", "resolved": "https://registry.npmjs.org/@nodelib/fs.stat/-/fs.stat-2.0.5.tgz", "integrity": "sha512-RkhPPp2zrqDAQA/2jNhnztcPAlv64XdhIp7a7454A5ovI7Bukxgt7MX7udwAu3zg1DcpPU0rz3VV1SeaqvY4+A==", - "dev": true, "license": "MIT", "engines": { "node": ">= 8" @@ -1219,7 +1356,6 @@ "version": "1.2.8", "resolved": "https://registry.npmjs.org/@nodelib/fs.walk/-/fs.walk-1.2.8.tgz", "integrity": "sha512-oGB+UxlgWcgQkgwo8GcEGwemoTFt3FIO9ababBmaGwXIoBKZ+GTy0pP185beGg7Llih/NSHSV2XAs1lnznocSg==", - "dev": true, "license": "MIT", "dependencies": { "@nodelib/fs.scandir": "2.1.5", @@ -1246,7 +1382,6 @@ "version": "0.11.0", "resolved": "https://registry.npmjs.org/@pkgjs/parseargs/-/parseargs-0.11.0.tgz", "integrity": "sha512-+1VkjdD0QBLPodGrJUeqarH8VAIvQODIbwh9XpP5Syisf7YoQgsJKPNFoqqLQlu+VQ/tVSshMR6loPMn8U+dPg==", - "dev": true, "license": "MIT", "optional": true, "engines": { @@ -2364,6 +2499,37 @@ "storybook": "^8.3.0" } }, + "node_modules/@swc/counter": { + "version": "0.1.3", + "resolved": "https://registry.npmjs.org/@swc/counter/-/counter-0.1.3.tgz", + "integrity": "sha512-e2BR4lsJkkRlKZ/qCHPw9ZaSxc0MVUd7gtbtaB7aMvHeJVYe8sOB8DBZkP2DtISHGSku9sCK6T6cnY0CtXrOCQ==" + }, + "node_modules/@tailwindcss/typography": { + "version": "0.5.15", + "resolved": "https://registry.npmjs.org/@tailwindcss/typography/-/typography-0.5.15.tgz", + "integrity": "sha512-AqhlCXl+8grUz8uqExv5OTtgpjuVIwFTSXTrh8y9/pw6q2ek7fJ+Y8ZEVw7EB2DCcuCOtEjf9w3+J3rzts01uA==", + "dependencies": { + "lodash.castarray": "^4.4.0", + "lodash.isplainobject": "^4.0.6", + "lodash.merge": "^4.6.2", + "postcss-selector-parser": "6.0.10" + }, + "peerDependencies": { + "tailwindcss": ">=3.0.0 || insiders || >=4.0.0-alpha.20" + } + }, + "node_modules/@tailwindcss/typography/node_modules/postcss-selector-parser": { + "version": "6.0.10", + "resolved": "https://registry.npmjs.org/postcss-selector-parser/-/postcss-selector-parser-6.0.10.tgz", + "integrity": "sha512-IQ7TZdoaqbT+LCpShg46jnZVlhWD2w6iQYAcYXfHARZ7X1t/UGhhceQDs5X0cGqKvYlHNOuv7Oa1xmb0oQuA3w==", + "dependencies": { + "cssesc": "^3.0.0", + "util-deprecate": "^1.0.2" + }, + "engines": { + "node": ">=4" + } + }, "node_modules/@testing-library/dom": { "version": "10.4.0", "resolved": "https://registry.npmjs.org/@testing-library/dom/-/dom-10.4.0.tgz", @@ -2648,6 +2814,14 @@ "@types/node": "*" } }, + "node_modules/@types/debug": { + "version": "4.1.12", + "resolved": "https://registry.npmjs.org/@types/debug/-/debug-4.1.12.tgz", + "integrity": "sha512-vIChWdVG3LG1SMxEvI/AK+FWJthlrqlTu7fbrlywTkkaONwk/UAGaULXRlf8vkzFBLVm0zkMdCquhL5aOjhXPQ==", + "dependencies": { + "@types/ms": "*" + } + }, "node_modules/@types/doctrine": { "version": "0.0.9", "resolved": "https://registry.npmjs.org/@types/doctrine/-/doctrine-0.0.9.tgz", @@ -2666,9 +2840,16 @@ "version": "1.0.5", "resolved": "https://registry.npmjs.org/@types/estree/-/estree-1.0.5.tgz", "integrity": "sha512-/kYRxGDLWzHOB7q+wtSUQlFrtcdUccpfy+X+9iMBpHK8QLLhx2wIPYuS5DYtR9Wa/YlZAbIovy7qVdB1Aq6Lyw==", - "dev": true, "license": "MIT" }, + "node_modules/@types/estree-jsx": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/@types/estree-jsx/-/estree-jsx-1.0.5.tgz", + "integrity": "sha512-52CcUVNFyfb1A2ALocQw/Dd1BQFNmSdkuC3BkZ6iqhdMfQz7JWOFRuJFloOzjk+6WijU56m9oKXFAXc7o3Towg==", + "dependencies": { + "@types/estree": "*" + } + }, "node_modules/@types/express": { "version": "4.17.21", "resolved": "https://registry.npmjs.org/@types/express/-/express-4.17.21.tgz", @@ -2717,7 +2898,6 @@ "version": "3.0.4", "resolved": "https://registry.npmjs.org/@types/hast/-/hast-3.0.4.tgz", "integrity": "sha512-WPs+bbQw5aCj+x6laNGWLH3wviHtoCv/P3+otBhbOhJgG8qtpdAMlTCxLtsTWA7LH1Oh/bFCHsBn0TPS5m30EQ==", - "dev": true, "license": "MIT", "dependencies": { "@types/unist": "*" @@ -2756,6 +2936,14 @@ "dev": true, "license": "MIT" }, + "node_modules/@types/mdast": { + "version": "4.0.4", + "resolved": "https://registry.npmjs.org/@types/mdast/-/mdast-4.0.4.tgz", + "integrity": "sha512-kGaNbPh1k7AFzgpud/gMdvIm5xuECykRR+JnWKQno9TAXVa6WIVCGTPvYGekIDL4uwCZQSYbUxNBSb1aUo79oA==", + "dependencies": { + "@types/unist": "*" + } + }, "node_modules/@types/mdx": { "version": "2.0.13", "resolved": "https://registry.npmjs.org/@types/mdx/-/mdx-2.0.13.tgz", @@ -2777,6 +2965,11 @@ "dev": true, "license": "MIT" }, + "node_modules/@types/ms": { + "version": "0.7.34", + "resolved": "https://registry.npmjs.org/@types/ms/-/ms-0.7.34.tgz", + "integrity": "sha512-nG96G3Wp6acyAgJqGasjODb+acrI7KltPiRxzHPXnP3NgI28bpQDRv53olbqGXbfcgF5aiiHmO3xpwEpS5Ld9g==" + }, "node_modules/@types/node": { "version": "20.16.5", "resolved": "https://registry.npmjs.org/@types/node/-/node-20.16.5.tgz", @@ -2807,7 +3000,6 @@ "version": "15.7.12", "resolved": "https://registry.npmjs.org/@types/prop-types/-/prop-types-15.7.12.tgz", "integrity": "sha512-5zvhXYtRNRluoE/jAp4GVsSduVUzNWKkOZrCDBWYtE7biZywwdC2AcEzg+cSMLFRfVgeAFqpfNabiPjxFddV1Q==", - "dev": true, "license": "MIT" }, "node_modules/@types/qs": { @@ -2828,7 +3020,6 @@ "version": "18.3.5", "resolved": "https://registry.npmjs.org/@types/react/-/react-18.3.5.tgz", "integrity": "sha512-WeqMfGJLGuLCqHGYRGHxnKrXcTitc6L/nBUWfWPcTarG3t9PsquqUMuVeXZeca+mglY4Vo5GZjCi0A3Or2lnxA==", - "dev": true, "license": "MIT", "dependencies": { "@types/prop-types": "*", @@ -2896,7 +3087,6 @@ "version": "3.0.3", "resolved": "https://registry.npmjs.org/@types/unist/-/unist-3.0.3.tgz", "integrity": "sha512-ko/gIFJRv177XgZsZcBwnqJN5x/Gien8qNOn0D5bQU/zAzVf9Zt3BlcUiLqhV9y4ARk0GbT3tnUiPNgnTXzc/Q==", - "dev": true, "license": "MIT" }, "node_modules/@types/uuid": { @@ -3152,7 +3342,6 @@ "version": "1.2.0", "resolved": "https://registry.npmjs.org/@ungap/structured-clone/-/structured-clone-1.2.0.tgz", "integrity": "sha512-zuVdFrMJiuCDQUMCzQaD6KL28MjnqqN8XnAqiEq9PNm/hCPTSGfrXCOfwj1ow4LFb/tNymJPwsNbVePc1xFqrQ==", - "dev": true, "license": "ISC" }, "node_modules/@vitejs/plugin-react": { @@ -3435,7 +3624,6 @@ "version": "5.0.1", "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz", "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==", - "dev": true, "license": "MIT", "engines": { "node": ">=8" @@ -3458,14 +3646,12 @@ "version": "1.3.0", "resolved": "https://registry.npmjs.org/any-promise/-/any-promise-1.3.0.tgz", "integrity": "sha512-7UvmKalWRt1wgjL1RrGxoSJW/0QZFIegpeGvZG9kjp8vrRu55XTHbwnqq2GpXm9uLbcuhxm3IqX9OB4MZR1b2A==", - "dev": true, "license": "MIT" }, "node_modules/anymatch": { "version": "3.1.3", "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-3.1.3.tgz", "integrity": "sha512-KMReFUr0B4t+D+OBkjR3KYqvocp2XaSzO55UcB6mgQMd3KbcE+mWTyvVV7D/zsdEbNnV6acZUutkiHQXvTr1Rw==", - "dev": true, "license": "ISC", "dependencies": { "normalize-path": "^3.0.0", @@ -3479,7 +3665,6 @@ "version": "5.0.2", "resolved": "https://registry.npmjs.org/arg/-/arg-5.0.2.tgz", "integrity": "sha512-PYjyFOLKQ9y57JvQ6QLo8dAgNqswh8M1RMJYdQduT6xbWSgK36P/Z/v+p888pM69jMMfS8Xd8F6I1kQ/I9HUGg==", - "dev": true, "license": "MIT" }, "node_modules/argparse": { @@ -3568,11 +3753,19 @@ "url": "https://github.com/sponsors/ljharb" } }, + "node_modules/bail": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/bail/-/bail-2.0.2.tgz", + "integrity": "sha512-0xO6mYd7JB2YesxDKplafRpsiOzPt9V02ddPCLbY1xYGPOX24NTyN50qnUxgCPcSoYMhKpAuBTjQoRZCAkUDRw==", + "funding": { + "type": "github", + "url": "https://github.com/sponsors/wooorm" + } + }, "node_modules/balanced-match": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz", "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==", - "dev": true, "license": "MIT" }, "node_modules/base64-js": { @@ -3600,7 +3793,6 @@ "version": "2.3.0", "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-2.3.0.tgz", "integrity": "sha512-Ceh+7ox5qe7LJuLHoY0feh3pHuUDHAcRUeyL2VYghZwfpkNIy/+8Ocg0a3UuSoYzavmylwuLWQOf3hl0jjMMIw==", - "dev": true, "license": "MIT", "engines": { "node": ">=8" @@ -3678,7 +3870,6 @@ "version": "3.0.3", "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.3.tgz", "integrity": "sha512-yQbXgO/OSZVD2IsiLlro+7Hf6Q18EJrKSEsdoMzKePKXct3gvD8oLcOQdIzGupr5Fj+EDe8gO/lxc1BzfMpxvA==", - "dev": true, "license": "MIT", "dependencies": { "fill-range": "^7.1.1" @@ -3751,6 +3942,16 @@ "ieee754": "^1.1.13" } }, + "node_modules/busboy": { + "version": "1.6.0", + "resolved": "https://registry.npmjs.org/busboy/-/busboy-1.6.0.tgz", + "integrity": "sha512-8SFQbg/0hQ9xy3UNTB0YEnsNBbWfhf7RtnzpL7TkBiTBRfrQ9Fxcnz7VJsleJpyp6rVLvXiuORqjlHi5q+PYuA==", + "dependencies": { + "streamsearch": "^1.1.0" + }, + "engines": { + "node": ">=10.16.0" + } "node_modules/buffer-from": { "version": "1.1.2", "resolved": "https://registry.npmjs.org/buffer-from/-/buffer-from-1.1.2.tgz", @@ -3812,7 +4013,6 @@ "version": "2.0.1", "resolved": "https://registry.npmjs.org/camelcase-css/-/camelcase-css-2.0.1.tgz", "integrity": "sha512-QOSvevhslijgYwRx6Rv7zKdMF8lbRmx+uQGx2+vDc+KI/eBnsy9kit5aj23AgGu3pa4t9AgwbnXWqS+iOY+2aA==", - "dev": true, "license": "MIT", "engines": { "node": ">= 6" @@ -3838,6 +4038,15 @@ ], "license": "CC-BY-4.0" }, + "node_modules/ccount": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/ccount/-/ccount-2.0.1.tgz", + "integrity": "sha512-eyrF0jiFpY+3drT6383f1qhkbGsLSifNAjA61IUjZjmLCWjItY6LB9ft9YhoDgwfmclB2zhu51Lc7+95b8NRAg==", + "funding": { + "type": "github", + "url": "https://github.com/sponsors/wooorm" + } + }, "node_modules/chai": { "version": "5.1.1", "resolved": "https://registry.npmjs.org/chai/-/chai-5.1.1.tgz", @@ -3870,6 +4079,42 @@ "node": ">=4" } }, + "node_modules/character-entities": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/character-entities/-/character-entities-2.0.2.tgz", + "integrity": "sha512-shx7oQ0Awen/BRIdkjkvz54PnEEI/EjwXDSIZp86/KKdbafHh1Df/RYGBhn4hbe2+uKC9FnT5UCEdyPz3ai9hQ==", + "funding": { + "type": "github", + "url": "https://github.com/sponsors/wooorm" + } + }, + "node_modules/character-entities-html4": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/character-entities-html4/-/character-entities-html4-2.1.0.tgz", + "integrity": "sha512-1v7fgQRj6hnSwFpq1Eu0ynr/CDEw0rXo2B61qXrLNdHZmPKgb7fqS1a2JwF0rISo9q77jDI8VMEHoApn8qDoZA==", + "funding": { + "type": "github", + "url": "https://github.com/sponsors/wooorm" + } + }, + "node_modules/character-entities-legacy": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/character-entities-legacy/-/character-entities-legacy-3.0.0.tgz", + "integrity": "sha512-RpPp0asT/6ufRm//AJVwpViZbGM/MkjQFxJccQRHmISF/22NBtsHqAWmL+/pmkPWoIUJdWyeVleTl1wydHATVQ==", + "funding": { + "type": "github", + "url": "https://github.com/sponsors/wooorm" + } + }, + "node_modules/character-reference-invalid": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/character-reference-invalid/-/character-reference-invalid-2.0.1.tgz", + "integrity": "sha512-iBZ4F4wRbyORVsu0jPV7gXkOsGYjGHPmAyv+HiHG8gi5PtC9KI2j1+v8/tlibRvjoWX027ypmG/n0HtO5t7unw==", + "funding": { + "type": "github", + "url": "https://github.com/sponsors/wooorm" + } + }, "node_modules/chardet": { "version": "0.7.0", "resolved": "https://registry.npmjs.org/chardet/-/chardet-0.7.0.tgz", @@ -3891,7 +4136,6 @@ "version": "3.6.0", "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-3.6.0.tgz", "integrity": "sha512-7VT13fmjotKpGipCW9JEQAusEPE+Ei8nl6/g4FBAmIm0GOOLMua9NDDo/DWp0ZAxCr3cPq5ZpBqmPAQgDda2Pw==", - "dev": true, "license": "MIT", "dependencies": { "anymatch": "~3.1.2", @@ -3916,7 +4160,6 @@ "version": "5.1.2", "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.2.tgz", "integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==", - "dev": true, "license": "ISC", "dependencies": { "is-glob": "^4.0.1" @@ -4059,6 +4302,10 @@ "node": ">= 10" } }, + "node_modules/client-only": { + "version": "0.0.1", + "resolved": "https://registry.npmjs.org/client-only/-/client-only-0.0.1.tgz", + "integrity": "sha512-IV3Ou0jSMzZrd3pZ48nLkT9DA7Ag1pnPzaiQhpW7c3RbcqqzvzzVu+L8gfqMp/8IM2MQtSiqaCxrrcfu8I8rMA==" "node_modules/clipboard-copy": { "version": "4.0.1", "resolved": "https://registry.npmjs.org/clipboard-copy/-/clipboard-copy-4.0.1.tgz", @@ -4153,11 +4400,19 @@ "node": ">= 0.8" } }, + "node_modules/comma-separated-tokens": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/comma-separated-tokens/-/comma-separated-tokens-2.0.3.tgz", + "integrity": "sha512-Fu4hJdvzeylCfQPp9SGWidpzrMs7tTrlu6Vb8XGaRGck8QSNZJJp538Wrb60Lax4fPwR64ViY468OIUTbRlGZg==", + "funding": { + "type": "github", + "url": "https://github.com/sponsors/wooorm" + } + }, "node_modules/commander": { "version": "4.1.1", "resolved": "https://registry.npmjs.org/commander/-/commander-4.1.1.tgz", "integrity": "sha512-NOKm8xhkzAjzFx8B2v5OAHT+u5pRQc2UCa2Vq9jYL/31o2wi9mxBA7LIFs3sV5VSC49z6pEhfbMULvShKj26WA==", - "dev": true, "license": "MIT", "engines": { "node": ">= 6" @@ -4276,7 +4531,6 @@ "version": "7.0.3", "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.3.tgz", "integrity": "sha512-iRDPJKUPVEND7dHPO8rkbOnPpyDygcDFtWjpeWNCgy8WP2rXcxXL8TskReQl6OrB2G7+UJrags1q15Fudc7G6w==", - "dev": true, "license": "MIT", "dependencies": { "path-key": "^3.1.0", @@ -4298,7 +4552,6 @@ "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" @@ -4311,14 +4564,12 @@ "version": "3.1.3", "resolved": "https://registry.npmjs.org/csstype/-/csstype-3.1.3.tgz", "integrity": "sha512-M1uQkMl8rQK/szD0LNhtqxIPLpimGm8sOBwU7lLnCpSbTyY3yeU1Vc7l4KT5zT4s/yOxHH5O7tIuuLOCnLADRw==", - "dev": true, "license": "MIT" }, "node_modules/debug": { "version": "4.3.7", "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.7.tgz", "integrity": "sha512-Er2nc/H7RrMXZBFCEim6TCmMk02Z8vLC2Rbi1KEBggpo0fS6l0S1nnapwmIi3yW/+GOJap1Krg4w0Hg80oCqgQ==", - "dev": true, "license": "MIT", "dependencies": { "ms": "^2.1.3" @@ -4332,6 +4583,18 @@ } } }, + "node_modules/decode-named-character-reference": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/decode-named-character-reference/-/decode-named-character-reference-1.0.2.tgz", + "integrity": "sha512-O8x12RzrUF8xyVcY0KJowWsmaJxQbmy0/EtnNtHRpsOcT7dFk5W598coHqBVpmWo1oQQfsCqfCmkZN5DJrZVdg==", + "dependencies": { + "character-entities": "^2.0.0" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/wooorm" + } + }, "node_modules/deep-eql": { "version": "5.0.2", "resolved": "https://registry.npmjs.org/deep-eql/-/deep-eql-5.0.2.tgz", @@ -4410,7 +4673,6 @@ "version": "2.0.3", "resolved": "https://registry.npmjs.org/dequal/-/dequal-2.0.3.tgz", "integrity": "sha512-0je+qPKHEMohvfRTCEo3CrPG6cAzAYgmzKyxRiYSSDkS6eGJdyVJm7WaYA5ECaAD9wLB2T4EEeymA5aFVcYXCA==", - "dev": true, "license": "MIT", "engines": { "node": ">=6" @@ -4427,11 +4689,22 @@ "npm": "1.2.8000 || >= 1.4.16" } }, + "node_modules/devlop": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/devlop/-/devlop-1.1.0.tgz", + "integrity": "sha512-RWmIqhcFf1lRYBvNmr7qTNuyCt/7/ns2jbpp1+PalgE/rDQcBT0fioSMUpJ93irlUhC5hrg4cYqe6U+0ImW0rA==", + "dependencies": { + "dequal": "^2.0.0" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/wooorm" + } + }, "node_modules/didyoumean": { "version": "1.2.2", "resolved": "https://registry.npmjs.org/didyoumean/-/didyoumean-1.2.2.tgz", "integrity": "sha512-gxtyfqMg7GKyhQmb056K7M3xszy/myH8w+B4RT+QXBQsvAOdc3XymqDDPHx1BgPgsdAA5SIifona89YtRATDzw==", - "dev": true, "license": "Apache-2.0" }, "node_modules/diff-match-patch": { @@ -4458,7 +4731,6 @@ "version": "1.1.3", "resolved": "https://registry.npmjs.org/dlv/-/dlv-1.1.3.tgz", "integrity": "sha512-+HlytyjlPKnIG8XuRG8WvmBP8xs8P71y+SKKS6ZXWoEgLuePxtDoUEiH7WkdePWrQ5JBpE6aoVqfZfJUQkjXwA==", - "dev": true, "license": "MIT" }, "node_modules/docs": { @@ -4499,7 +4771,6 @@ "version": "0.2.0", "resolved": "https://registry.npmjs.org/eastasianwidth/-/eastasianwidth-0.2.0.tgz", "integrity": "sha512-I88TYZWc9XiYHRQ4/3c5rjjfgkjhLyW2luGIheGERbNQ6OY7yTybanSpDXZa8y7VUP9YmDcYa+eyq4ca7iLqWA==", - "dev": true, "license": "MIT" }, "node_modules/ee-first": { @@ -4520,7 +4791,6 @@ "version": "9.2.2", "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-9.2.2.tgz", "integrity": "sha512-L18DaJsXSUk2+42pv8mLs5jJT2hqFkFE4j21wOmgbUqsZ2hL72NsUU785g9RXgo3s0ZNgVl42TiHp3ZtOv/Vyg==", - "dev": true, "license": "MIT" }, "node_modules/encodeurl": { @@ -4533,6 +4803,17 @@ "node": ">= 0.8" } }, + "node_modules/entities": { + "version": "4.5.0", + "resolved": "https://registry.npmjs.org/entities/-/entities-4.5.0.tgz", + "integrity": "sha512-V0hjH4dGPh9Ao5p0MoRY6BVqtwCjhz6vI5LT8AJ55H+4g9/4vbHx1I54fS0XuclLhDHArPQCiMjDxjaL8fPxhw==", + "engines": { + "node": ">=0.12" + }, + "funding": { + "url": "https://github.com/fb55/entities?sponsor=1" + } + }, "node_modules/environment": { "version": "1.1.0", "resolved": "https://registry.npmjs.org/environment/-/environment-1.1.0.tgz", @@ -5085,7 +5366,6 @@ "version": "4.0.1", "resolved": "https://registry.npmjs.org/esprima/-/esprima-4.0.1.tgz", "integrity": "sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A==", - "dev": true, "license": "BSD-2-Clause", "bin": { "esparse": "bin/esparse.js", @@ -5131,6 +5411,15 @@ "node": ">=4.0" } }, + "node_modules/estree-util-is-identifier-name": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/estree-util-is-identifier-name/-/estree-util-is-identifier-name-3.0.0.tgz", + "integrity": "sha512-hFtqIDZTIUZ9BXLb8y4pYGyk6+wekIivNVTcmvk8NoOh+VeRn5y6cEHzbURrWbfp1fIqdVipilzj+lfaadNZmg==", + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, "node_modules/estree-walker": { "version": "2.0.2", "resolved": "https://registry.npmjs.org/estree-walker/-/estree-walker-2.0.2.tgz", @@ -5249,6 +5538,22 @@ "dev": true, "license": "MIT" }, + "node_modules/extend": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/extend/-/extend-3.0.2.tgz", + "integrity": "sha512-fjquC59cD7CyW6urNXK0FBufkZcoiGG80wTuPujX590cB5Ttln20E2UB4S/WARVqhXffZl2LNgS+gQdPIIim/g==" + }, + "node_modules/extend-shallow": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", + "integrity": "sha512-zCnTtlxNoAiDc3gqY2aYAWFx7XWWiasuF2K8Me5WbN8otHKTUKBwjPtNpRs/rbUZm7KxWAaNj7P1a/p52GbVug==", + "dependencies": { + "is-extendable": "^0.1.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, "node_modules/external-editor": { "version": "3.1.0", "resolved": "https://registry.npmjs.org/external-editor/-/external-editor-3.1.0.tgz", @@ -5289,7 +5594,6 @@ "version": "3.3.2", "resolved": "https://registry.npmjs.org/fast-glob/-/fast-glob-3.3.2.tgz", "integrity": "sha512-oX2ruAFQwf/Orj8m737Y5adxDQO0LAB7/S5MnxCdTNDd4p6BsyIVsv9JQsATbTSq8KHRpLwIHbVlUNatxd+1Ow==", - "dev": true, "license": "MIT", "dependencies": { "@nodelib/fs.stat": "^2.0.2", @@ -5306,7 +5610,6 @@ "version": "5.1.2", "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.2.tgz", "integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==", - "dev": true, "license": "ISC", "dependencies": { "is-glob": "^4.0.1" @@ -5333,7 +5636,6 @@ "version": "1.17.1", "resolved": "https://registry.npmjs.org/fastq/-/fastq-1.17.1.tgz", "integrity": "sha512-sRVD3lWVIXWg6By68ZN7vho9a1pQcN/WBFaAAsDDFzlJjvoGx0P8z7V1t72grFJfJhu3YPZBuu25f7Kaw2jN1w==", - "dev": true, "license": "ISC", "dependencies": { "reusify": "^1.0.4" @@ -5382,7 +5684,6 @@ "version": "7.1.1", "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.1.1.tgz", "integrity": "sha512-YsGpe3WHLK8ZYi4tWDg2Jy3ebRz2rXowDxnld4bkQB00cc/1Zw9AWnC0i9ztDJitivtQvaI9KaLyKrc+hBW0yg==", - "dev": true, "license": "MIT", "dependencies": { "to-regex-range": "^5.0.1" @@ -5497,7 +5798,6 @@ "version": "3.3.0", "resolved": "https://registry.npmjs.org/foreground-child/-/foreground-child-3.3.0.tgz", "integrity": "sha512-Ld2g8rrAyMYFXBhEqMz8ZAHBi4J4uS1i/CxGMDnjyFWddMXLVcDp051DZfu+t7+ab7Wv6SMqpWmyFIj5UbfFvg==", - "dev": true, "license": "ISC", "dependencies": { "cross-spawn": "^7.0.0", @@ -5570,7 +5870,6 @@ "version": "2.3.3", "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.3.tgz", "integrity": "sha512-5xoDfX+fL7faATnagmWPpbFtwh/R77WmMMqqHGS65C3vvB0YHrgF+B1YmZ3441tMj5n63k0212XNoJwzlhffQw==", - "dev": true, "hasInstallScript": true, "license": "MIT", "optional": true, @@ -5585,7 +5884,6 @@ "version": "1.1.2", "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.2.tgz", "integrity": "sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA==", - "dev": true, "license": "MIT", "funding": { "url": "https://github.com/sponsors/ljharb" @@ -5668,7 +5966,6 @@ "version": "10.4.5", "resolved": "https://registry.npmjs.org/glob/-/glob-10.4.5.tgz", "integrity": "sha512-7Bv8RF0k6xjo7d4A/PxYLbUCfb6c+Vpd2/mB2yRDlew7Jb5hEXiCD9ibfO7wpk8i4sevK6DFny9h7EYbM3/sHg==", - "dev": true, "license": "ISC", "dependencies": { "foreground-child": "^3.1.0", @@ -5689,7 +5986,6 @@ "version": "6.0.2", "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-6.0.2.tgz", "integrity": "sha512-XxwI8EOhVQgWp6iDL+3b0r86f4d6AX6zSU55HfB4ydCEuXLXc5FcYeOu+nnGftS4TEju/11rt4KJPTMgbfmv4A==", - "dev": true, "license": "ISC", "dependencies": { "is-glob": "^4.0.3" @@ -5702,7 +5998,6 @@ "version": "2.0.1", "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-2.0.1.tgz", "integrity": "sha512-XnAIvQ8eM+kC6aULx6wuQiwVsnzsi9d3WxzV3FpWTGA19F621kwdbsAcFKXgKUHZWsy+mY6iL1sHTxWEFCytDA==", - "dev": true, "license": "MIT", "dependencies": { "balanced-match": "^1.0.0" @@ -5712,7 +6007,6 @@ "version": "9.0.5", "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-9.0.5.tgz", "integrity": "sha512-G6T0ZX48xgozx7587koeX9Ys2NYy6Gmv//P89sEte9V9whIapMNF4idKxnW2QtCcLiTWlb/wfCabAtAFWhhBow==", - "dev": true, "license": "ISC", "dependencies": { "brace-expansion": "^2.0.1" @@ -5812,6 +6106,40 @@ "dev": true, "license": "MIT" }, + "node_modules/gray-matter": { + "version": "4.0.3", + "resolved": "https://registry.npmjs.org/gray-matter/-/gray-matter-4.0.3.tgz", + "integrity": "sha512-5v6yZd4JK3eMI3FqqCouswVqwugaA9r4dNZB1wwcmrD02QkV5H0y7XBQW8QwQqEaZY1pM9aqORSORhJRdNK44Q==", + "dependencies": { + "js-yaml": "^3.13.1", + "kind-of": "^6.0.2", + "section-matter": "^1.0.0", + "strip-bom-string": "^1.0.0" + }, + "engines": { + "node": ">=6.0" + } + }, + "node_modules/gray-matter/node_modules/argparse": { + "version": "1.0.10", + "resolved": "https://registry.npmjs.org/argparse/-/argparse-1.0.10.tgz", + "integrity": "sha512-o5Roy6tNG4SL/FOkCAN6RzjiakZS25RLYFrcMttJqbdd8BWrnA+fGz57iN5Pb06pvBGvl5gQ0B48dJlslXvoTg==", + "dependencies": { + "sprintf-js": "~1.0.2" + } + }, + "node_modules/gray-matter/node_modules/js-yaml": { + "version": "3.14.1", + "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-3.14.1.tgz", + "integrity": "sha512-okMH7OXXJ7YrN9Ok3/SXrnu4iX9yOk+25nqX4imS2npuvTYDmo/QEZoqwZkYaIDk3jVvBOTOIEgEhaLOynBS9g==", + "dependencies": { + "argparse": "^1.0.7", + "esprima": "^4.0.0" + }, + "bin": { + "js-yaml": "bin/js-yaml.js" + } + }, "node_modules/has-flag": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz", @@ -5891,7 +6219,6 @@ "version": "2.0.2", "resolved": "https://registry.npmjs.org/hasown/-/hasown-2.0.2.tgz", "integrity": "sha512-0hJU9SCPvmMzIBdZFqNPXWa6dqh7WdH0cII9y+CyS8rG3nL48Bclra9HmKhVVUHyPWNH5Y7xDwAB7bfgSjkUMQ==", - "dev": true, "license": "MIT", "dependencies": { "function-bind": "^1.1.2" @@ -5900,6 +6227,25 @@ "node": ">= 0.4" } }, + "node_modules/hast-util-from-parse5": { + "version": "8.0.1", + "resolved": "https://registry.npmjs.org/hast-util-from-parse5/-/hast-util-from-parse5-8.0.1.tgz", + "integrity": "sha512-Er/Iixbc7IEa7r/XLtuG52zoqn/b3Xng/w6aZQ0xGVxzhw5xUFxcRqdPzP6yFi/4HBYRaifaI5fQ1RH8n0ZeOQ==", + "dependencies": { + "@types/hast": "^3.0.0", + "@types/unist": "^3.0.0", + "devlop": "^1.0.0", + "hastscript": "^8.0.0", + "property-information": "^6.0.0", + "vfile": "^6.0.0", + "vfile-location": "^5.0.0", + "web-namespaces": "^2.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, "node_modules/hast-util-heading-rank": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/hast-util-heading-rank/-/hast-util-heading-rank-3.0.0.tgz", @@ -5918,7 +6264,6 @@ "version": "3.0.0", "resolved": "https://registry.npmjs.org/hast-util-is-element/-/hast-util-is-element-3.0.0.tgz", "integrity": "sha512-Val9mnv2IWpLbNPqc/pUem+a7Ipj2aHacCwgNfTiK0vJKl0LF+4Ba4+v1oPHFpf3bLYmreq0/l3Gud9S5OH42g==", - "dev": true, "license": "MIT", "dependencies": { "@types/hast": "^3.0.0" @@ -5928,6 +6273,122 @@ "url": "https://opencollective.com/unified" } }, + "node_modules/hast-util-parse-selector": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/hast-util-parse-selector/-/hast-util-parse-selector-4.0.0.tgz", + "integrity": "sha512-wkQCkSYoOGCRKERFWcxMVMOcYE2K1AaNLU8DXS9arxnLOUEWbOXKXiJUNzEpqZ3JOKpnha3jkFrumEjVliDe7A==", + "dependencies": { + "@types/hast": "^3.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/hast-util-raw": { + "version": "9.0.4", + "resolved": "https://registry.npmjs.org/hast-util-raw/-/hast-util-raw-9.0.4.tgz", + "integrity": "sha512-LHE65TD2YiNsHD3YuXcKPHXPLuYh/gjp12mOfU8jxSrm1f/yJpsb0F/KKljS6U9LJoP0Ux+tCe8iJ2AsPzTdgA==", + "dependencies": { + "@types/hast": "^3.0.0", + "@types/unist": "^3.0.0", + "@ungap/structured-clone": "^1.0.0", + "hast-util-from-parse5": "^8.0.0", + "hast-util-to-parse5": "^8.0.0", + "html-void-elements": "^3.0.0", + "mdast-util-to-hast": "^13.0.0", + "parse5": "^7.0.0", + "unist-util-position": "^5.0.0", + "unist-util-visit": "^5.0.0", + "vfile": "^6.0.0", + "web-namespaces": "^2.0.0", + "zwitch": "^2.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/hast-util-sanitize": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/hast-util-sanitize/-/hast-util-sanitize-5.0.1.tgz", + "integrity": "sha512-IGrgWLuip4O2nq5CugXy4GI2V8kx4sFVy5Hd4vF7AR2gxS0N9s7nEAVUyeMtZKZvzrxVsHt73XdTsno1tClIkQ==", + "dependencies": { + "@types/hast": "^3.0.0", + "@ungap/structured-clone": "^1.2.0", + "unist-util-position": "^5.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/hast-util-to-html": { + "version": "9.0.3", + "resolved": "https://registry.npmjs.org/hast-util-to-html/-/hast-util-to-html-9.0.3.tgz", + "integrity": "sha512-M17uBDzMJ9RPCqLMO92gNNUDuBSq10a25SDBI08iCCxmorf4Yy6sYHK57n9WAbRAAaU+DuR4W6GN9K4DFZesYg==", + "dependencies": { + "@types/hast": "^3.0.0", + "@types/unist": "^3.0.0", + "ccount": "^2.0.0", + "comma-separated-tokens": "^2.0.0", + "hast-util-whitespace": "^3.0.0", + "html-void-elements": "^3.0.0", + "mdast-util-to-hast": "^13.0.0", + "property-information": "^6.0.0", + "space-separated-tokens": "^2.0.0", + "stringify-entities": "^4.0.0", + "zwitch": "^2.0.4" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/hast-util-to-jsx-runtime": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/hast-util-to-jsx-runtime/-/hast-util-to-jsx-runtime-2.3.0.tgz", + "integrity": "sha512-H/y0+IWPdsLLS738P8tDnrQ8Z+dj12zQQ6WC11TIM21C8WFVoIxcqWXf2H3hiTVZjF1AWqoimGwrTWecWrnmRQ==", + "dependencies": { + "@types/estree": "^1.0.0", + "@types/hast": "^3.0.0", + "@types/unist": "^3.0.0", + "comma-separated-tokens": "^2.0.0", + "devlop": "^1.0.0", + "estree-util-is-identifier-name": "^3.0.0", + "hast-util-whitespace": "^3.0.0", + "mdast-util-mdx-expression": "^2.0.0", + "mdast-util-mdx-jsx": "^3.0.0", + "mdast-util-mdxjs-esm": "^2.0.0", + "property-information": "^6.0.0", + "space-separated-tokens": "^2.0.0", + "style-to-object": "^1.0.0", + "unist-util-position": "^5.0.0", + "vfile-message": "^4.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/hast-util-to-parse5": { + "version": "8.0.0", + "resolved": "https://registry.npmjs.org/hast-util-to-parse5/-/hast-util-to-parse5-8.0.0.tgz", + "integrity": "sha512-3KKrV5ZVI8if87DVSi1vDeByYrkGzg4mEfeu4alwgmmIeARiBLKCZS2uw5Gb6nU9x9Yufyj3iudm6i7nl52PFw==", + "dependencies": { + "@types/hast": "^3.0.0", + "comma-separated-tokens": "^2.0.0", + "devlop": "^1.0.0", + "property-information": "^6.0.0", + "space-separated-tokens": "^2.0.0", + "web-namespaces": "^2.0.0", + "zwitch": "^2.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, "node_modules/hast-util-to-string": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/hast-util-to-string/-/hast-util-to-string-3.0.0.tgz", @@ -5942,6 +6403,49 @@ "url": "https://opencollective.com/unified" } }, + "node_modules/hast-util-to-text": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/hast-util-to-text/-/hast-util-to-text-4.0.2.tgz", + "integrity": "sha512-KK6y/BN8lbaq654j7JgBydev7wuNMcID54lkRav1P0CaE1e47P72AWWPiGKXTJU271ooYzcvTAn/Zt0REnvc7A==", + "dependencies": { + "@types/hast": "^3.0.0", + "@types/unist": "^3.0.0", + "hast-util-is-element": "^3.0.0", + "unist-util-find-after": "^5.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/hast-util-whitespace": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/hast-util-whitespace/-/hast-util-whitespace-3.0.0.tgz", + "integrity": "sha512-88JUN06ipLwsnv+dVn+OIYOvAuvBMy/Qoi6O7mQHxdPXpjy+Cd6xRkWwux7DKO+4sYILtLBRIKgsdpS2gQc7qw==", + "dependencies": { + "@types/hast": "^3.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/hastscript": { + "version": "8.0.0", + "resolved": "https://registry.npmjs.org/hastscript/-/hastscript-8.0.0.tgz", + "integrity": "sha512-dMOtzCEd3ABUeSIISmrETiKuyydk1w0pa+gE/uormcTpSYuaNJPbX1NU3JLyscSLjwAQM8bWMhhIlnCqnRvDTw==", + "dependencies": { + "@types/hast": "^3.0.0", + "comma-separated-tokens": "^2.0.0", + "hast-util-parse-selector": "^4.0.0", + "property-information": "^6.0.0", + "space-separated-tokens": "^2.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, "node_modules/he": { "version": "1.2.0", "resolved": "https://registry.npmjs.org/he/-/he-1.2.0.tgz", @@ -5964,6 +6468,24 @@ "url": "https://github.com/sponsors/sindresorhus" } }, + "node_modules/html-url-attributes": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/html-url-attributes/-/html-url-attributes-3.0.1.tgz", + "integrity": "sha512-ol6UPyBWqsrO6EJySPz2O7ZSr856WDrEzM5zMqp+FJJLGMW35cLYmmZnl0vztAZxRUoNZJFTCohfjuIJ8I4QBQ==", + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/html-void-elements": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/html-void-elements/-/html-void-elements-3.0.0.tgz", + "integrity": "sha512-bEqo66MRXsUGxWHV5IP0PUiAWwoEjba4VCzg0LjFJBpchPaTfyfCKTG6bc5F8ucKec3q5y6qOdGyYTSBEvhCrg==", + "funding": { + "type": "github", + "url": "https://github.com/sponsors/wooorm" + } + }, "node_modules/http-errors": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-2.0.0.tgz", @@ -6114,6 +6636,11 @@ "dev": true, "license": "ISC" }, + "node_modules/inline-style-parser": { + "version": "0.2.4", + "resolved": "https://registry.npmjs.org/inline-style-parser/-/inline-style-parser-0.2.4.tgz", + "integrity": "sha512-0aO8FkhNZlj/ZIbNi7Lxxr12obT7cL1moPfE4tg1LkX7LlLfC6DeX4l2ZEud1ukP9jNQyNnfzQVqwbwmAATY4Q==" + }, "node_modules/inquirer": { "version": "8.2.6", "resolved": "https://registry.npmjs.org/inquirer/-/inquirer-8.2.6.tgz", @@ -6366,6 +6893,28 @@ "url": "https://github.com/sponsors/sindresorhus" } }, + "node_modules/is-alphabetical": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/is-alphabetical/-/is-alphabetical-2.0.1.tgz", + "integrity": "sha512-FWyyY60MeTNyeSRpkM2Iry0G9hpr7/9kD40mD/cGQEuilcZYS4okz8SN2Q6rLCJ8gbCt6fN+rC+6tMGS99LaxQ==", + "funding": { + "type": "github", + "url": "https://github.com/sponsors/wooorm" + } + }, + "node_modules/is-alphanumerical": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/is-alphanumerical/-/is-alphanumerical-2.0.1.tgz", + "integrity": "sha512-hmbYhX/9MUMF5uh7tOXyK/n0ZvWpad5caBA17GsC6vyuCqaWliRG5K1qS9inmUhEMaOBIW7/whAnSwveW/LtZw==", + "dependencies": { + "is-alphabetical": "^2.0.0", + "is-decimal": "^2.0.0" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/wooorm" + } + }, "node_modules/is-arguments": { "version": "1.1.1", "resolved": "https://registry.npmjs.org/is-arguments/-/is-arguments-1.1.1.tgz", @@ -6387,7 +6936,6 @@ "version": "2.1.0", "resolved": "https://registry.npmjs.org/is-binary-path/-/is-binary-path-2.1.0.tgz", "integrity": "sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==", - "dev": true, "license": "MIT", "dependencies": { "binary-extensions": "^2.0.0" @@ -6413,7 +6961,6 @@ "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" @@ -6425,11 +6972,27 @@ "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", + "integrity": "sha512-AAB9hiomQs5DXWcRB1rqsxGUstbRroFOPPVAomNk/3XHR5JyEZChOyTWe2oayKnsSsr/kcGqF+z6yuH6HHpN0A==", + "funding": { + "type": "github", + "url": "https://github.com/sponsors/wooorm" + } + }, + "node_modules/is-extendable": { + "version": "0.1.1", + "resolved": "https://registry.npmjs.org/is-extendable/-/is-extendable-0.1.1.tgz", + "integrity": "sha512-5BMULNob1vgFX6EjQw5izWDxrecWK9AM72rugNr0TFldMOi0fj6Jk+zeKIt0xGj4cEfQIJth4w3OKWOJ4f+AFw==", + "engines": { + "node": ">=0.10.0" + } + }, "node_modules/is-extglob": { "version": "2.1.1", "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", "integrity": "sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ==", - "dev": true, "license": "MIT", "engines": { "node": ">=0.10.0" @@ -6439,7 +7002,6 @@ "version": "3.0.0", "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==", - "dev": true, "license": "MIT", "engines": { "node": ">=8" @@ -6465,7 +7027,6 @@ "version": "4.0.3", "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.3.tgz", "integrity": "sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==", - "dev": true, "license": "MIT", "dependencies": { "is-extglob": "^2.1.1" @@ -6474,6 +7035,15 @@ "node": ">=0.10.0" } }, + "node_modules/is-hexadecimal": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/is-hexadecimal/-/is-hexadecimal-2.0.1.tgz", + "integrity": "sha512-DgZQp241c8oO6cA1SbTEWiXeoxV42vlcJxgH+B3hi1AiqqKruZR3ZGF8In3fj4+/y/7rHvlOZLZtgJ/4ttYGZg==", + "funding": { + "type": "github", + "url": "https://github.com/sponsors/wooorm" + } + }, "node_modules/is-interactive": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/is-interactive/-/is-interactive-1.0.0.tgz", @@ -6488,7 +7058,6 @@ "version": "7.0.0", "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz", "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==", - "dev": true, "license": "MIT", "engines": { "node": ">=0.12.0" @@ -6504,6 +7073,17 @@ "node": ">=8" } }, + "node_modules/is-plain-obj": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/is-plain-obj/-/is-plain-obj-4.1.0.tgz", + "integrity": "sha512-+Pgi+vMuUNkJyExiMBt5IlFoMyKnr5zhJ4Uspz58WOhBF5QoIZkFyNHIbBAtHwzVAgk5RtndVNsDRN61/mmDqg==", + "engines": { + "node": ">=12" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, "node_modules/is-plain-object": { "version": "5.0.0", "resolved": "https://registry.npmjs.org/is-plain-object/-/is-plain-object-5.0.0.tgz", @@ -6560,7 +7140,6 @@ "version": "2.0.0", "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz", "integrity": "sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw==", - "dev": true, "license": "ISC" }, "node_modules/isobject": { @@ -6577,7 +7156,6 @@ "version": "3.4.3", "resolved": "https://registry.npmjs.org/jackspeak/-/jackspeak-3.4.3.tgz", "integrity": "sha512-OGlZQpz2yfahA/Rd1Y8Cd9SIEsqvXkLVoSw/cgwhnhFMDbsQFeZYoJJ7bIZBS9BcamUW96asq/npPWugM+RQBw==", - "dev": true, "license": "BlueOak-1.0.0", "dependencies": { "@isaacs/cliui": "^8.0.2" @@ -6593,7 +7171,6 @@ "version": "1.21.6", "resolved": "https://registry.npmjs.org/jiti/-/jiti-1.21.6.tgz", "integrity": "sha512-2yTgeWTWzMWkHu6Jp9NKgePDaYHbntiwvYuuJLbbN9vl7DC9DvXKOB2BC3ZZ92D3cvV/aflH0osDfwpHepQ53w==", - "dev": true, "license": "MIT", "bin": { "jiti": "bin/jiti.js" @@ -6717,7 +7294,6 @@ "version": "6.0.3", "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-6.0.3.tgz", "integrity": "sha512-dcS1ul+9tmeD95T+x28/ehLgd9mENa3LsvDTtzm3vyBEO7RPptvAD+t44WVXaUjTBRcrpFeFlC8WCruUR456hw==", - "dev": true, "license": "MIT", "engines": { "node": ">=0.10.0" @@ -6741,7 +7317,6 @@ "version": "2.1.0", "resolved": "https://registry.npmjs.org/lilconfig/-/lilconfig-2.1.0.tgz", "integrity": "sha512-utWOt/GHzuUxnLKxB6dk81RoOeoNeHgbrXiuGk4yyF5qlRz+iIVWu56E2fqGHFrXz0QNUhLB/8nKqvRH66JKGQ==", - "dev": true, "license": "MIT", "engines": { "node": ">=10" @@ -6751,7 +7326,6 @@ "version": "1.2.4", "resolved": "https://registry.npmjs.org/lines-and-columns/-/lines-and-columns-1.2.4.tgz", "integrity": "sha512-7ylylesZQ/PV29jhEDl3Ufjo6ZX7gCqJr5F7PKrqc93v7fzSymt1BpwEU8nAUXs8qzzvqhbjhK5QZg6Mt/HkBg==", - "dev": true, "license": "MIT" }, "node_modules/lint-staged": { @@ -6944,11 +7518,20 @@ "dev": true, "license": "MIT" }, + "node_modules/lodash.castarray": { + "version": "4.4.0", + "resolved": "https://registry.npmjs.org/lodash.castarray/-/lodash.castarray-4.4.0.tgz", + "integrity": "sha512-aVx8ztPv7/2ULbArGJ2Y42bG1mEQ5mGjpdvrbJcJFU3TbYybe+QlLS4pst9zV52ymy2in1KpFPiZnAOATxD4+Q==" + }, + "node_modules/lodash.isplainobject": { + "version": "4.0.6", + "resolved": "https://registry.npmjs.org/lodash.isplainobject/-/lodash.isplainobject-4.0.6.tgz", + "integrity": "sha512-oSXzaWypCMHkPC3NvBEaPHf0KsA5mvPrOPgQWDsbg8n7orZ290M0BmC/jgRZ4vcJ6DTAhjrsSYgdsW/F+MFOBA==" + }, "node_modules/lodash.merge": { "version": "4.6.2", "resolved": "https://registry.npmjs.org/lodash.merge/-/lodash.merge-4.6.2.tgz", "integrity": "sha512-0KpjqXRVvrYyCsX1swR/XTK0va6VQkQM6MNo7PqW77ByjAhoARA8EfrP1N4+KlKj8YS0ZUCtRT/YUuhyYDujIQ==", - "dev": true, "license": "MIT" }, "node_modules/log-symbols": { @@ -7182,6 +7765,15 @@ "url": "https://github.com/chalk/wrap-ansi?sponsor=1" } }, + "node_modules/longest-streak": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/longest-streak/-/longest-streak-3.1.0.tgz", + "integrity": "sha512-9Ri+o0JYgehTaVBBDoMqIl8GXtbWg711O3srftcHhZ0dqnETqLaoIK0x17fUw9rFSlK/0NlsKe0Ahhyl5pXE2g==", + "funding": { + "type": "github", + "url": "https://github.com/sponsors/wooorm" + } + }, "node_modules/loose-envify": { "version": "1.4.0", "resolved": "https://registry.npmjs.org/loose-envify/-/loose-envify-1.4.0.tgz", @@ -7204,6 +7796,28 @@ "get-func-name": "^2.0.1" } }, + "node_modules/lowlight": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/lowlight/-/lowlight-3.1.0.tgz", + "integrity": "sha512-CEbNVoSikAxwDMDPjXlqlFYiZLkDJHwyGu/MfOsJnF3d7f3tds5J3z8s/l9TMXhzfsJCCJEAsD78842mwmg0PQ==", + "dependencies": { + "@types/hast": "^3.0.0", + "devlop": "^1.0.0", + "highlight.js": "~11.9.0" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/wooorm" + } + }, + "node_modules/lowlight/node_modules/highlight.js": { + "version": "11.9.0", + "resolved": "https://registry.npmjs.org/highlight.js/-/highlight.js-11.9.0.tgz", + "integrity": "sha512-fJ7cW7fQGCYAkgv4CPfwFHrfd/cLS4Hau96JuJ+ZTOWhjnhoeN1ub1tFmALm/+lW5z4WCAuAV9bm05AP0mS6Gw==", + "engines": { + "node": ">=12.0.0" + } + }, "node_modules/lru-cache": { "version": "5.1.1", "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-5.1.1.tgz", @@ -7270,28 +7884,172 @@ "react": ">= 0.14.0" } }, - "node_modules/media-typer": { - "version": "0.3.0", - "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-0.3.0.tgz", - "integrity": "sha512-dq+qelQ9akHpcOl/gUVRTxVIOkAJ1wR3QAvb4RsVjS8oVoFjDGTc679wJYmUmknUF5HwMLOgb5O+a3KxfWapPQ==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">= 0.6" + "node_modules/mdast-util-from-markdown": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/mdast-util-from-markdown/-/mdast-util-from-markdown-2.0.1.tgz", + "integrity": "sha512-aJEUyzZ6TzlsX2s5B4Of7lN7EQtAxvtradMMglCQDyaTFgse6CmtmdJ15ElnVRlCg1vpNyVtbem0PWzlNieZsA==", + "dependencies": { + "@types/mdast": "^4.0.0", + "@types/unist": "^3.0.0", + "decode-named-character-reference": "^1.0.0", + "devlop": "^1.0.0", + "mdast-util-to-string": "^4.0.0", + "micromark": "^4.0.0", + "micromark-util-decode-numeric-character-reference": "^2.0.0", + "micromark-util-decode-string": "^2.0.0", + "micromark-util-normalize-identifier": "^2.0.0", + "micromark-util-symbol": "^2.0.0", + "micromark-util-types": "^2.0.0", + "unist-util-stringify-position": "^4.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" } }, - "node_modules/memoizerific": { - "version": "1.11.3", - "resolved": "https://registry.npmjs.org/memoizerific/-/memoizerific-1.11.3.tgz", - "integrity": "sha512-/EuHYwAPdLtXwAwSZkh/Gutery6pD2KYd44oQLhAvQp/50mpyduZh8Q7PYHXTCJ+wuXxt7oij2LXyIJOOYFPog==", - "dev": true, - "license": "MIT", + "node_modules/mdast-util-mdx-expression": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/mdast-util-mdx-expression/-/mdast-util-mdx-expression-2.0.1.tgz", + "integrity": "sha512-J6f+9hUp+ldTZqKRSg7Vw5V6MqjATc+3E4gf3CFNcuZNWD8XdyI6zQ8GqH7f8169MM6P7hMBRDVGnn7oHB9kXQ==", "dependencies": { - "map-or-similar": "^1.5.0" + "@types/estree-jsx": "^1.0.0", + "@types/hast": "^3.0.0", + "@types/mdast": "^4.0.0", + "devlop": "^1.0.0", + "mdast-util-from-markdown": "^2.0.0", + "mdast-util-to-markdown": "^2.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" } }, - "node_modules/merge-descriptors": { - "version": "1.0.3", + "node_modules/mdast-util-mdx-jsx": { + "version": "3.1.3", + "resolved": "https://registry.npmjs.org/mdast-util-mdx-jsx/-/mdast-util-mdx-jsx-3.1.3.tgz", + "integrity": "sha512-bfOjvNt+1AcbPLTFMFWY149nJz0OjmewJs3LQQ5pIyVGxP4CdOqNVJL6kTaM5c68p8q82Xv3nCyFfUnuEcH3UQ==", + "dependencies": { + "@types/estree-jsx": "^1.0.0", + "@types/hast": "^3.0.0", + "@types/mdast": "^4.0.0", + "@types/unist": "^3.0.0", + "ccount": "^2.0.0", + "devlop": "^1.1.0", + "mdast-util-from-markdown": "^2.0.0", + "mdast-util-to-markdown": "^2.0.0", + "parse-entities": "^4.0.0", + "stringify-entities": "^4.0.0", + "unist-util-stringify-position": "^4.0.0", + "vfile-message": "^4.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/mdast-util-mdxjs-esm": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/mdast-util-mdxjs-esm/-/mdast-util-mdxjs-esm-2.0.1.tgz", + "integrity": "sha512-EcmOpxsZ96CvlP03NghtH1EsLtr0n9Tm4lPUJUBccV9RwUOneqSycg19n5HGzCf+10LozMRSObtVr3ee1WoHtg==", + "dependencies": { + "@types/estree-jsx": "^1.0.0", + "@types/hast": "^3.0.0", + "@types/mdast": "^4.0.0", + "devlop": "^1.0.0", + "mdast-util-from-markdown": "^2.0.0", + "mdast-util-to-markdown": "^2.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/mdast-util-phrasing": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/mdast-util-phrasing/-/mdast-util-phrasing-4.1.0.tgz", + "integrity": "sha512-TqICwyvJJpBwvGAMZjj4J2n0X8QWp21b9l0o7eXyVJ25YNWYbJDVIyD1bZXE6WtV6RmKJVYmQAKWa0zWOABz2w==", + "dependencies": { + "@types/mdast": "^4.0.0", + "unist-util-is": "^6.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/mdast-util-to-hast": { + "version": "13.2.0", + "resolved": "https://registry.npmjs.org/mdast-util-to-hast/-/mdast-util-to-hast-13.2.0.tgz", + "integrity": "sha512-QGYKEuUsYT9ykKBCMOEDLsU5JRObWQusAolFMeko/tYPufNkRffBAQjIE+99jbA87xv6FgmjLtwjh9wBWajwAA==", + "dependencies": { + "@types/hast": "^3.0.0", + "@types/mdast": "^4.0.0", + "@ungap/structured-clone": "^1.0.0", + "devlop": "^1.0.0", + "micromark-util-sanitize-uri": "^2.0.0", + "trim-lines": "^3.0.0", + "unist-util-position": "^5.0.0", + "unist-util-visit": "^5.0.0", + "vfile": "^6.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/mdast-util-to-markdown": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/mdast-util-to-markdown/-/mdast-util-to-markdown-2.1.0.tgz", + "integrity": "sha512-SR2VnIEdVNCJbP6y7kVTJgPLifdr8WEU440fQec7qHoHOUz/oJ2jmNRqdDQ3rbiStOXb2mCDGTuwsK5OPUgYlQ==", + "dependencies": { + "@types/mdast": "^4.0.0", + "@types/unist": "^3.0.0", + "longest-streak": "^3.0.0", + "mdast-util-phrasing": "^4.0.0", + "mdast-util-to-string": "^4.0.0", + "micromark-util-decode-string": "^2.0.0", + "unist-util-visit": "^5.0.0", + "zwitch": "^2.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/mdast-util-to-string": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/mdast-util-to-string/-/mdast-util-to-string-4.0.0.tgz", + "integrity": "sha512-0H44vDimn51F0YwvxSJSm0eCDOJTRlmN0R1yBh4HLj9wiV1Dn0QoXGbvFAWj2hSItVTlCmBF1hqKlIyUBVFLPg==", + "dependencies": { + "@types/mdast": "^4.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/media-typer": { + "version": "0.3.0", + "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-0.3.0.tgz", + "integrity": "sha512-dq+qelQ9akHpcOl/gUVRTxVIOkAJ1wR3QAvb4RsVjS8oVoFjDGTc679wJYmUmknUF5HwMLOgb5O+a3KxfWapPQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/memoizerific": { + "version": "1.11.3", + "resolved": "https://registry.npmjs.org/memoizerific/-/memoizerific-1.11.3.tgz", + "integrity": "sha512-/EuHYwAPdLtXwAwSZkh/Gutery6pD2KYd44oQLhAvQp/50mpyduZh8Q7PYHXTCJ+wuXxt7oij2LXyIJOOYFPog==", + "dev": true, + "license": "MIT", + "dependencies": { + "map-or-similar": "^1.5.0" + } + }, + "node_modules/merge-descriptors": { + "version": "1.0.3", "resolved": "https://registry.npmjs.org/merge-descriptors/-/merge-descriptors-1.0.3.tgz", "integrity": "sha512-gaNvAS7TZ897/rVaZ0nMtAyxNyi/pdbjbAwUpFQpN70GqnVfOiXpeUUMKRBmzXaSQ8DdTX4/0ms62r2K+hE6mQ==", "dev": true, @@ -7311,7 +8069,6 @@ "version": "1.4.1", "resolved": "https://registry.npmjs.org/merge2/-/merge2-1.4.1.tgz", "integrity": "sha512-8q7VEgMJW4J8tcfVPy8g09NcQwZdbwFEqhe/WZkoIzjn/3TGDwtOCYtXGxA3O8tPzpczCCDgv+P2P5y00ZJOOg==", - "dev": true, "license": "MIT", "engines": { "node": ">= 8" @@ -7327,11 +8084,431 @@ "node": ">= 0.6" } }, + "node_modules/micromark": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/micromark/-/micromark-4.0.0.tgz", + "integrity": "sha512-o/sd0nMof8kYff+TqcDx3VSrgBTcZpSvYcAHIfHhv5VAuNmisCxjhx6YmxS8PFEpb9z5WKWKPdzf0jM23ro3RQ==", + "funding": [ + { + "type": "GitHub Sponsors", + "url": "https://github.com/sponsors/unifiedjs" + }, + { + "type": "OpenCollective", + "url": "https://opencollective.com/unified" + } + ], + "dependencies": { + "@types/debug": "^4.0.0", + "debug": "^4.0.0", + "decode-named-character-reference": "^1.0.0", + "devlop": "^1.0.0", + "micromark-core-commonmark": "^2.0.0", + "micromark-factory-space": "^2.0.0", + "micromark-util-character": "^2.0.0", + "micromark-util-chunked": "^2.0.0", + "micromark-util-combine-extensions": "^2.0.0", + "micromark-util-decode-numeric-character-reference": "^2.0.0", + "micromark-util-encode": "^2.0.0", + "micromark-util-normalize-identifier": "^2.0.0", + "micromark-util-resolve-all": "^2.0.0", + "micromark-util-sanitize-uri": "^2.0.0", + "micromark-util-subtokenize": "^2.0.0", + "micromark-util-symbol": "^2.0.0", + "micromark-util-types": "^2.0.0" + } + }, + "node_modules/micromark-core-commonmark": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/micromark-core-commonmark/-/micromark-core-commonmark-2.0.1.tgz", + "integrity": "sha512-CUQyKr1e///ZODyD1U3xit6zXwy1a8q2a1S1HKtIlmgvurrEpaw/Y9y6KSIbF8P59cn/NjzHyO+Q2fAyYLQrAA==", + "funding": [ + { + "type": "GitHub Sponsors", + "url": "https://github.com/sponsors/unifiedjs" + }, + { + "type": "OpenCollective", + "url": "https://opencollective.com/unified" + } + ], + "dependencies": { + "decode-named-character-reference": "^1.0.0", + "devlop": "^1.0.0", + "micromark-factory-destination": "^2.0.0", + "micromark-factory-label": "^2.0.0", + "micromark-factory-space": "^2.0.0", + "micromark-factory-title": "^2.0.0", + "micromark-factory-whitespace": "^2.0.0", + "micromark-util-character": "^2.0.0", + "micromark-util-chunked": "^2.0.0", + "micromark-util-classify-character": "^2.0.0", + "micromark-util-html-tag-name": "^2.0.0", + "micromark-util-normalize-identifier": "^2.0.0", + "micromark-util-resolve-all": "^2.0.0", + "micromark-util-subtokenize": "^2.0.0", + "micromark-util-symbol": "^2.0.0", + "micromark-util-types": "^2.0.0" + } + }, + "node_modules/micromark-factory-destination": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/micromark-factory-destination/-/micromark-factory-destination-2.0.0.tgz", + "integrity": "sha512-j9DGrQLm/Uhl2tCzcbLhy5kXsgkHUrjJHg4fFAeoMRwJmJerT9aw4FEhIbZStWN8A3qMwOp1uzHr4UL8AInxtA==", + "funding": [ + { + "type": "GitHub Sponsors", + "url": "https://github.com/sponsors/unifiedjs" + }, + { + "type": "OpenCollective", + "url": "https://opencollective.com/unified" + } + ], + "dependencies": { + "micromark-util-character": "^2.0.0", + "micromark-util-symbol": "^2.0.0", + "micromark-util-types": "^2.0.0" + } + }, + "node_modules/micromark-factory-label": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/micromark-factory-label/-/micromark-factory-label-2.0.0.tgz", + "integrity": "sha512-RR3i96ohZGde//4WSe/dJsxOX6vxIg9TimLAS3i4EhBAFx8Sm5SmqVfR8E87DPSR31nEAjZfbt91OMZWcNgdZw==", + "funding": [ + { + "type": "GitHub Sponsors", + "url": "https://github.com/sponsors/unifiedjs" + }, + { + "type": "OpenCollective", + "url": "https://opencollective.com/unified" + } + ], + "dependencies": { + "devlop": "^1.0.0", + "micromark-util-character": "^2.0.0", + "micromark-util-symbol": "^2.0.0", + "micromark-util-types": "^2.0.0" + } + }, + "node_modules/micromark-factory-space": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/micromark-factory-space/-/micromark-factory-space-2.0.0.tgz", + "integrity": "sha512-TKr+LIDX2pkBJXFLzpyPyljzYK3MtmllMUMODTQJIUfDGncESaqB90db9IAUcz4AZAJFdd8U9zOp9ty1458rxg==", + "funding": [ + { + "type": "GitHub Sponsors", + "url": "https://github.com/sponsors/unifiedjs" + }, + { + "type": "OpenCollective", + "url": "https://opencollective.com/unified" + } + ], + "dependencies": { + "micromark-util-character": "^2.0.0", + "micromark-util-types": "^2.0.0" + } + }, + "node_modules/micromark-factory-title": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/micromark-factory-title/-/micromark-factory-title-2.0.0.tgz", + "integrity": "sha512-jY8CSxmpWLOxS+t8W+FG3Xigc0RDQA9bKMY/EwILvsesiRniiVMejYTE4wumNc2f4UbAa4WsHqe3J1QS1sli+A==", + "funding": [ + { + "type": "GitHub Sponsors", + "url": "https://github.com/sponsors/unifiedjs" + }, + { + "type": "OpenCollective", + "url": "https://opencollective.com/unified" + } + ], + "dependencies": { + "micromark-factory-space": "^2.0.0", + "micromark-util-character": "^2.0.0", + "micromark-util-symbol": "^2.0.0", + "micromark-util-types": "^2.0.0" + } + }, + "node_modules/micromark-factory-whitespace": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/micromark-factory-whitespace/-/micromark-factory-whitespace-2.0.0.tgz", + "integrity": "sha512-28kbwaBjc5yAI1XadbdPYHX/eDnqaUFVikLwrO7FDnKG7lpgxnvk/XGRhX/PN0mOZ+dBSZ+LgunHS+6tYQAzhA==", + "funding": [ + { + "type": "GitHub Sponsors", + "url": "https://github.com/sponsors/unifiedjs" + }, + { + "type": "OpenCollective", + "url": "https://opencollective.com/unified" + } + ], + "dependencies": { + "micromark-factory-space": "^2.0.0", + "micromark-util-character": "^2.0.0", + "micromark-util-symbol": "^2.0.0", + "micromark-util-types": "^2.0.0" + } + }, + "node_modules/micromark-util-character": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/micromark-util-character/-/micromark-util-character-2.1.0.tgz", + "integrity": "sha512-KvOVV+X1yLBfs9dCBSopq/+G1PcgT3lAK07mC4BzXi5E7ahzMAF8oIupDDJ6mievI6F+lAATkbQQlQixJfT3aQ==", + "funding": [ + { + "type": "GitHub Sponsors", + "url": "https://github.com/sponsors/unifiedjs" + }, + { + "type": "OpenCollective", + "url": "https://opencollective.com/unified" + } + ], + "dependencies": { + "micromark-util-symbol": "^2.0.0", + "micromark-util-types": "^2.0.0" + } + }, + "node_modules/micromark-util-chunked": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/micromark-util-chunked/-/micromark-util-chunked-2.0.0.tgz", + "integrity": "sha512-anK8SWmNphkXdaKgz5hJvGa7l00qmcaUQoMYsBwDlSKFKjc6gjGXPDw3FNL3Nbwq5L8gE+RCbGqTw49FK5Qyvg==", + "funding": [ + { + "type": "GitHub Sponsors", + "url": "https://github.com/sponsors/unifiedjs" + }, + { + "type": "OpenCollective", + "url": "https://opencollective.com/unified" + } + ], + "dependencies": { + "micromark-util-symbol": "^2.0.0" + } + }, + "node_modules/micromark-util-classify-character": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/micromark-util-classify-character/-/micromark-util-classify-character-2.0.0.tgz", + "integrity": "sha512-S0ze2R9GH+fu41FA7pbSqNWObo/kzwf8rN/+IGlW/4tC6oACOs8B++bh+i9bVyNnwCcuksbFwsBme5OCKXCwIw==", + "funding": [ + { + "type": "GitHub Sponsors", + "url": "https://github.com/sponsors/unifiedjs" + }, + { + "type": "OpenCollective", + "url": "https://opencollective.com/unified" + } + ], + "dependencies": { + "micromark-util-character": "^2.0.0", + "micromark-util-symbol": "^2.0.0", + "micromark-util-types": "^2.0.0" + } + }, + "node_modules/micromark-util-combine-extensions": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/micromark-util-combine-extensions/-/micromark-util-combine-extensions-2.0.0.tgz", + "integrity": "sha512-vZZio48k7ON0fVS3CUgFatWHoKbbLTK/rT7pzpJ4Bjp5JjkZeasRfrS9wsBdDJK2cJLHMckXZdzPSSr1B8a4oQ==", + "funding": [ + { + "type": "GitHub Sponsors", + "url": "https://github.com/sponsors/unifiedjs" + }, + { + "type": "OpenCollective", + "url": "https://opencollective.com/unified" + } + ], + "dependencies": { + "micromark-util-chunked": "^2.0.0", + "micromark-util-types": "^2.0.0" + } + }, + "node_modules/micromark-util-decode-numeric-character-reference": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/micromark-util-decode-numeric-character-reference/-/micromark-util-decode-numeric-character-reference-2.0.1.tgz", + "integrity": "sha512-bmkNc7z8Wn6kgjZmVHOX3SowGmVdhYS7yBpMnuMnPzDq/6xwVA604DuOXMZTO1lvq01g+Adfa0pE2UKGlxL1XQ==", + "funding": [ + { + "type": "GitHub Sponsors", + "url": "https://github.com/sponsors/unifiedjs" + }, + { + "type": "OpenCollective", + "url": "https://opencollective.com/unified" + } + ], + "dependencies": { + "micromark-util-symbol": "^2.0.0" + } + }, + "node_modules/micromark-util-decode-string": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/micromark-util-decode-string/-/micromark-util-decode-string-2.0.0.tgz", + "integrity": "sha512-r4Sc6leeUTn3P6gk20aFMj2ntPwn6qpDZqWvYmAG6NgvFTIlj4WtrAudLi65qYoaGdXYViXYw2pkmn7QnIFasA==", + "funding": [ + { + "type": "GitHub Sponsors", + "url": "https://github.com/sponsors/unifiedjs" + }, + { + "type": "OpenCollective", + "url": "https://opencollective.com/unified" + } + ], + "dependencies": { + "decode-named-character-reference": "^1.0.0", + "micromark-util-character": "^2.0.0", + "micromark-util-decode-numeric-character-reference": "^2.0.0", + "micromark-util-symbol": "^2.0.0" + } + }, + "node_modules/micromark-util-encode": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/micromark-util-encode/-/micromark-util-encode-2.0.0.tgz", + "integrity": "sha512-pS+ROfCXAGLWCOc8egcBvT0kf27GoWMqtdarNfDcjb6YLuV5cM3ioG45Ys2qOVqeqSbjaKg72vU+Wby3eddPsA==", + "funding": [ + { + "type": "GitHub Sponsors", + "url": "https://github.com/sponsors/unifiedjs" + }, + { + "type": "OpenCollective", + "url": "https://opencollective.com/unified" + } + ] + }, + "node_modules/micromark-util-html-tag-name": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/micromark-util-html-tag-name/-/micromark-util-html-tag-name-2.0.0.tgz", + "integrity": "sha512-xNn4Pqkj2puRhKdKTm8t1YHC/BAjx6CEwRFXntTaRf/x16aqka6ouVoutm+QdkISTlT7e2zU7U4ZdlDLJd2Mcw==", + "funding": [ + { + "type": "GitHub Sponsors", + "url": "https://github.com/sponsors/unifiedjs" + }, + { + "type": "OpenCollective", + "url": "https://opencollective.com/unified" + } + ] + }, + "node_modules/micromark-util-normalize-identifier": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/micromark-util-normalize-identifier/-/micromark-util-normalize-identifier-2.0.0.tgz", + "integrity": "sha512-2xhYT0sfo85FMrUPtHcPo2rrp1lwbDEEzpx7jiH2xXJLqBuy4H0GgXk5ToU8IEwoROtXuL8ND0ttVa4rNqYK3w==", + "funding": [ + { + "type": "GitHub Sponsors", + "url": "https://github.com/sponsors/unifiedjs" + }, + { + "type": "OpenCollective", + "url": "https://opencollective.com/unified" + } + ], + "dependencies": { + "micromark-util-symbol": "^2.0.0" + } + }, + "node_modules/micromark-util-resolve-all": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/micromark-util-resolve-all/-/micromark-util-resolve-all-2.0.0.tgz", + "integrity": "sha512-6KU6qO7DZ7GJkaCgwBNtplXCvGkJToU86ybBAUdavvgsCiG8lSSvYxr9MhwmQ+udpzywHsl4RpGJsYWG1pDOcA==", + "funding": [ + { + "type": "GitHub Sponsors", + "url": "https://github.com/sponsors/unifiedjs" + }, + { + "type": "OpenCollective", + "url": "https://opencollective.com/unified" + } + ], + "dependencies": { + "micromark-util-types": "^2.0.0" + } + }, + "node_modules/micromark-util-sanitize-uri": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/micromark-util-sanitize-uri/-/micromark-util-sanitize-uri-2.0.0.tgz", + "integrity": "sha512-WhYv5UEcZrbAtlsnPuChHUAsu/iBPOVaEVsntLBIdpibO0ddy8OzavZz3iL2xVvBZOpolujSliP65Kq0/7KIYw==", + "funding": [ + { + "type": "GitHub Sponsors", + "url": "https://github.com/sponsors/unifiedjs" + }, + { + "type": "OpenCollective", + "url": "https://opencollective.com/unified" + } + ], + "dependencies": { + "micromark-util-character": "^2.0.0", + "micromark-util-encode": "^2.0.0", + "micromark-util-symbol": "^2.0.0" + } + }, + "node_modules/micromark-util-subtokenize": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/micromark-util-subtokenize/-/micromark-util-subtokenize-2.0.1.tgz", + "integrity": "sha512-jZNtiFl/1aY73yS3UGQkutD0UbhTt68qnRpw2Pifmz5wV9h8gOVsN70v+Lq/f1rKaU/W8pxRe8y8Q9FX1AOe1Q==", + "funding": [ + { + "type": "GitHub Sponsors", + "url": "https://github.com/sponsors/unifiedjs" + }, + { + "type": "OpenCollective", + "url": "https://opencollective.com/unified" + } + ], + "dependencies": { + "devlop": "^1.0.0", + "micromark-util-chunked": "^2.0.0", + "micromark-util-symbol": "^2.0.0", + "micromark-util-types": "^2.0.0" + } + }, + "node_modules/micromark-util-symbol": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/micromark-util-symbol/-/micromark-util-symbol-2.0.0.tgz", + "integrity": "sha512-8JZt9ElZ5kyTnO94muPxIGS8oyElRJaiJO8EzV6ZSyGQ1Is8xwl4Q45qU5UOg+bGH4AikWziz0iN4sFLWs8PGw==", + "funding": [ + { + "type": "GitHub Sponsors", + "url": "https://github.com/sponsors/unifiedjs" + }, + { + "type": "OpenCollective", + "url": "https://opencollective.com/unified" + } + ] + }, + "node_modules/micromark-util-types": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/micromark-util-types/-/micromark-util-types-2.0.0.tgz", + "integrity": "sha512-oNh6S2WMHWRZrmutsRmDDfkzKtxF+bc2VxLC9dvtrDIRFln627VsFP6fLMgTryGDljgLPjkrzQSDcPrjPyDJ5w==", + "funding": [ + { + "type": "GitHub Sponsors", + "url": "https://github.com/sponsors/unifiedjs" + }, + { + "type": "OpenCollective", + "url": "https://opencollective.com/unified" + } + ] + }, "node_modules/micromatch": { "version": "4.0.8", "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-4.0.8.tgz", "integrity": "sha512-PXwfBhYu0hBCPw8Dn0E+WDYb7af3dSLVWKi3HGv84IdF4TyFoC0ysxFd0Goxw7nSv4T/PzEJQxsYsEiFCKo2BA==", - "dev": true, "license": "MIT", "dependencies": { "braces": "^3.0.3", @@ -7438,7 +8615,6 @@ "version": "7.1.2", "resolved": "https://registry.npmjs.org/minipass/-/minipass-7.1.2.tgz", "integrity": "sha512-qOOzS1cBTWYF4BH8fVePDBOO9iptMnGUEZwNc/cMWnTV2nVLZ7VoNWEPHkYczZA0pdoA7dl6e7FL659nX9S2aw==", - "dev": true, "license": "ISC", "engines": { "node": ">=16 || 14 >=14.17" @@ -7461,7 +8637,6 @@ "version": "2.1.3", "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==", - "dev": true, "license": "MIT" }, "node_modules/mute-stream": { @@ -7475,7 +8650,6 @@ "version": "2.7.0", "resolved": "https://registry.npmjs.org/mz/-/mz-2.7.0.tgz", "integrity": "sha512-z81GNO7nnYMEhrGh9LeymoE4+Yr0Wn5McHIZMK5cfQCl+NDX08sCZgUc9/6MHni9IWuFLm1Z3HTCXu2z9fN62Q==", - "dev": true, "license": "MIT", "dependencies": { "any-promise": "^1.0.0", @@ -7518,6 +8692,15 @@ "node": ">= 0.6" } }, + "node_modules/next-themes": { + "version": "0.3.0", + "resolved": "https://registry.npmjs.org/next-themes/-/next-themes-0.3.0.tgz", + "integrity": "sha512-/QHIrsYpd6Kfk7xakK4svpDI5mmXP0gfvCoJdGpZQ2TOrQZmsW0QxjaiLn8wbIKjtm4BTSqLoix4lxYYOnLJ/w==", + "peerDependencies": { + "react": "^16.8 || ^17 || ^18", + "react-dom": "^16.8 || ^17 || ^18" + } + }, "node_modules/node-fetch": { "version": "2.7.0", "resolved": "https://registry.npmjs.org/node-fetch/-/node-fetch-2.7.0.tgz", @@ -7549,7 +8732,6 @@ "version": "3.0.0", "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-3.0.0.tgz", "integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==", - "dev": true, "license": "MIT", "engines": { "node": ">=0.10.0" @@ -7601,7 +8783,6 @@ "version": "3.0.0", "resolved": "https://registry.npmjs.org/object-hash/-/object-hash-3.0.0.tgz", "integrity": "sha512-RSn9F68PjH9HqtltsSnqYC1XXoWe9Bju5+213R98cNGttag9q9yAOTzdbsqvIa7aNm5WffBZFpWYr2aWrklWAw==", - "dev": true, "license": "MIT", "engines": { "node": ">= 6" @@ -7893,7 +9074,6 @@ "version": "1.0.0", "resolved": "https://registry.npmjs.org/package-json-from-dist/-/package-json-from-dist-1.0.0.tgz", "integrity": "sha512-dATvCeZN/8wQsGywez1mzHtTlP22H8OEfPrVMLNr4/eGa+ijtLn/6M5f0dY8UKNrC2O9UCU6SSoG3qRKnt7STw==", - "dev": true, "license": "BlueOak-1.0.0" }, "node_modules/package-manager-detector": { @@ -7916,6 +9096,41 @@ "node": ">=6" } }, + "node_modules/parse-entities": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/parse-entities/-/parse-entities-4.0.1.tgz", + "integrity": "sha512-SWzvYcSJh4d/SGLIOQfZ/CoNv6BTlI6YEQ7Nj82oDVnRpwe/Z/F1EMx42x3JAOwGBlCjeCH0BRJQbQ/opHL17w==", + "dependencies": { + "@types/unist": "^2.0.0", + "character-entities": "^2.0.0", + "character-entities-legacy": "^3.0.0", + "character-reference-invalid": "^2.0.0", + "decode-named-character-reference": "^1.0.0", + "is-alphanumerical": "^2.0.0", + "is-decimal": "^2.0.0", + "is-hexadecimal": "^2.0.0" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/wooorm" + } + }, + "node_modules/parse-entities/node_modules/@types/unist": { + "version": "2.0.11", + "resolved": "https://registry.npmjs.org/@types/unist/-/unist-2.0.11.tgz", + "integrity": "sha512-CmBKiL6NNo/OqgmMn95Fk9Whlp2mtvIv+KNpQKN2F4SjvrEesubTRWGYSg+BnWZOnlCaSTU1sMpsBOzgbYhnsA==" + }, + "node_modules/parse5": { + "version": "7.1.2", + "resolved": "https://registry.npmjs.org/parse5/-/parse5-7.1.2.tgz", + "integrity": "sha512-Czj1WaSVpaoj0wbhMzLmWD69anp2WH7FXMB9n1Sy8/ZFF9jolSQVMu1Ij5WIyGmcBmhk7EOndpO4mIpihVqAXw==", + "dependencies": { + "entities": "^4.4.0" + }, + "funding": { + "url": "https://github.com/inikulin/parse5?sponsor=1" + } + }, "node_modules/parseurl": { "version": "1.3.3", "resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.3.tgz", @@ -7950,7 +9165,6 @@ "version": "3.1.1", "resolved": "https://registry.npmjs.org/path-key/-/path-key-3.1.1.tgz", "integrity": "sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q==", - "dev": true, "license": "MIT", "engines": { "node": ">=8" @@ -7960,14 +9174,12 @@ "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": "1.11.1", "resolved": "https://registry.npmjs.org/path-scurry/-/path-scurry-1.11.1.tgz", "integrity": "sha512-Xa4Nw17FS9ApQFJ9umLiJS4orGjm7ZzwUrwamcGQuHSzDyth9boKDaycYdDcZDuqYATXw4HFXgaqWTctW/v1HA==", - "dev": true, "license": "BlueOak-1.0.0", "dependencies": { "lru-cache": "^10.2.0", @@ -7984,7 +9196,6 @@ "version": "10.4.3", "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-10.4.3.tgz", "integrity": "sha512-JNAzZcXrCt42VGLuYz0zfAzDfAvJWW6AfYlDBQyDV5DClI2m5sAmK+OIO7s59XfsRsWHp02jAJrRadPRGTt6SQ==", - "dev": true, "license": "ISC" }, "node_modules/path-to-regexp": { @@ -8031,7 +9242,6 @@ "version": "2.3.1", "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.1.tgz", "integrity": "sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==", - "dev": true, "license": "MIT", "engines": { "node": ">=8.6" @@ -8057,7 +9267,6 @@ "version": "2.3.0", "resolved": "https://registry.npmjs.org/pify/-/pify-2.3.0.tgz", "integrity": "sha512-udgsAY+fTnvv7kI7aaxbqwWNb0AHiB0qBO89PZKPkoTmGOgdbrHDKD+0B2X4uTfJ/FT1R09r9gTsjUjNJotuog==", - "dev": true, "license": "MIT", "engines": { "node": ">=0.10.0" @@ -8067,7 +9276,6 @@ "version": "4.0.6", "resolved": "https://registry.npmjs.org/pirates/-/pirates-4.0.6.tgz", "integrity": "sha512-saLsH7WeYYPiD25LDuLRRY/i+6HaPYr6G1OUlN39otzkSTxKnubR9RTxS3/Kk50s1g2JTgFwWQDQyplC5/SHZg==", - "dev": true, "license": "MIT", "engines": { "node": ">= 6" @@ -8181,7 +9389,6 @@ "version": "8.4.47", "resolved": "https://registry.npmjs.org/postcss/-/postcss-8.4.47.tgz", "integrity": "sha512-56rxCq7G/XfB4EkXq9Egn5GCqugWvDFjafDOThIdMBsI15iqPqR5r15TfSr1YPYeEI19YeaXMCbY6u88Y76GLQ==", - "dev": true, "funding": [ { "type": "opencollective", @@ -8210,7 +9417,6 @@ "version": "15.1.0", "resolved": "https://registry.npmjs.org/postcss-import/-/postcss-import-15.1.0.tgz", "integrity": "sha512-hpr+J05B2FVYUAXHeK1YyI267J/dDDhMU6B6civm8hSY1jYJnBXxzKDKDswzJmtLHryrjhnDjqqp/49t8FALew==", - "dev": true, "license": "MIT", "dependencies": { "postcss-value-parser": "^4.0.0", @@ -8228,7 +9434,6 @@ "version": "4.0.1", "resolved": "https://registry.npmjs.org/postcss-js/-/postcss-js-4.0.1.tgz", "integrity": "sha512-dDLF8pEO191hJMtlHFPRa8xsizHaM82MLfNkUHdUtVEV3tgTp5oj+8qbEqYM57SLfc74KSbw//4SeJma2LRVIw==", - "dev": true, "license": "MIT", "dependencies": { "camelcase-css": "^2.0.1" @@ -8248,7 +9453,6 @@ "version": "4.0.2", "resolved": "https://registry.npmjs.org/postcss-load-config/-/postcss-load-config-4.0.2.tgz", "integrity": "sha512-bSVhyJGL00wMVoPUzAVAnbEoWyqRxkjv64tUl427SKnPrENtq6hJwUojroMz2VB+Q1edmi4IfrAPpami5VVgMQ==", - "dev": true, "funding": [ { "type": "opencollective", @@ -8284,7 +9488,6 @@ "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" @@ -8297,7 +9500,6 @@ "version": "6.2.0", "resolved": "https://registry.npmjs.org/postcss-nested/-/postcss-nested-6.2.0.tgz", "integrity": "sha512-HQbt28KulC5AJzG+cZtj9kvKB93CFCdLvog1WFLf1D+xmMvPGlBstkpTEZfK5+AN9hfJocyBFCNiqyS48bpgzQ==", - "dev": true, "funding": [ { "type": "opencollective", @@ -8323,7 +9525,6 @@ "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", @@ -8337,7 +9538,6 @@ "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/prelude-ls": { @@ -8430,6 +9630,15 @@ "integrity": "sha512-24e6ynE2H+OKt4kqsOvNd8kBpV65zoxbA4BVsEOB3ARVWQki/DHzaUoC5KuON/BiccDaCCTZBuOcfZs70kR8bQ==", "license": "MIT" }, + "node_modules/property-information": { + "version": "6.5.0", + "resolved": "https://registry.npmjs.org/property-information/-/property-information-6.5.0.tgz", + "integrity": "sha512-PgTgs/BlvHxOu8QuEN7wi5A0OmXaBcHpmCSTehcs6Uuu9IkDIEo13Hy7n898RHfrQ49vKCoGeWZSaAK01nwVig==", + "funding": { + "type": "github", + "url": "https://github.com/sponsors/wooorm" + } + }, "node_modules/proxy-addr": { "version": "2.0.7", "resolved": "https://registry.npmjs.org/proxy-addr/-/proxy-addr-2.0.7.tgz", @@ -8474,7 +9683,6 @@ "version": "1.2.3", "resolved": "https://registry.npmjs.org/queue-microtask/-/queue-microtask-1.2.3.tgz", "integrity": "sha512-NuaNSa6flKT5JaSYQzJok04JzTL1CA6aGhv5rfLW3PgqA+M2ChpZQnAC8h8i4ZFkBS8X5RqkDBHA7r4hej3K9A==", - "dev": true, "funding": [ { "type": "github", @@ -8666,6 +9874,31 @@ "dev": true, "license": "MIT" }, + "node_modules/react-markdown": { + "version": "9.0.1", + "resolved": "https://registry.npmjs.org/react-markdown/-/react-markdown-9.0.1.tgz", + "integrity": "sha512-186Gw/vF1uRkydbsOIkcGXw7aHq0sZOCRFFjGrr7b9+nVZg4UfA4enXCaxm4fUzecU38sWfrNDitGhshuU7rdg==", + "dependencies": { + "@types/hast": "^3.0.0", + "devlop": "^1.0.0", + "hast-util-to-jsx-runtime": "^2.0.0", + "html-url-attributes": "^3.0.0", + "mdast-util-to-hast": "^13.0.0", + "remark-parse": "^11.0.0", + "remark-rehype": "^11.0.0", + "unified": "^11.0.0", + "unist-util-visit": "^5.0.0", + "vfile": "^6.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + }, + "peerDependencies": { + "@types/react": ">=18", + "react": ">=18" + } + }, "node_modules/react-notion-custom": { "resolved": "packages/react-notion-custom", "link": true @@ -8693,7 +9926,6 @@ "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" @@ -8718,7 +9950,6 @@ "version": "3.6.0", "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-3.6.0.tgz", "integrity": "sha512-hOS089on8RduqdbhvQ5Z37A0ESjsqz6qnRcffsMU3495FuTdqSm+7bhJ29JvIOsBDEEnan5DPu9t3To9VRlMzA==", - "dev": true, "license": "MIT", "dependencies": { "picomatch": "^2.2.1" @@ -8797,6 +10028,36 @@ "url": "https://opencollective.com/unified" } }, + "node_modules/rehype-highlight": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/rehype-highlight/-/rehype-highlight-7.0.0.tgz", + "integrity": "sha512-QtobgRgYoQaK6p1eSr2SD1i61f7bjF2kZHAQHxeCHAuJf7ZUDMvQ7owDq9YTkmar5m5TSUol+2D3bp3KfJf/oA==", + "dependencies": { + "@types/hast": "^3.0.0", + "hast-util-to-text": "^4.0.0", + "lowlight": "^3.0.0", + "unist-util-visit": "^5.0.0", + "vfile": "^6.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/rehype-raw": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/rehype-raw/-/rehype-raw-7.0.0.tgz", + "integrity": "sha512-/aE8hCfKlQeA8LmyeyQvQF3eBiLRGNlfBJEvWH7ivp9sBqs7TNqBL5X3v157rM4IFETqDnIOO+z5M/biZbo9Ww==", + "dependencies": { + "@types/hast": "^3.0.0", + "hast-util-raw": "^9.0.0", + "vfile": "^6.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, "node_modules/rehype-slug": { "version": "6.0.0", "resolved": "https://registry.npmjs.org/rehype-slug/-/rehype-slug-6.0.0.tgz", @@ -8815,6 +10076,82 @@ "url": "https://opencollective.com/unified" } }, + "node_modules/remark": { + "version": "15.0.1", + "resolved": "https://registry.npmjs.org/remark/-/remark-15.0.1.tgz", + "integrity": "sha512-Eht5w30ruCXgFmxVUSlNWQ9iiimq07URKeFS3hNc8cUWy1llX4KDWfyEDZRycMc+znsN9Ux5/tJ/BFdgdOwA3A==", + "dependencies": { + "@types/mdast": "^4.0.0", + "remark-parse": "^11.0.0", + "remark-stringify": "^11.0.0", + "unified": "^11.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/remark-html": { + "version": "16.0.1", + "resolved": "https://registry.npmjs.org/remark-html/-/remark-html-16.0.1.tgz", + "integrity": "sha512-B9JqA5i0qZe0Nsf49q3OXyGvyXuZFDzAP2iOFLEumymuYJITVpiH1IgsTEwTpdptDmZlMDMWeDmSawdaJIGCXQ==", + "dependencies": { + "@types/mdast": "^4.0.0", + "hast-util-sanitize": "^5.0.0", + "hast-util-to-html": "^9.0.0", + "mdast-util-to-hast": "^13.0.0", + "unified": "^11.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/remark-parse": { + "version": "11.0.0", + "resolved": "https://registry.npmjs.org/remark-parse/-/remark-parse-11.0.0.tgz", + "integrity": "sha512-FCxlKLNGknS5ba/1lmpYijMUzX2esxW5xQqjWxw2eHFfS2MSdaHVINFmhjo+qN1WhZhNimq0dZATN9pH0IDrpA==", + "dependencies": { + "@types/mdast": "^4.0.0", + "mdast-util-from-markdown": "^2.0.0", + "micromark-util-types": "^2.0.0", + "unified": "^11.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/remark-rehype": { + "version": "11.1.1", + "resolved": "https://registry.npmjs.org/remark-rehype/-/remark-rehype-11.1.1.tgz", + "integrity": "sha512-g/osARvjkBXb6Wo0XvAeXQohVta8i84ACbenPpoSsxTOQH/Ae0/RGP4WZgnMH5pMLpsj4FG7OHmcIcXxpza8eQ==", + "dependencies": { + "@types/hast": "^3.0.0", + "@types/mdast": "^4.0.0", + "mdast-util-to-hast": "^13.0.0", + "unified": "^11.0.0", + "vfile": "^6.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/remark-stringify": { + "version": "11.0.0", + "resolved": "https://registry.npmjs.org/remark-stringify/-/remark-stringify-11.0.0.tgz", + "integrity": "sha512-1OSmLd3awB/t8qdoEOMazZkNsfVTeY4fTsgzcQFdXNq8ToTN4ZGwrMnlda4K6smTFKD+GRV6O48i6Z4iKgPPpw==", + "dependencies": { + "@types/mdast": "^4.0.0", + "mdast-util-to-markdown": "^2.0.0", + "unified": "^11.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, "node_modules/repeat-string": { "version": "1.6.1", "resolved": "https://registry.npmjs.org/repeat-string/-/repeat-string-1.6.1.tgz", @@ -8839,7 +10176,6 @@ "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", @@ -8910,7 +10246,6 @@ "version": "1.0.4", "resolved": "https://registry.npmjs.org/reusify/-/reusify-1.0.4.tgz", "integrity": "sha512-U9nH88a3fc/ekCF1l0/UP1IosiuIjyTh7hBvXVMHYgVcfGvt897Xguj2UOLDeI5BG2m7/uwyaLVT6fbtCwTyzw==", - "dev": true, "license": "MIT", "engines": { "iojs": ">=1.0.0", @@ -8981,7 +10316,6 @@ "version": "1.2.0", "resolved": "https://registry.npmjs.org/run-parallel/-/run-parallel-1.2.0.tgz", "integrity": "sha512-5l4VyZR86LZ/lDxZTR6jqL8AFE2S0IFLMP26AbjsLVADxHdhB/c0GUsH+y39UfCi3dzz8OlQuPmnaJOMoDHQBA==", - "dev": true, "funding": [ { "type": "github", @@ -9048,6 +10382,18 @@ "loose-envify": "^1.1.0" } }, + "node_modules/section-matter": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/section-matter/-/section-matter-1.0.0.tgz", + "integrity": "sha512-vfD3pmTzGpufjScBh50YHKzEu2lxBWhVEHsNGoEXmCmn2hKGfeNLYMzCJpe8cD7gqX7TJluOVpBkAequ6dgMmA==", + "dependencies": { + "extend-shallow": "^2.0.1", + "kind-of": "^6.0.0" + }, + "engines": { + "node": ">=4" + } + }, "node_modules/semver": { "version": "6.3.1", "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.1.tgz", @@ -9168,7 +10514,6 @@ "version": "2.0.0", "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-2.0.0.tgz", "integrity": "sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA==", - "dev": true, "license": "MIT", "dependencies": { "shebang-regex": "^3.0.0" @@ -9181,7 +10526,6 @@ "version": "3.0.0", "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-3.0.0.tgz", "integrity": "sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==", - "dev": true, "license": "MIT", "engines": { "node": ">=8" @@ -9217,7 +10561,6 @@ "version": "4.1.0", "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-4.1.0.tgz", "integrity": "sha512-bzyZ1e88w9O1iNJbKnOlvYTrWPDl46O1bG0D3XInv+9tkPrxrN8jUUTiFlDkkmKWgn1M6CfIA13SuGqOa9Korw==", - "dev": true, "license": "ISC", "engines": { "node": ">=14" @@ -9313,13 +10656,17 @@ "version": "2.0.2", "resolved": "https://registry.npmjs.org/space-separated-tokens/-/space-separated-tokens-2.0.2.tgz", "integrity": "sha512-PEGlAwrG8yXGXRjW32fGbg66JAlOAwbObuqVoJpv/mRgoWDQfgH1wDPvtzWyUSNAXBGSk8h755YDbbcEy3SH2Q==", - "dev": true, "license": "MIT", "funding": { "type": "github", "url": "https://github.com/sponsors/wooorm" } }, + "node_modules/sprintf-js": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/sprintf-js/-/sprintf-js-1.0.3.tgz", + "integrity": "sha512-D9cPgkvLlV3t3IzL0D0YLvGA9Ahk4PcvVwUbN0dSGr1aP0Nrt4AEnTUbuGvquEC0mA64Gqt1fzirlRs5ibXx8g==" + }, "node_modules/stackback": { "version": "0.0.2", "resolved": "https://registry.npmjs.org/stackback/-/stackback-0.0.2.tgz", @@ -9367,6 +10714,14 @@ "url": "https://opencollective.com/storybook" } }, + "node_modules/streamsearch": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/streamsearch/-/streamsearch-1.1.0.tgz", + "integrity": "sha512-Mcc5wHehp9aXz1ax6bZUyY5afg9u2rv5cqQI3mRrYkGC8rW2hM02jWuwjtL++LS5qinSyhj2QfLyNsuc+VsExg==", + "engines": { + "node": ">=10.0.0" + } + }, "node_modules/string_decoder": { "version": "1.3.0", "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.3.0.tgz", @@ -9391,7 +10746,6 @@ "version": "5.1.2", "resolved": "https://registry.npmjs.org/string-width/-/string-width-5.1.2.tgz", "integrity": "sha512-HnLOCR3vjcY8beoNLtcjZ5/nxn2afmME6lhrDrebokqMap+XbeW8n9TXpPDOqdGK5qcI3oT0GKTW6wC7EMiVqA==", - "dev": true, "license": "MIT", "dependencies": { "eastasianwidth": "^0.2.0", @@ -9410,7 +10764,6 @@ "version": "4.2.3", "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.3.tgz", "integrity": "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==", - "dev": true, "license": "MIT", "dependencies": { "emoji-regex": "^8.0.0", @@ -9425,14 +10778,12 @@ "version": "8.0.0", "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz", "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==", - "dev": true, "license": "MIT" }, "node_modules/string-width/node_modules/ansi-regex": { "version": "6.1.0", "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-6.1.0.tgz", "integrity": "sha512-7HSX4QQb4CspciLpVFwyRe79O3xsIZDDLER21kERQ71oaPodF8jL725AgJMFAYbooIqolJoRLuM81SpeUkpkvA==", - "dev": true, "license": "MIT", "engines": { "node": ">=12" @@ -9445,7 +10796,6 @@ "version": "7.1.0", "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-7.1.0.tgz", "integrity": "sha512-iq6eVVI64nQQTRYq2KtEg2d2uU7LElhTJwsH4YzIHZshxlgZms/wIc4VoDQTlG/IvVIrBKG06CrZnp0qv7hkcQ==", - "dev": true, "license": "MIT", "dependencies": { "ansi-regex": "^6.0.1" @@ -9457,11 +10807,23 @@ "url": "https://github.com/chalk/strip-ansi?sponsor=1" } }, + "node_modules/stringify-entities": { + "version": "4.0.4", + "resolved": "https://registry.npmjs.org/stringify-entities/-/stringify-entities-4.0.4.tgz", + "integrity": "sha512-IwfBptatlO+QCJUo19AqvrPNqlVMpW9YEL2LIVY+Rpv2qsjCGxaDLNRgeGsQWJhfItebuJhsGSLjaBbNSQ+ieg==", + "dependencies": { + "character-entities-html4": "^2.0.0", + "character-entities-legacy": "^3.0.0" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/wooorm" + } + }, "node_modules/strip-ansi": { "version": "6.0.1", "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", - "dev": true, "license": "MIT", "dependencies": { "ansi-regex": "^5.0.1" @@ -9475,7 +10837,6 @@ "version": "6.0.1", "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", - "dev": true, "license": "MIT", "dependencies": { "ansi-regex": "^5.0.1" @@ -9494,6 +10855,14 @@ "node": ">=4" } }, + "node_modules/strip-bom-string": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/strip-bom-string/-/strip-bom-string-1.0.0.tgz", + "integrity": "sha512-uCC2VHvQRYu+lMh4My/sFNmF2klFymLX1wHJeXnbEJERpV/ZsVuonzerjfrGpIGF7LBVa1O7i9kjiWvJiFck8g==", + "engines": { + "node": ">=0.10.0" + } + }, "node_modules/strip-final-newline": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/strip-final-newline/-/strip-final-newline-3.0.0.tgz", @@ -9536,11 +10905,40 @@ "url": "https://github.com/sponsors/sindresorhus" } }, + "node_modules/style-to-object": { + "version": "1.0.8", + "resolved": "https://registry.npmjs.org/style-to-object/-/style-to-object-1.0.8.tgz", + "integrity": "sha512-xT47I/Eo0rwJmaXC4oilDGDWLohVhR6o/xAQcPQN8q6QBuZVL8qMYL85kLmST5cPjAorwvqIA4qXTRQoYHaL6g==", + "dependencies": { + "inline-style-parser": "0.2.4" + } + }, + "node_modules/styled-jsx": { + "version": "5.1.1", + "resolved": "https://registry.npmjs.org/styled-jsx/-/styled-jsx-5.1.1.tgz", + "integrity": "sha512-pW7uC1l4mBZ8ugbiZrcIsiIvVx1UmTfw7UkC3Um2tmfUq9Bhk8IiyEIPl6F8agHgjzku6j0xQEZbfA5uSgSaCw==", + "dependencies": { + "client-only": "0.0.1" + }, + "engines": { + "node": ">= 12.0.0" + }, + "peerDependencies": { + "react": ">= 16.8.0 || 17.x.x || ^18.0.0-0" + }, + "peerDependenciesMeta": { + "@babel/core": { + "optional": true + }, + "babel-plugin-macros": { + "optional": true + } + } + }, "node_modules/sucrase": { "version": "3.35.0", "resolved": "https://registry.npmjs.org/sucrase/-/sucrase-3.35.0.tgz", "integrity": "sha512-8EbVDiu9iN/nESwxeSxDKe0dunta1GOlHufmSSXxMD2z2/tMZpDMpvXQGsc+ajGo8y2uYUmixaSRUc/QPoQ0GA==", - "dev": true, "license": "MIT", "dependencies": { "@jridgewell/gen-mapping": "^0.3.2", @@ -9576,7 +10974,6 @@ "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" @@ -9589,7 +10986,6 @@ "version": "3.4.11", "resolved": "https://registry.npmjs.org/tailwindcss/-/tailwindcss-3.4.11.tgz", "integrity": "sha512-qhEuBcLemjSJk5ajccN9xJFtM/h0AVCPaA6C92jNP+M2J8kX+eMJHI7R2HFKUvvAsMpcfLILMCFYSeDwpMmlUg==", - "dev": true, "license": "MIT", "dependencies": { "@alloc/quick-lru": "^5.2.0", @@ -9644,7 +11040,6 @@ "version": "3.3.1", "resolved": "https://registry.npmjs.org/thenify/-/thenify-3.3.1.tgz", "integrity": "sha512-RVZSIV5IG10Hk3enotrhvz0T9em6cyHBLkH/YAZuKqd8hRkKhSfCGIcP2KUY0EPxndzANBmNllzWPwak+bheSw==", - "dev": true, "license": "MIT", "dependencies": { "any-promise": "^1.0.0" @@ -9654,7 +11049,6 @@ "version": "1.6.0", "resolved": "https://registry.npmjs.org/thenify-all/-/thenify-all-1.6.0.tgz", "integrity": "sha512-RNxQH/qI8/t3thXJDwcstUO4zeqo64+Uy/+sNVRBx4Xn2OX+OZ9oP+iJnNFqplFra2ZUVeKCSa2oVWi3T4uVmA==", - "dev": true, "license": "MIT", "dependencies": { "thenify": ">= 3.1.0 < 4" @@ -9748,7 +11142,6 @@ "version": "5.0.1", "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz", "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==", - "dev": true, "license": "MIT", "dependencies": { "is-number": "^7.0.0" @@ -9773,6 +11166,24 @@ "integrity": "sha512-N3WMsuqV66lT30CrXNbEjx4GEwlow3v6rr4mCcv6prnfwhS01rkgyFdjPNBYd9br7LpXV1+Emh01fHnq2Gdgrw==", "license": "MIT" }, + "node_modules/trim-lines": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/trim-lines/-/trim-lines-3.0.1.tgz", + "integrity": "sha512-kRj8B+YHZCc9kQYdWfJB2/oUl9rA99qbowYYBtr4ui4mZyAQ2JpvVBd/6U2YloATfqBhBTSMhTpgBHtU0Mf3Rg==", + "funding": { + "type": "github", + "url": "https://github.com/sponsors/wooorm" + } + }, + "node_modules/trough": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/trough/-/trough-2.2.0.tgz", + "integrity": "sha512-tmMpK00BjZiUyVyvrBK7knerNgmgvcV/KLVyuma/SC+TQN167GrMRciANTz09+k3zW8L8t60jWO1GpfkZdjTaw==", + "funding": { + "type": "github", + "url": "https://github.com/sponsors/wooorm" + } + }, "node_modules/ts-api-utils": { "version": "1.3.0", "resolved": "https://registry.npmjs.org/ts-api-utils/-/ts-api-utils-1.3.0.tgz", @@ -9800,7 +11211,6 @@ "version": "0.1.13", "resolved": "https://registry.npmjs.org/ts-interface-checker/-/ts-interface-checker-0.1.13.tgz", "integrity": "sha512-Y/arvbn+rrz3JCKl9C4kVNfTfSm2/mEp5FSz5EsZSANGPSlQrpRI5M4PKF+mJnE52jOO90PnPSc3Ur3bTQw0gA==", - "dev": true, "license": "Apache-2.0" }, "node_modules/ts-patch": { @@ -9930,7 +11340,6 @@ "version": "2.7.0", "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.7.0.tgz", "integrity": "sha512-gLXCKdN1/j47AiHiOkJN69hJmcbGTHI0ImLmbYLHykhgeN0jVGola9yVjFgzCUklsZQMW55o+dW7IXv3RCXDzA==", - "dev": true, "license": "0BSD" }, "node_modules/tsutils": { @@ -10085,11 +11494,41 @@ "integrity": "sha512-ve2KP6f/JnbPBFyobGHuerC9g1FYGn/F8n1LWTwNxCEzd6IfqTwUQcNXgEtmmQ6DlRrC1hrSrBnCZPokRrDHjw==", "license": "MIT" }, + "node_modules/unified": { + "version": "11.0.5", + "resolved": "https://registry.npmjs.org/unified/-/unified-11.0.5.tgz", + "integrity": "sha512-xKvGhPWw3k84Qjh8bI3ZeJjqnyadK+GEFtazSfZv/rKeTkTjOJho6mFqh2SM96iIcZokxiOpg78GazTSg8+KHA==", + "dependencies": { + "@types/unist": "^3.0.0", + "bail": "^2.0.0", + "devlop": "^1.0.0", + "extend": "^3.0.0", + "is-plain-obj": "^4.0.0", + "trough": "^2.0.0", + "vfile": "^6.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/unist-util-find-after": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/unist-util-find-after/-/unist-util-find-after-5.0.0.tgz", + "integrity": "sha512-amQa0Ep2m6hE2g72AugUItjbuM8X8cGQnFoHk0pGfrFeT9GZhzN5SW8nRsiGKK7Aif4CrACPENkA6P/Lw6fHGQ==", + "dependencies": { + "@types/unist": "^3.0.0", + "unist-util-is": "^6.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, "node_modules/unist-util-is": { "version": "6.0.0", "resolved": "https://registry.npmjs.org/unist-util-is/-/unist-util-is-6.0.0.tgz", "integrity": "sha512-2qCTHimwdxLfz+YzdGfkqNlH0tLi9xjTnHddPmJwtIG9MGsdbutfTc4P+haPD7l7Cjxf/WZj+we5qfVPvvxfYw==", - "dev": true, "license": "MIT", "dependencies": { "@types/unist": "^3.0.0" @@ -10099,11 +11538,34 @@ "url": "https://opencollective.com/unified" } }, + "node_modules/unist-util-position": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/unist-util-position/-/unist-util-position-5.0.0.tgz", + "integrity": "sha512-fucsC7HjXvkB5R3kTCO7kUjRdrS0BJt3M/FPxmHMBOm8JQi2BsHAHFsy27E0EolP8rp0NzXsJ+jNPyDWvOJZPA==", + "dependencies": { + "@types/unist": "^3.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/unist-util-stringify-position": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/unist-util-stringify-position/-/unist-util-stringify-position-4.0.0.tgz", + "integrity": "sha512-0ASV06AAoKCDkS2+xw5RXJywruurpbC4JZSm7nr7MOt1ojAzvyyaO+UxZf18j8FCF6kmzCZKcAgN/yu2gm2XgQ==", + "dependencies": { + "@types/unist": "^3.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, "node_modules/unist-util-visit": { "version": "5.0.0", "resolved": "https://registry.npmjs.org/unist-util-visit/-/unist-util-visit-5.0.0.tgz", "integrity": "sha512-MR04uvD+07cwl/yhVuVWAtw+3GOR/knlL55Nd/wAdblk27GCVt3lqpTivy/tkJcZoNPzTwS1Y+KMojlLDhoTzg==", - "dev": true, "license": "MIT", "dependencies": { "@types/unist": "^3.0.0", @@ -10119,7 +11581,6 @@ "version": "6.0.1", "resolved": "https://registry.npmjs.org/unist-util-visit-parents/-/unist-util-visit-parents-6.0.1.tgz", "integrity": "sha512-L/PqWzfTP9lzzEa6CKs0k2nARxTdZduw3zyh8d2NVBnsyvHjSX4TWse388YrrQKbvI8w20fGjGlhgT96WwKykw==", - "dev": true, "license": "MIT", "dependencies": { "@types/unist": "^3.0.0", @@ -10231,7 +11692,6 @@ "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": { @@ -10268,6 +11728,45 @@ "node": ">= 0.8" } }, + "node_modules/vfile": { + "version": "6.0.3", + "resolved": "https://registry.npmjs.org/vfile/-/vfile-6.0.3.tgz", + "integrity": "sha512-KzIbH/9tXat2u30jf+smMwFCsno4wHVdNmzFyL+T/L3UGqqk6JKfVqOFOZEpZSHADH1k40ab6NUIXZq422ov3Q==", + "dependencies": { + "@types/unist": "^3.0.0", + "vfile-message": "^4.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/vfile-location": { + "version": "5.0.3", + "resolved": "https://registry.npmjs.org/vfile-location/-/vfile-location-5.0.3.tgz", + "integrity": "sha512-5yXvWDEgqeiYiBe1lbxYF7UMAIm/IcopxMHrMQDq3nvKcjPKIhZklUKL+AE7J7uApI4kwe2snsK+eI6UTj9EHg==", + "dependencies": { + "@types/unist": "^3.0.0", + "vfile": "^6.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/vfile-message": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/vfile-message/-/vfile-message-4.0.2.tgz", + "integrity": "sha512-jRDZ1IMLttGj41KcZvlrYAaI3CfqpLpfpf+Mfig13viT6NKvRzWZ+lXz0Y5D60w6uJIBAOGq9mSHf0gktF0duw==", + "dependencies": { + "@types/unist": "^3.0.0", + "unist-util-stringify-position": "^4.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, "node_modules/vite": { "version": "5.4.5", "resolved": "https://registry.npmjs.org/vite/-/vite-5.4.5.tgz", @@ -10454,6 +11953,15 @@ "defaults": "^1.0.3" } }, + "node_modules/web-namespaces": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/web-namespaces/-/web-namespaces-2.0.1.tgz", + "integrity": "sha512-bKr1DkiNa2krS7qxNtdrtHAmzuYGFQLiQ13TsorsdT6ULTkPLKuu5+GsFpDlg6JFjUTwX2DyhMPG2be8uPrqsQ==", + "funding": { + "type": "github", + "url": "https://github.com/sponsors/wooorm" + } + }, "node_modules/webidl-conversions": { "version": "3.0.1", "resolved": "https://registry.npmjs.org/webidl-conversions/-/webidl-conversions-3.0.1.tgz", @@ -10481,7 +11989,6 @@ "version": "2.0.2", "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz", "integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==", - "dev": true, "license": "ISC", "dependencies": { "isexe": "^2.0.0" @@ -10544,7 +12051,6 @@ "version": "8.1.0", "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-8.1.0.tgz", "integrity": "sha512-si7QWI6zUMq56bESFvagtmzMdGOtoxfR+Sez11Mobfc7tm+VkUckk9bW2UeffTGVUbOksxmSw0AA2gs8g71NCQ==", - "dev": true, "license": "MIT", "dependencies": { "ansi-styles": "^6.1.0", @@ -10563,7 +12069,6 @@ "version": "7.0.0", "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-7.0.0.tgz", "integrity": "sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q==", - "dev": true, "license": "MIT", "dependencies": { "ansi-styles": "^4.0.0", @@ -10581,7 +12086,6 @@ "version": "4.3.0", "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", - "dev": true, "license": "MIT", "dependencies": { "color-convert": "^2.0.1" @@ -10597,7 +12101,6 @@ "version": "2.0.1", "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", - "dev": true, "license": "MIT", "dependencies": { "color-name": "~1.1.4" @@ -10610,21 +12113,18 @@ "version": "1.1.4", "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", - "dev": true, "license": "MIT" }, "node_modules/wrap-ansi-cjs/node_modules/emoji-regex": { "version": "8.0.0", "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz", "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==", - "dev": true, "license": "MIT" }, "node_modules/wrap-ansi-cjs/node_modules/string-width": { "version": "4.2.3", "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.3.tgz", "integrity": "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==", - "dev": true, "license": "MIT", "dependencies": { "emoji-regex": "^8.0.0", @@ -10639,7 +12139,6 @@ "version": "6.1.0", "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-6.1.0.tgz", "integrity": "sha512-7HSX4QQb4CspciLpVFwyRe79O3xsIZDDLER21kERQ71oaPodF8jL725AgJMFAYbooIqolJoRLuM81SpeUkpkvA==", - "dev": true, "license": "MIT", "engines": { "node": ">=12" @@ -10652,7 +12151,6 @@ "version": "6.2.1", "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-6.2.1.tgz", "integrity": "sha512-bN798gFfQX+viw3R7yrGWRqnrN2oRkEkUjjl4JNn4E8GxxbjtG3FbrEIIY3l8/hrwUwIeCZvi4QuOTP4MErVug==", - "dev": true, "license": "MIT", "engines": { "node": ">=12" @@ -10665,7 +12163,6 @@ "version": "7.1.0", "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-7.1.0.tgz", "integrity": "sha512-iq6eVVI64nQQTRYq2KtEg2d2uU7LElhTJwsH4YzIHZshxlgZms/wIc4VoDQTlG/IvVIrBKG06CrZnp0qv7hkcQ==", - "dev": true, "license": "MIT", "dependencies": { "ansi-regex": "^6.0.1" @@ -10717,7 +12214,6 @@ "version": "2.5.1", "resolved": "https://registry.npmjs.org/yaml/-/yaml-2.5.1.tgz", "integrity": "sha512-bLQOjaX/ADgQ20isPJRvF0iRUHIxVhYvr53Of7wGcWlO2jvtUlH5m87DsmulFVxRpNLOnI4tB6p/oh8D7kpn9Q==", - "dev": true, "license": "ISC", "bin": { "yaml": "bin.mjs" @@ -10739,12 +12235,31 @@ "url": "https://github.com/sponsors/sindresorhus" } }, + "node_modules/zwitch": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/zwitch/-/zwitch-2.0.4.tgz", + "integrity": "sha512-bXE4cR/kVZhKZX/RjPEflHaKVhUVl85noU3v6b8apfQEc1x4A+zBxjZ4lN8LqGd6WZ3dl98pY4o717VFmoPp+A==", + "funding": { + "type": "github", + "url": "https://github.com/sponsors/wooorm" + } + }, "packages/docs": { "version": "0.1.0", "dependencies": { + "@heroicons/react": "2.1.5", + "@tailwindcss/typography": "0.5.15", + "@types/hast": "3.0.4", + "gray-matter": "4.0.3", "next": "14.2.11", + "next-themes": "0.3.0", "react": "^18", - "react-dom": "^18" + "react-dom": "^18", + "react-markdown": "9.0.1", + "rehype-highlight": "7.0.0", + "rehype-raw": "7.0.0", + "remark": "15.0.1", + "remark-html": "16.0.1" }, "devDependencies": { "@types/node": "^20", @@ -10755,6 +12270,15 @@ "typescript": "^5" } }, + "packages/docs/node_modules/@swc/helpers": { + "version": "0.5.5", + "resolved": "https://registry.npmjs.org/@swc/helpers/-/helpers-0.5.5.tgz", + "integrity": "sha512-KGYxvIOXcceOAbEk4bi/dVLEK9z8sZ0uBB3Il5b1rhfClSpcX0yfRO0KmTkqR2cnQDymwLB+25ZyMzICg/cm/A==", + "dependencies": { + "@swc/counter": "^0.1.3", + "tslib": "^2.4.0" + } + }, "packages/docs/node_modules/next": { "version": "14.2.11", "license": "MIT", @@ -10941,6 +12465,141 @@ "typescript-eslint": "^8.0.1", "vite": "^5.4.1" } + }, + "packages/docs/node_modules/@next/swc-darwin-arm64": { + "version": "14.2.11", + "resolved": "https://registry.npmjs.org/@next/swc-darwin-arm64/-/swc-darwin-arm64-14.2.11.tgz", + "integrity": "sha512-eiY9u7wEJZWp/Pga07Qy3ZmNEfALmmSS1HtsJF3y1QEyaExu7boENz11fWqDmZ3uvcyAxCMhTrA1jfVxITQW8g==", + "cpu": [ + "arm64" + ], + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": ">= 10" + } + }, + "packages/docs/node_modules/@next/swc-darwin-x64": { + "version": "14.2.11", + "resolved": "https://registry.npmjs.org/@next/swc-darwin-x64/-/swc-darwin-x64-14.2.11.tgz", + "integrity": "sha512-lnB0zYCld4yE0IX3ANrVMmtAbziBb7MYekcmR6iE9bujmgERl6+FK+b0MBq0pl304lYe7zO4yxJus9H/Af8jbg==", + "cpu": [ + "x64" + ], + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": ">= 10" + } + }, + "packages/docs/node_modules/@next/swc-linux-arm64-gnu": { + "version": "14.2.11", + "resolved": "https://registry.npmjs.org/@next/swc-linux-arm64-gnu/-/swc-linux-arm64-gnu-14.2.11.tgz", + "integrity": "sha512-Ulo9TZVocYmUAtzvZ7FfldtwUoQY0+9z3BiXZCLSUwU2bp7GqHA7/bqrfsArDlUb2xeGwn3ZuBbKtNK8TR0A8w==", + "cpu": [ + "arm64" + ], + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">= 10" + } + }, + "packages/docs/node_modules/@next/swc-linux-arm64-musl": { + "version": "14.2.11", + "resolved": "https://registry.npmjs.org/@next/swc-linux-arm64-musl/-/swc-linux-arm64-musl-14.2.11.tgz", + "integrity": "sha512-fH377DnKGyUnkWlmUpFF1T90m0dADBfK11dF8sOQkiELF9M+YwDRCGe8ZyDzvQcUd20Rr5U7vpZRrAxKwd3Rzg==", + "cpu": [ + "arm64" + ], + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">= 10" + } + }, + "packages/docs/node_modules/@next/swc-linux-x64-gnu": { + "version": "14.2.11", + "resolved": "https://registry.npmjs.org/@next/swc-linux-x64-gnu/-/swc-linux-x64-gnu-14.2.11.tgz", + "integrity": "sha512-a0TH4ZZp4NS0LgXP/488kgvWelNpwfgGTUCDXVhPGH6pInb7yIYNgM4kmNWOxBFt+TIuOH6Pi9NnGG4XWFUyXQ==", + "cpu": [ + "x64" + ], + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">= 10" + } + }, + "packages/docs/node_modules/@next/swc-linux-x64-musl": { + "version": "14.2.11", + "resolved": "https://registry.npmjs.org/@next/swc-linux-x64-musl/-/swc-linux-x64-musl-14.2.11.tgz", + "integrity": "sha512-DYYZcO4Uir2gZxA4D2JcOAKVs8ZxbOFYPpXSVIgeoQbREbeEHxysVsg3nY4FrQy51e5opxt5mOHl/LzIyZBoKA==", + "cpu": [ + "x64" + ], + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">= 10" + } + }, + "packages/docs/node_modules/@next/swc-win32-arm64-msvc": { + "version": "14.2.11", + "resolved": "https://registry.npmjs.org/@next/swc-win32-arm64-msvc/-/swc-win32-arm64-msvc-14.2.11.tgz", + "integrity": "sha512-PwqHeKG3/kKfPpM6of1B9UJ+Er6ySUy59PeFu0Un0LBzJTRKKAg2V6J60Yqzp99m55mLa+YTbU6xj61ImTv9mg==", + "cpu": [ + "arm64" + ], + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">= 10" + } + }, + "packages/docs/node_modules/@next/swc-win32-ia32-msvc": { + "version": "14.2.11", + "resolved": "https://registry.npmjs.org/@next/swc-win32-ia32-msvc/-/swc-win32-ia32-msvc-14.2.11.tgz", + "integrity": "sha512-0U7PWMnOYIvM74GY6rbH6w7v+vNPDVH1gUhlwHpfInJnNe5LkmUZqhp7FNWeNa5wbVgRcRi1F1cyxp4dmeLLvA==", + "cpu": [ + "ia32" + ], + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">= 10" + } + }, + "packages/docs/node_modules/@next/swc-win32-x64-msvc": { + "version": "14.2.11", + "resolved": "https://registry.npmjs.org/@next/swc-win32-x64-msvc/-/swc-win32-x64-msvc-14.2.11.tgz", + "integrity": "sha512-gQpS7mcgovWoaTG1FbS5/ojF7CGfql1Q0ZLsMrhcsi2Sr9HEqsUZ70MPJyaYBXbk6iEAP7UXMD9HC8KY1qNwvA==", + "cpu": [ + "x64" + ], + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">= 10" + } } } } diff --git a/packages/docs/app/[lang]/guide/[group]/[slug]/guide-page.tsx b/packages/docs/app/[lang]/guide/[group]/[slug]/guide-page.tsx new file mode 100644 index 0000000..c5e2407 --- /dev/null +++ b/packages/docs/app/[lang]/guide/[group]/[slug]/guide-page.tsx @@ -0,0 +1,101 @@ +"use client"; + +import React, { useEffect } from "react"; +import ReactMarkdown from "react-markdown"; +import rehypeRaw from "rehype-raw"; +import rehypeHighlight from "rehype-highlight"; +import { + DynamicLayout, + MobileSidebar, + NavigationButton, + Sidebar, + TOC, +} from "@/components"; +import { generateTOC } from "@/lib/generateTOC"; + +interface GuidePageProps { + params: { + lang: string; + group: string; + slug: string; + }; + content: string; + title: string; + prevDocument: any; + nextDocument: any; + allDocuments: any[]; +} + +export default function GuidePage({ + params, + content, + title, + prevDocument, + nextDocument, + allDocuments, +}: GuidePageProps) { + const { lang, group, slug } = params; + const tocItems = generateTOC(content); + + useEffect(() => { + const headings = document.querySelectorAll("h1, h2, h3, h4, h5, h6"); + headings.forEach((heading) => { + heading.id = + heading.textContent?.toLowerCase().replace(/\s+/g, "-") ?? ""; + }); + }, [content]); + + return ( + + } + mobileSidebar={ + + } + toc={} + > +
+

+ {group} > {title} +

+ + {content} + +
+
+
+ {prevDocument && ( + + )} +
+
+ {nextDocument && ( + + )} +
+
+
+ ); +} diff --git a/packages/docs/app/[lang]/guide/[group]/[slug]/page.tsx b/packages/docs/app/[lang]/guide/[group]/[slug]/page.tsx new file mode 100644 index 0000000..74f647b --- /dev/null +++ b/packages/docs/app/[lang]/guide/[group]/[slug]/page.tsx @@ -0,0 +1,51 @@ +import { getAllDocuments, getDocumentBySlug } from "@/lib/mdx"; +import { Metadata } from "next"; +import GuidePage from "./guide-page"; + +interface GuidePageProps { + params: { + lang: string; + group: string; + slug: string; + }; +} + +// TODO: Add URL and image +export async function generateMetadata({ + params, +}: GuidePageProps): Promise { + const { lang, group, slug } = params; + const { title, content } = await getDocumentBySlug(lang, group, slug); + + return { + title: `${title} | react-notion-custom Docs`, + description: content.slice(0, 160), + openGraph: { + title: `${title} | react-notion-custom Docs`, + description: content.slice(0, 160), + // url: "", + }, + alternates: { + // canonical: "", + }, + }; +} + +export default async function Page({ params }: GuidePageProps) { + const { lang, group, slug } = params; + + const { content, title, prevDocument, nextDocument } = + await getDocumentBySlug(lang, group, slug); + const allDocuments = getAllDocuments(lang); + + return ( + + ); +} diff --git a/packages/docs/app/[lang]/page.tsx b/packages/docs/app/[lang]/page.tsx new file mode 100644 index 0000000..50596d8 --- /dev/null +++ b/packages/docs/app/[lang]/page.tsx @@ -0,0 +1,9 @@ +import MouseImageEffect from "@/components/effects/mouse-image-effect"; + +export default function Home({ + params: { lang }, +}: { + params: { lang: string }; +}) { + return ; +} diff --git a/packages/docs/app/favicon.ico b/packages/docs/app/favicon.ico index 718d6fe..71ca2b0 100644 Binary files a/packages/docs/app/favicon.ico and b/packages/docs/app/favicon.ico differ diff --git a/packages/docs/app/fonts/GeistMonoVF.woff b/packages/docs/app/fonts/GeistMonoVF.woff deleted file mode 100644 index f2ae185..0000000 Binary files a/packages/docs/app/fonts/GeistMonoVF.woff and /dev/null differ diff --git a/packages/docs/app/fonts/GeistVF.woff b/packages/docs/app/fonts/GeistVF.woff deleted file mode 100644 index 1b62daa..0000000 Binary files a/packages/docs/app/fonts/GeistVF.woff and /dev/null differ diff --git a/packages/docs/app/fonts/Pretendard-Bold.woff b/packages/docs/app/fonts/Pretendard-Bold.woff new file mode 100644 index 0000000..7837ae5 Binary files /dev/null and b/packages/docs/app/fonts/Pretendard-Bold.woff differ diff --git a/packages/docs/app/fonts/Pretendard-Regular.woff b/packages/docs/app/fonts/Pretendard-Regular.woff new file mode 100644 index 0000000..e3b3a35 Binary files /dev/null and b/packages/docs/app/fonts/Pretendard-Regular.woff differ diff --git a/packages/docs/app/fonts/Pretendard-SemiBold.woff b/packages/docs/app/fonts/Pretendard-SemiBold.woff new file mode 100644 index 0000000..682e7a4 Binary files /dev/null and b/packages/docs/app/fonts/Pretendard-SemiBold.woff differ diff --git a/packages/docs/app/fonts/Pretendard-Thin.woff b/packages/docs/app/fonts/Pretendard-Thin.woff new file mode 100644 index 0000000..d28e448 Binary files /dev/null and b/packages/docs/app/fonts/Pretendard-Thin.woff differ diff --git a/packages/docs/app/globals.css b/packages/docs/app/globals.css index 13d40b8..caacb7c 100644 --- a/packages/docs/app/globals.css +++ b/packages/docs/app/globals.css @@ -2,26 +2,85 @@ @tailwind components; @tailwind utilities; -:root { - --background: #ffffff; - --foreground: #171717; +/* CSS Reset */ +* { + margin: 0; + padding: 0; + box-sizing: border-box; } -@media (prefers-color-scheme: dark) { - :root { - --background: #0a0a0a; - --foreground: #ededed; - } +html, +body { + height: 100%; +} + +img, +picture, +video, +canvas, +svg { + display: block; + max-width: 100%; +} + +input, +button, +textarea, +select { + font: inherit; +} + +button { + cursor: pointer; +} + +a { + text-decoration: none; + color: inherit; } +ul, +ol { + list-style: none; +} + +/* Global Styles */ body { - color: var(--foreground); - background: var(--background); - font-family: Arial, Helvetica, sans-serif; + line-height: 1.6; + background-color: #f8f9fa; + color: #343a40; } -@layer utilities { - .text-balance { - text-wrap: balance; +h1, +h2, +h3, +h4, +h5, +h6 { + font-weight: bold; + line-height: 1.2; +} + +@keyframes shake { + 0% { + transform: rotate(0deg); + } + 25% { + transform: rotate(5deg); } + 50% { + transform: rotate(0eg); + } + 75% { + transform: rotate(-5deg); + } + 100% { + transform: rotate(0deg); + } +} +.logo-image { + transition: transform 0.1s ease-in-out; +} +.logo-image:hover { + animation: shake 0.5s ease-in-out; } diff --git a/packages/docs/app/layout.tsx b/packages/docs/app/layout.tsx index a36cde0..0724422 100644 --- a/packages/docs/app/layout.tsx +++ b/packages/docs/app/layout.tsx @@ -1,21 +1,28 @@ import type { Metadata } from "next"; import localFont from "next/font/local"; import "./globals.css"; +import { Navigation, ThemeProvider } from "@/components"; -const geistSans = localFont({ - src: "./fonts/GeistVF.woff", - variable: "--font-geist-sans", - weight: "100 900", -}); -const geistMono = localFont({ - src: "./fonts/GeistMonoVF.woff", - variable: "--font-geist-mono", +const pretendard = localFont({ + src: "./fonts/Pretendard-Regular.woff", + variable: "--font-pretendard", weight: "100 900", }); export const metadata: Metadata = { - title: "Create Next App", - description: "Generated by create next app", + title: "react-notion-custom Docs", + description: + "Comprehensive documentation for react-notion-custom library, a powerful tool for rendering Notion pages in React applications.", + keywords: "react, notion, custom, documentation, library, rendering", + openGraph: { + title: "react-notion-custom Documentation", + description: + "Learn how to use react-notion-custom to render Notion pages in your React apps.", + type: "website", + // TODO: Add URL and image + // url: "", + // image: "", + }, }; export default function RootLayout({ @@ -24,11 +31,12 @@ export default function RootLayout({ children: React.ReactNode; }>) { return ( - - - {children} + + + + +
{children}
+
); diff --git a/packages/docs/app/page.tsx b/packages/docs/app/page.tsx deleted file mode 100644 index 433c8aa..0000000 --- a/packages/docs/app/page.tsx +++ /dev/null @@ -1,101 +0,0 @@ -import Image from "next/image"; - -export default function Home() { - return ( -
-
- Next.js logo -
    -
  1. - Get started by editing{" "} - - app/page.tsx - - . -
  2. -
  3. Save and see your changes instantly.
  4. -
- - -
- -
- ); -} diff --git a/packages/docs/components/dynamic-layout.tsx b/packages/docs/components/dynamic-layout.tsx new file mode 100644 index 0000000..d8b8e71 --- /dev/null +++ b/packages/docs/components/dynamic-layout.tsx @@ -0,0 +1,30 @@ +"use client"; + +import useIsMobile from "../hooks/useIsMobile"; + +interface DynamicLayoutProps { + sidebar: React.ReactNode; + toc: React.ReactNode; + mobileSidebar: React.ReactNode; + children: React.ReactNode; +} + +export default function DynamicLayout({ + sidebar, + toc, + mobileSidebar, + children, +}: DynamicLayoutProps) { + const isMobile = useIsMobile(); + + return ( +
+ {!isMobile &&
{sidebar}
} + {isMobile &&
{mobileSidebar}
} +
{children}
+ {!isMobile &&
{toc}
} +
+ ); +} diff --git a/packages/docs/components/effects/mouse-image-effect.tsx b/packages/docs/components/effects/mouse-image-effect.tsx new file mode 100644 index 0000000..2d1e9f9 --- /dev/null +++ b/packages/docs/components/effects/mouse-image-effect.tsx @@ -0,0 +1,71 @@ +"use client"; +import React, { MouseEvent, useEffect, useRef, useState } from "react"; + +const images = Array(16).fill("/logo.svg"); + +export default function MouseImageEffect() { + const [imagePositions, setImagePositions] = useState< + { src: string; x: number; y: number }[] + >([]); + const containerRef = useRef(null); + + useEffect(() => { + images.forEach((src) => { + const img = new Image(); + img.src = src; + }); + }, []); + + const handleMouseMove = (e: MouseEvent) => { + if (!containerRef.current) return; + + const { clientX, clientY } = e; + const { left, top } = containerRef.current.getBoundingClientRect(); + + const x = clientX - left; + const y = clientY - top; + + if (imagePositions.length < images.length) { + setImagePositions((prev) => [ + ...prev, + { src: images[prev.length], x, y }, + ]); + } else { + setImagePositions((prev) => [ + ...prev.slice(1), + { src: images[prev.length % images.length], x, y }, + ]); + } + }; + + return ( +
+
+

+ react-notion-custom +

+

+ Create Your Custom Blog with Notion +

+
+ + {imagePositions.map((image, index) => ( + {`Image + ))} +
+ ); +} diff --git a/packages/docs/components/index.ts b/packages/docs/components/index.ts new file mode 100644 index 0000000..32b6e73 --- /dev/null +++ b/packages/docs/components/index.ts @@ -0,0 +1,9 @@ +export { default as Navigation } from "./navigation"; +export { default as LanguageSelector } from "./language-selector"; +export { default as Sidebar } from "./sidebar"; +export { ThemeProvider } from "./theme-provider"; +export { default as ThemeToggle } from "./theme-toggle"; +export { default as TOC } from "./toc"; +export { default as DynamicLayout } from "./dynamic-layout"; +export { default as NavigationButton } from "./navigation-button"; +export { default as MobileSidebar } from "./mobile-sidebar"; diff --git a/packages/docs/components/language-selector.tsx b/packages/docs/components/language-selector.tsx new file mode 100644 index 0000000..5345b2b --- /dev/null +++ b/packages/docs/components/language-selector.tsx @@ -0,0 +1,83 @@ +"use client"; + +import { useState, useEffect, useRef } from "react"; +import { useRouter, usePathname } from "next/navigation"; +import { GlobeAltIcon } from "@heroicons/react/24/solid"; + +const languages = [ + { code: "en", name: "English" }, + { code: "kr", name: "한국어" }, +]; + +export default function LanguageSelector() { + const router = useRouter(); + const pathname = usePathname(); + const [currentLang, setCurrentLang] = useState("en"); + const [isMenuOpen, setIsMenuOpen] = useState(false); + const menuRef = useRef(null); + + useEffect(() => { + const lang = pathname.split("/")[1]; + if (languages.some((l) => l.code === lang)) { + setCurrentLang(lang); + } + }, [pathname]); + + useEffect(() => { + function handleClickOutside(event: MouseEvent) { + if (menuRef.current && !menuRef.current.contains(event.target as Node)) { + setIsMenuOpen(false); + } + } + + document.addEventListener("mousedown", handleClickOutside); + return () => { + document.removeEventListener("mousedown", handleClickOutside); + }; + }, []); + + const changeLang = (newLang: string) => { + const newPathname = pathname.replace(`/${currentLang}`, `/${newLang}`); + router.push(newPathname); + }; + + return ( +
+ + {isMenuOpen && ( +
+
+ {languages.map((lang) => ( + + ))} +
+
+ )} +
+ ); +} diff --git a/packages/docs/components/mobile-sidebar.tsx b/packages/docs/components/mobile-sidebar.tsx new file mode 100644 index 0000000..e009e05 --- /dev/null +++ b/packages/docs/components/mobile-sidebar.tsx @@ -0,0 +1,103 @@ +"use client"; + +import { useState } from "react"; +import Link from "next/link"; +import { ChevronDownIcon, ChevronRightIcon } from "@heroicons/react/24/solid"; +import ThemeToggle from "./theme-toggle"; + +type Document = { + title: string; + slug: string; + group: string; + order: number; +}; + +type MobileSidebarProps = { + documents: Document[]; + currentSlug: string; + currentGroup: string; + lang: string; +}; + +export default function MobileSidebar({ + documents, + currentSlug, + currentGroup, + lang, +}: MobileSidebarProps) { + const [isOpen, setIsOpen] = useState(false); + const [openGroup, setOpenGroup] = useState(null); + + const groupedDocuments = documents.reduce( + (acc, doc) => { + if (!acc[doc.group]) { + acc[doc.group] = []; + } + acc[doc.group].push(doc); + return acc; + }, + {} as Record, + ); + + const toggleDropdown = () => setIsOpen(!isOpen); + const toggleGroup = (group: string) => { + setOpenGroup(openGroup === group ? null : group); + }; + + return ( +
+ + {isOpen && ( +
+ {Object.entries(groupedDocuments).map(([group, docs]) => ( +
+ + {openGroup === group && ( +
    + {docs.map((doc) => ( +
  • + + {doc.title} + + +
  • + ))} +
+ )} +
+ ))} +
+ +
+
+ )} +
+ ); +} diff --git a/packages/docs/components/navigation-button.tsx b/packages/docs/components/navigation-button.tsx new file mode 100644 index 0000000..c8b5419 --- /dev/null +++ b/packages/docs/components/navigation-button.tsx @@ -0,0 +1,37 @@ +import Link from "next/link"; +import { ChevronRightIcon, ChevronLeftIcon } from "@heroicons/react/24/solid"; + +interface NavigationButtonProps { + document: { + group: string; + slug: string; + title: string; + }; + lang: string; + direction: "prev" | "next"; +} + +export default function NavigationButton({ + document, + lang, + direction, +}: NavigationButtonProps) { + return ( + + {direction === "prev" ? ( +
+ +

{document.title}

+
+ ) : ( +
+

{document.title}

+ +
+ )} + + ); +} diff --git a/packages/docs/components/navigation.tsx b/packages/docs/components/navigation.tsx new file mode 100644 index 0000000..a5c5d31 --- /dev/null +++ b/packages/docs/components/navigation.tsx @@ -0,0 +1,48 @@ +"use client"; + +import Link from "next/link"; +import { useParams } from "next/navigation"; +import Image from "next/image"; +import LanguageSelector from "./language-selector"; + +export default function Navigation() { + const params = useParams(); + const lang = (params.lang as string) || "en"; + + return ( + + ); +} diff --git a/packages/docs/components/sidebar.tsx b/packages/docs/components/sidebar.tsx new file mode 100644 index 0000000..72e7885 --- /dev/null +++ b/packages/docs/components/sidebar.tsx @@ -0,0 +1,68 @@ +import Link from "next/link"; +import ThemeToggle from "./theme-toggle"; +import { ChevronRightIcon } from "@heroicons/react/24/solid"; + +type Document = { + title: string; + slug: string; + group: string; + order: number; +}; + +type SidebarProps = { + documents: Document[]; + currentSlug: string; + currentGroup: string; + lang: string; +}; + +export default function Sidebar({ + documents, + currentSlug, + currentGroup, + lang, +}: SidebarProps) { + const groupedDocuments = documents.reduce( + (acc, doc) => { + if (!acc[doc.group]) { + acc[doc.group] = []; + } + acc[doc.group].push(doc); + return acc; + }, + {} as Record, + ); + + return ( + + ); +} diff --git a/packages/docs/components/theme-provider.tsx b/packages/docs/components/theme-provider.tsx new file mode 100644 index 0000000..a11e499 --- /dev/null +++ b/packages/docs/components/theme-provider.tsx @@ -0,0 +1,8 @@ +"use client"; + +import { ThemeProvider as NextThemesProvider } from "next-themes"; +import { type ThemeProviderProps } from "next-themes/dist/types"; + +export function ThemeProvider({ children, ...props }: ThemeProviderProps) { + return {children}; +} diff --git a/packages/docs/components/theme-toggle.tsx b/packages/docs/components/theme-toggle.tsx new file mode 100644 index 0000000..1cd899e --- /dev/null +++ b/packages/docs/components/theme-toggle.tsx @@ -0,0 +1,29 @@ +"use client"; + +import { useTheme } from "next-themes"; +import { useEffect, useState } from "react"; +import { SunIcon, MoonIcon } from "@heroicons/react/24/solid"; + +export default function ThemeToggle() { + const [mounted, setMounted] = useState(false); + const { theme, setTheme } = useTheme(); + + useEffect(() => setMounted(true), []); + + if (!mounted) return null; + + return ( + + ); +} diff --git a/packages/docs/components/toc.tsx b/packages/docs/components/toc.tsx new file mode 100644 index 0000000..9eeb24a --- /dev/null +++ b/packages/docs/components/toc.tsx @@ -0,0 +1,27 @@ +interface TOCItem { + id: string; + title: string; +} + +interface TOCProps { + items: TOCItem[]; +} + +export default function TOC({ items }: TOCProps) { + return ( + + ); +} diff --git a/packages/docs/constants/group.ts b/packages/docs/constants/group.ts new file mode 100644 index 0000000..fcb1bc1 --- /dev/null +++ b/packages/docs/constants/group.ts @@ -0,0 +1,12 @@ +export const GROUPS = [ + { id: "getting-started", name: "Getting Started", order: "01" }, + { id: "customization-guide", name: "Customization Guide", order: "02" }, + { id: "block-types", name: "Block Types", order: "03" }, +] as const; + +export type GroupId = (typeof GROUPS)[number]["id"]; + +export function getGroupOrder(groupId: GroupId): string { + const group = GROUPS.find((g) => g.id === groupId); + return group ? group.order : "00"; +} diff --git a/packages/docs/content/guide/en/01. getting-started/01. introduction.md b/packages/docs/content/guide/en/01. getting-started/01. introduction.md new file mode 100644 index 0000000..f3d3fdf --- /dev/null +++ b/packages/docs/content/guide/en/01. getting-started/01. introduction.md @@ -0,0 +1,102 @@ +--- +group: "getting-started" +order: 1 +title: "Project Introduction" +slug: "introduction" +description: "An overview of the React-Notion-Custom project, its key features, and potential applications." +--- + +# Introduction to React-Notion-Custom Project + +## What is React-Notion-Custom? + +React-Notion-Custom is a tool that allows you to easily bring Notion's powerful content creation capabilities to your website. This library enables you to render content created in Notion directly on your React-based website. It's a "magic potion" for both developers and content creators - allowing developers to quickly produce results and providing content creators with a convenient management tool. ✨ + +## Key Features and Provided Value + +### 🛠️ React Rendering of Notion Data + +You can easily convert various content from Notion such as text, images, lists, code blocks, etc., into React components. Notion pages are reborn on the web with the following structure: + +```jsx +import { Notion, type NotionPageData } from 'react-notion-custom'; + +interface NotionPageProps { + pageData: NotionPageData; +} + +function NotionPage({ pageData }: NotionPageProps) { + return ( + + + + + + + + ); +} +``` + +In this example, the `NotionPageData` type defines the data structure of a Notion page. This allows you to use TypeScript's type checking feature to write more stable code. + +### 🎨 Freedom of Customization + +You can easily change themes by just tweaking CSS, or partially modify the provided components to customize only the parts you need. You can even create completely new components and combine them with React-Notion-Custom. It offers unlimited possibilities for developers who handle code. + +### 🚀 Rapid Site Construction + +You can easily extract Notion data using the Notion-dump CLI and quickly build static sites through frameworks like Next.js. The process of building a site with just a few commands feels like a website is completed right from the text. + +```bash +npx notion-dump --page YOUR_PAGE_URL --auth YOUR_API_KEY +``` + +When you run notion-dump, Notion data is extracted into the following folder structure: + +``` +my-notion-project/ +├── notion-data/ +│ ├── page-slug-1.json +│ ├── page-slug-2.json +│ └── ... +└── public/ + └── images/ + ├── page-slug-1/ + │ ├── image1.png + │ └── image2.jpg + ├── page-slug-2/ + │ └── image1.png + └── ... +``` + +- `notion-data/`: The content of each Notion page is stored in JSON format. The file name is generated based on the page's slug. +- `public/images/`: Images used in Notion pages are stored and categorized by each page. + +The extracted data can be easily integrated with React-Notion-Custom components, allowing you to quickly convert Notion content into a website. By simply passing the content of the JSON file to React components, the structure and style of the Notion page are reflected directly on the website. + +## Target Users and Their Potential Applications + +### Developers + +Developers are given the freedom to easily render and customize content created in Notion on websites. If you want to build a website quickly or style it as you want beyond the limitations of commercial services, React-Notion-Custom is the answer. + +### Designers and Creators + +You can directly reflect content written in Notion on the web, allowing you to quickly update content while maintaining design consistency. + +### Potential Use Cases + +- **Personal Blog**: Easily write posts in Notion and automatically reflect them on your website. +- **Company Introduction and Recruitment Pages**: Quickly update company profiles or recruitment information using Notion. +- **Portfolio Sites**: Easily update projects and works, making portfolio management much easier. + +## Why Choose React-Notion-Custom + +1. **Developer-Centric Customization**: You can modify even the finest details that can't be touched in other commercial services. The infinite possibilities through code - this is the charm of React-Notion-Custom. + +2. **Freedom of Open Source**: Everything is transparent and can be handled directly by you. If needed, you can modify the code and self-host the project on your own server, eliminating cost burdens. + +3. **Quick Results and High Efficiency**: You can write content in Notion and use it directly on your website with simple settings. The process of content creation and deployment is further simplified. + +React-Notion-Custom is the bridge that turns your Notion pages into amazing websites. No more complicated setups, start right from Notion now! diff --git a/packages/docs/content/guide/en/01. getting-started/02. quick-start.md b/packages/docs/content/guide/en/01. getting-started/02. quick-start.md new file mode 100644 index 0000000..0754f68 --- /dev/null +++ b/packages/docs/content/guide/en/01. getting-started/02. quick-start.md @@ -0,0 +1,216 @@ +--- +group: "getting-started" +order: 2 +title: "Quick Start Guide" +slug: "quick-start" +description: "A step-by-step guide on building a website with Notion content using React-Notion-Custom." +--- + +# React-Notion-Custom Quick Start Guide + +## 2.1. Installation and Setup + +### Installing React-Notion-Custom + +Install React-Notion-Custom and Notion-dump, a CLI tool for extracting Notion data. + +```bash +npm install react-notion-custom notion-dump +``` + +### Setting up a Next.js Project + +If you don't have a Next.js project, create a new one. Next.js provides optimal performance with static site generation and server-side rendering support. + +```bash +npx create-next-app my-notion-blog +cd my-notion-blog +``` + +### Generating a Notion API Key + +1. Go to the Notion developer portal (https://developers.notion.com/). +2. Create a new integration. +3. Securely store the generated API key. + +## 2.2. Extracting Notion Data (notion-dump) + +Use the Notion-dump CLI to extract Notion data in JSON format and prepare it for use on your website. + +### Using the Data Extraction Command + +Extract data using the Notion page URL and API key. + +```bash +npx notion-dump --page YOUR_PAGE_URL --auth YOUR_API_KEY +``` + +### notion-dump Folder Structure and Options + +notion-dump is tailored to the Next.js project structure by default. The extracted data is stored in the following structure: + +``` +my-notion-blog/ +├── notion-data/ +│ ├── page-id-1.json +│ ├── page-id-2.json +│ └── ... +└── public/ + └── images/ + ├── page-id-1/ + │ ├── image1.png + │ └── image2.jpg + ├── page-id-2/ + │ └── image1.png + └── ... +``` + +Main options for notion-dump: + +- `--dir`: Specifies the directory where JSON files will be stored. Default is `./notion-data`. +- `--image-dir`: Specifies the directory where image files will be stored. Default is `./public/images`. + +For example, if you want to change the storage location: + +```bash +npx notion-dump --page YOUR_PAGE_URL --auth YOUR_API_KEY --dir ./data --image-dir ./public/assets/images +``` + +If options are not specified, default values are used, which are optimized for the Next.js project structure. + +## 2.3. Rendering Notion Data in React + +Render the extracted Notion data using React components. This section covers basic page structure and component usage. + +### Basic Rendering Example Code + +A simple example of rendering Notion data as React components. + +```jsx +import { Notion } from "react-notion-custom"; +import pageData from "./notion-data/your-page-id.json"; + +function NotionPage() { + return ( + + + + + + + + ); +} + +export default NotionPage; +``` + +### Explanation + +- The `` component takes Notion data and constructs the page. +- Basic Notion elements like ``, ``, `<Blocks>` can be expressed on the web as they are. + +## 2.4. Creating the First Page with Next.js Integration + +Generate dynamic web pages using Next.js and optimize performance using Static Site Generation (SSG). + +### Creating Next.js Pages and Setting Up Dynamic Routing + +Generate each post page with dynamic routing and use `getStaticPaths` and `getStaticProps` for static site generation. + +```jsx +// pages/[slug].js +import { Notion } from "react-notion-custom"; +import posts from "../posts"; + +export async function getStaticPaths() { + const paths = posts.map((post) => ({ + params: { slug: post.slug }, + })); + return { paths, fallback: false }; +} + +export async function getStaticProps({ params }) { + const post = posts.find((p) => p.slug === params.slug); + return { props: { post } }; +} + +export default function PostPage({ post }) { + return ( + <Notion> + <Notion.Blocks blocks={post.content.blocks} /> + </Notion> + ); +} +``` + +### Explanation + +- `getStaticPaths` pre-generates the paths for the posts. +- `getStaticProps` fetches data for each post and renders it. + +## 2.5. Creating a Blog + +This section explains how to set up the overall structure of the blog and how to manage and display each post. + +### Setting up posts/index.js + +Centrally manage blog post data. Organize the metadata and content of each post into an array for use. + +```javascript +// posts/index.js +import post1Content from "../notion-data/post1.json"; +import post2Content from "../notion-data/post2.json"; + +const posts = [ + { + slug: "post1", + title: "First Blog Post", + date: "2023-10-01", + description: "Description of the first blog post.", + content: post1Content, + }, + { + slug: "post2", + title: "Second Blog Post", + date: "2023-10-05", + description: "Description of the second blog post.", + content: post2Content, + }, +]; + +export default posts; +``` + +### Using the Data: Blog Main Page + +Use the `posts` array to render a list of posts on the main page. + +```jsx +// pages/index.js +import Link from "next/link"; +import posts from "../posts"; + +export default function Home() { + return ( + <div> + <h1>Blog Post List</h1> + <ul> + {posts.map((post) => ( + <li key={post.slug}> + <Link href={`/${post.slug}`}> + <a>{post.title}</a> + </Link> + <p>{post.description}</p> + <small>{post.date}</small> + </li> + ))} + </ul> + </div> + ); +} +``` + +### Creating Individual Post Pages + +Each post page is automatically generated through dynamic routing. Posts written in Notion are directly reflected on the website, allowing for easy blog updates. diff --git a/packages/docs/content/guide/en/02. customization-guide/01. structure-and-customization.md b/packages/docs/content/guide/en/02. customization-guide/01. structure-and-customization.md new file mode 100644 index 0000000..268206b --- /dev/null +++ b/packages/docs/content/guide/en/02. customization-guide/01. structure-and-customization.md @@ -0,0 +1,175 @@ +--- +group: "customization-guide" +order: 1 +title: "Understanding React-Notion-Custom Structure and Customization" +slug: "structure-and-customization" +description: "Learn about the data structure of React-Notion-Custom and effective customization methods based on this understanding." +--- + +# Understanding React-Notion-Custom Structure and Customization + +To effectively customize React-Notion-Custom, it's essential to first understand its basic structure and data flow. This section examines the structure of Notion data and explains how to create custom components based on this understanding. + +## Understanding the Data Structure + +The Notion data used by React-Notion-Custom is structured in JSON format and includes page information and detailed content for each block. Understanding this structure is the first step towards effective customization. + +### Basic Data Structure + +```json +{ + "object": "page", + "id": "page-id", + "created_time": "2024-09-20T13:54:00.000Z", + "last_edited_time": "2024-09-21T04:28:00.000Z", + "cover": null, + "icon": null, + "parent": { + "type": "page_id", + "page_id": "parent-page-id" + }, + "properties": { + "title": { + "id": "title", + "type": "title", + "title": [ + { + "type": "text", + "text": { + "content": "Page Title", + "link": null + }, + "annotations": { + "bold": false, + "italic": false, + "color": "default" + }, + "plain_text": "Page Title" + } + ] + } + }, + "blocks": [ + { + "object": "block", + "id": "block-id-1", + "type": "toggle", + "toggle": { + "rich_text": [ + { + "type": "text", + "text": { + "content": "Toggle content", + "link": null + }, + "annotations": { + "bold": false, + "italic": false, + "color": "default" + }, + "plain_text": "Toggle content" + } + ], + "color": "default" + }, + "blocks": [ + { + "object": "block", + "id": "block-id-2", + "type": "paragraph", + "paragraph": { + "rich_text": [ + { + "type": "text", + "text": { + "content": "Paragraph inside toggle", + "link": null + }, + "annotations": { + "bold": false, + "italic": false, + "color": "default" + }, + "plain_text": "Paragraph inside toggle" + } + ], + "color": "default" + } + } + ] + } + ] +} +``` + +### Key Structural Elements + +1. **Page Metadata**: Includes basic page information such as `id`, `created_time`, `last_edited_time`, `parent`, etc. + +2. **Cover and Icon**: Information about the page's cover image and icon. This can be used when applying custom designs. + +3. **Properties**: Page property information, mainly including the page title (`title`). This information can be used when creating custom headers. + +4. **Blocks**: An array of blocks that make up the page content. Each block has the following structure: + + - `object`: Always "block" + - `id`: Unique identifier for the block + - `type`: Block type (e.g., "toggle", "paragraph", "heading_1", etc.) + - `[type]`: A key corresponding to the block type, containing specific content for that block. + - `blocks`: An array of child blocks. This forms a recursive structure. + +5. **Recursive Structure**: Each block within the `blocks` array can have its own `blocks` array, allowing for nested structures. This is important when customizing nested content such as toggles and lists. + +6. **rich_text**: Most text content is represented as a `rich_text` array, including text style information (`annotations`). This allows for fine-grained customization of text styles. + +## Component Structure for Customization + +The basic component structure of React-Notion-Custom is as follows: + +<pre style="font-family: monospace; line-height: 1.2; white-space: pre;"> +┌────────────────────────────────────────────────────────────────┐ +│ <Notion> │ +│ ┌────────────────────────────────────────────────────────┐ │ +│ │ <Cover /> │ │ +│ │ // Page cover image │ │ +│ └────────────────────────────────────────────────────────┘ │ +│ │ +│ ┌────────────────────────────┐ │ +│ │ <Body> │ │ +│ │ ┌──────────────────────┐ │ │ +│ │ │ <Title /> │ │ │ +│ │ │ // Page title │ │ │ +│ │ └──────────────────────┘ │ │ +│ │ ┌──────────────────────┐ │ │ +│ │ │ <Blocks> │ │ │ +│ │ │ ┌──────────────────┐ │ │ │ +│ │ │ │<Block type="...">│ │ │ │ +│ │ │ │ // e.g. Heading │ │ │ │ +│ │ │ └──────────────────┘ │ │ │ +│ │ │ ┌──────────────────┐ │ │ │ +│ │ │ │<Block type="...">│ │ │ │ +│ │ │ │ // e.g. Para │ │ │ │ +│ │ │ └──────────────────┘ │ │ │ +│ │ │ ... │ │ │ +│ │ │ // More blocks │ │ │ +│ │ └──────────────────────┘ │ │ +│ └────────────────────────────┘ │ +│ │ +└────────────────────────────────────────────────────────────────┘ +</pre> + +In this structure, the `<Body>` component occupies about 60% of the total width and is centered. You can customize each component based on this layout. + +## Customization Strategies + +1. **Custom Components for Each Block**: Create custom components for each block type (`paragraph`, `heading_1`, `toggle`, etc.) to override default styles and behaviors. + +2. **Style Customization**: Use CSS variables and classes to adjust the overall design. + +3. **Layout Modification**: Adjust the width or alignment of the `<Body>` component to change the overall layout. + +4. **Utilize Recursive Rendering**: Implement complex layouts or interactions using the nested block structure. + +5. **Leverage Metadata**: Use page metadata, cover images, icons, etc. to create rich UIs. + +Understanding these structures and strategies allows you to create highly customized Notion-based websites using React-Notion-Custom. The next section will explore specific customization methods for each element. diff --git a/packages/docs/content/guide/en/02. customization-guide/02. custom-component.md b/packages/docs/content/guide/en/02. customization-guide/02. custom-component.md new file mode 100644 index 0000000..998072e --- /dev/null +++ b/packages/docs/content/guide/en/02. customization-guide/02. custom-component.md @@ -0,0 +1,215 @@ +--- +group: "customization-guide" +order: 2 +title: "Custom Component Usage" +slug: "custom-component-usage" +description: "Learn how to customize and effectively use components in React-Notion-Custom." +--- + +# Custom Component Usage + +React-Notion-Custom provides default components for various Notion block types. This section explains how to use and customize these default components, as well as how to create new custom components. + +## Examples of Using Basic Components + +### Heading Component + +The Heading component renders Notion's title block. Here's an example of customizing the Heading component: + +```tsx +import React from "react"; +import { Heading, type HeadingArgs } from "react-notion-custom"; + +const CustomHeading: React.FC<HeadingArgs> = (props) => { + // Add underline to level 3 headings + if (props.type === "heading_3") { + return <Heading {...props} style={{ textDecoration: "underline" }} />; + } + return <Heading {...props} />; +}; + +export default CustomHeading; +``` + +In this example, we check the `level` property to change the style for a specific heading level. + +### Paragraph Component + +The Paragraph component renders regular text blocks. Here's an example of customizing the Paragraph component: + +```tsx +import React from "react"; +import { Paragraph, type ParagraphArgs } from "react-notion-custom"; + +const CustomParagraph: React.FC<ParagraphArgs> = (props) => { + return ( + <Paragraph + {...props} + className="custom-paragraph" + style={{ lineHeight: "1.8", marginBottom: "1em" }} + /> + ); +}; + +export default CustomParagraph; +``` + +In this example, we adjust the line spacing and bottom margin for all paragraphs and add a custom class. + +## Example of Customizing the Toggle Component + +The Toggle component implements Notion's toggle block. Here's an advanced example of customizing the Toggle component: + +```tsx +import React, { useState } from "react"; +import { Toggle, type ToggleArgs } from "react-notion-custom"; +import { ChevronDown, ChevronRight } from "your-icon-library"; + +const CustomToggle: React.FC<ToggleArgs> = (props) => { + const [isOpen, setIsOpen] = useState(false); + + const handleChangeOpen = (open: boolean) => { + setIsOpen(open); + // You can add additional logic here + console.log(`Toggle state changed to: ${open}`); + }; + + return ( + <Toggle {...props} isOpen={isOpen} onChangeOpen={handleChangeOpen}> + <Toggle.Icon> + {isOpen ? <ChevronDown size={20} /> : <ChevronRight size={20} />} + </Toggle.Icon> + <Toggle.Content>{props.children}</Toggle.Content> + </Toggle> + ); +}; + +export default CustomToggle; +``` + +In this example: + +1. We use `useState` to manage the open/closed state of the toggle. +2. The `handleChangeOpen` function handles state changes and can implement additional logic as needed. +3. We pass `isOpen` and `onChangeOpen` props to the `Toggle` component to directly control its state. +4. We display different icons based on the `isOpen` state. + +If no content is provided to `<Toggle.Icon>`, the default icon will be used. + +## Applying Custom Components + +Here's how to use custom components in React-Notion-Custom: + +```jsx +import { Notion } from "react-notion-custom"; +import CustomHeading from "./CustomHeading"; +import CustomParagraph from "./CustomParagraph"; +import CustomToggle from "./CustomToggle"; + +const MyNotionPage = ({ pageData }) => { + const customComponents = { + heading_1: CustomHeading, + heading_2: CustomHeading, + heading_3: CustomHeading, + paragraph: CustomParagraph, + toggle: CustomToggle, + }; + + return ( + <Notion custom={customComponents}> + <Notion.Body> + <Notion.Blocks blocks={pageData.blocks} /> + </Notion.Body> + </Notion> + ); +}; +``` + +In this example, we pass custom components to the Notion component through the `custom` prop. You can specify a custom component for each block type. + +## Creating a Completely Custom Component + +You can also create entirely new components without using React-Notion-Custom's basic components. Here's an example of creating a Todo component from scratch: + +```tsx +import React from "react"; +import { RichText, type TodoProps } from "react-notion-custom"; + +const Todo: React.FC<TodoProps> = ({ children, ...props }) => { + const { + to_do: { color, rich_text: texts, checked }, + } = props; + + const colorClass = color !== "default" ? `notion-${color}` : ""; + + return ( + <div className={`notion-block notion-to-do ${colorClass}`}> + <div className="notion-to-do-item"> + <div className="notion-to-do-item-content"> + <input + type="checkbox" + checked={checked} + readOnly + className="notion-to-do-checkbox" + /> + <div + className={`notion-to-do-text ${checked ? "notion-to-do-checked" : ""}`} + > + <RichText props={texts} /> + </div> + </div> + {children} + </div> + </div> + ); +}; + +export default Todo; +``` + +Points to note in this example: + +- We only import the RichText component from react-notion-custom. This is to handle complex text rendering. +- All CSS class names start with notion-. This follows React-Notion-Custom's styling rules. +- We add a class in the format `notion-${color}`. For the default color ("default"), we don't apply an additional class. +- The component accepts a children prop to render nested content. This is to support nested structures. + +## Precautions When Creating Components + +1. **Type Safety**: Use TypeScript to clearly define prop types. Utilize the types provided by React-Notion-Custom (e.g., `HeadingArgs`, `ParagraphArgs`). +2. **Prop Passing**: Use `{...props}` to pass all props from custom components to basic components. This maintains type safety while preventing unnecessary prop listing. +3. **Style Overriding**: Be careful not to completely replace existing styles. Instead, add `className` or extend the `style` prop. +4. **Performance Consideration**: Use `React.memo`, `useMemo`, or `useCallback` appropriately to prevent unnecessary re-rendering. +5. **Accessibility**: Consider accessibility (a11y) when creating custom components. For example, it's good to add appropriate `aria-` attributes to toggle buttons. +6. **Theme Support**: Design custom components to be compatible with React-Notion-Custom's theme system. Using CSS variables makes it easy to support theme switching, such as dark mode. + +## Applying Custom Components + +Here's how to use custom components in React-Notion-Custom: + +```jsx +import { Notion, type NotionPage, type CustomComponents } from 'react-notion-custom'; +import CustomParagraph from './CustomParagraph'; +import CustomToggle from './CustomToggle'; + +const MyNotionPage = ({ pageData }: { pageData: NotionPage }) => { + const customComponents = useMemo<CustomComponents>(() => ({ + paragraph: CustomParagraph, + toggle: CustomToggle, + }), []); + + return ( + <Notion custom={customComponents}> + <Notion.Cover cover={pageData.cover} /> + <Notion.Body> + <Notion.Title title={pageData.title} /> + <Notion.Blocks blocks={pageData.blocks} /> + </Notion.Body> + </Notion> + ); +}; +``` + +In this example, we pass custom components to the Notion component through the `custom` prop. + +By following these guidelines, you can extend the functionality of React-Notion-Custom while maintaining consistency and maintainability. diff --git a/packages/docs/content/guide/en/02. customization-guide/03. custom-style.md b/packages/docs/content/guide/en/02. customization-guide/03. custom-style.md new file mode 100644 index 0000000..7720288 --- /dev/null +++ b/packages/docs/content/guide/en/02. customization-guide/03. custom-style.md @@ -0,0 +1,168 @@ +--- +group: "customization-guide" +order: 3 +title: "CSS Structure and Styling Guide" +slug: "css-structure-and-styling" +description: "Understand the CSS structure of React-Notion-Custom and learn effective styling methods." +--- + +# CSS Structure and Styling Guide + +React-Notion-Custom provides a flexible and extensible CSS structure. This section explains the CSS variable system, key styling points, and the principles of indent application. + +## CSS Variable System + +React-Notion-Custom uses CSS variables to define global styles. This allows for easy theme changes and overall style adjustments. + +The main CSS variables are as follows: + +```css +.notion { + --notion-font: ui-sans-serif, system-ui, -apple-system, BlinkMacSystemFont, + "Segoe UI", Helvetica, "Apple Color Emoji", Arial, sans-serif, + "Segoe UI Emoji", "Segoe UI Symbol"; + --notion-max-width: 720px; + --notion-header-height: 45px; + --notion-indent: 27px; + + /* Color variables */ + --fg-color: rgb(55, 53, 47); + --fg-color-0: rgba(55, 53, 47, 0.09); + --fg-color-1: rgba(55, 53, 47, 0.16); + --fg-color-2: rgba(55, 53, 47, 0.4); + --fg-color-3: rgba(55, 53, 47, 0.6); + --fg-color-4: #000; + --fg-color-5: rgb(233, 233, 231); + --fg-color-6: rgba(55, 53, 47, 0.8); + --fg-color-icon: var(--fg-color); + + --bg-color: #fff; + --bg-color-0: rgba(135, 131, 120, 0.15); + --bg-color-1: rgb(247, 246, 243); + --bg-color-2: rgba(135, 131, 120, 0.15); + + --select-color-0: rgb(46, 170, 220); + --select-color-1: rgba(35, 131, 226, 0.28); + --select-color-2: #d9eff8; +} +``` + +You can easily change the overall style by redefining these variables. + +## Key Styling Points + +1. **Class Naming Convention**: All classes start with `notion-`. This prevents style conflicts. + + Example: `.notion-block`, `.notion-h1`, `.notion-toggle` + +2. **Block Level Styles**: The `notion-block` class is applied to all block-level elements. + + ```css + .notion-block { + display: block; + } + ``` + +3. **Color Application**: Colors are applied through CSS variables. + + ```css + .notion-text { + color: var(--fg-color); + } + ``` + +4. **Responsive Design**: The `--notion-max-width` variable is used to implement responsive layouts. + + ```css + .notion-body { + width: 100%; + max-width: var(--notion-max-width); + margin: 0 auto; + } + ``` + +## Indent Application Principle + +React-Notion-Custom implements indentation through a recursive structure. This is handled in CSS as follows: + +```css +.notion-block > .notion-block { + margin-left: var(--notion-indent); +} + +.notion-block > .notion-display-contents > .notion-block { + margin-left: var(--notion-indent); +} +``` + +This CSS rule automatically applies indentation to nested blocks. You can easily adjust the degree of indentation by modifying the `--notion-indent` variable. + +## Dark Mode and Custom Themes + +React-Notion-Custom supports dark mode by default. Dark mode is implemented as follows: + +```css +[data-theme="dark"] .notion { + --fg-color: rgba(255, 255, 255, 0.9); + --bg-color: #2f3437; + /* Other dark mode related variables... */ +} +``` + +Also, you can easily apply custom themes. You can apply the desired theme using the `data-theme` attribute: + +```jsx +<Notion data-theme="custom">{/* Notion content */}</Notion> +``` + +And you can define styles in CSS as follows: + +```css +.notion[data-theme="custom"] { + --fg-color: #333; + --bg-color: #f4f4f4; + /* Other custom theme related variables... */ +} +``` + +Using this method, you can easily override default styles by utilizing selector priority. + +## Style Customization Examples + +1. **Changing Global Font**: + + ```css + .notion[data-theme="custom"] { + --notion-font: "Roboto", sans-serif; + } + ``` + +2. **Changing Heading Styles**: + + ```css + .notion[data-theme="custom"] .notion-h1 { + font-size: 2.5em; + color: var(--select-color-0); + border-bottom: 2px solid var(--fg-color-1); + } + ``` + +3. **Adjusting Indentation**: + + ```css + .notion[data-theme="custom"] { + --notion-indent: 20px; + } + ``` + +4. **Implementing Dark Mode**: + + ```css + [data-theme="dark"] .notion[data-theme="custom"] { + --fg-color: rgba(255, 255, 255, 0.9); + --bg-color: #2f3437; + /* Adjust other color variables appropriately */ + } + ``` + +By following this CSS structure and styling guide, you can implement consistent and easily maintainable styles using React-Notion-Custom. Furthermore, detailed customization is possible as needed. diff --git a/packages/docs/content/guide/en/03. block-types/01. block-type-guide-intro.md b/packages/docs/content/guide/en/03. block-types/01. block-type-guide-intro.md new file mode 100644 index 0000000..811c71e --- /dev/null +++ b/packages/docs/content/guide/en/03. block-types/01. block-type-guide-intro.md @@ -0,0 +1,242 @@ +--- +group: "block-types" +order: 1 +title: "Introduction to Block Type Guide" +slug: "block-type-guide-intro" +description: "This guide introduces an overview of block types supported in React-Notion-Custom, their usage methods, customization options, and internal working principles." +--- + +# Introduction to Block Type Guide + +React-Notion-Custom supports various block types from Notion, providing components that closely resemble Notion's style for each block type. This guide explains the currently supported block types, the Props for each block type, customization methods, and internal working principles. + +## Supported Block Types + +Here's a list of block types currently supported and planned for support in React-Notion-Custom: + +| Block Type | Support Status | Block Type Enum | +| ------------------------ | -------------- | ---------------------- | +| Paragraph | ✅ Yes | `paragraph` | +| Heading 1 | ✅ Yes | `heading_1` | +| Heading 2 | ✅ Yes | `heading_2` | +| Heading 3 | ✅ Yes | `heading_3` | +| Bulleted List Item | ✅ Yes | `bulleted_list_item` | +| Numbered List Item | ✅ Yes | `numbered_list_item` | +| To-do | ❌ No | `to_do` | +| Toggle | ✅ Yes | `toggle` | +| Quote | ✅ Yes | `quote` | +| Callout | ✅ Yes | `callout` | +| Equation | ❌ No | `equation` | +| Code | ❌ No | `code` | +| Image | ❌ No | `image` | +| Video | ❌ No | `video` | +| Bookmark | ❌ No | `bookmark` | +| Divider | ✅ Yes | `divider` | +| Table | ❌ No | `table` | +| Table Row | ❌ No | `table_row` | +| Column | ❌ No | `column` | +| Column List | ❌ No | `column_list` | +| Audio | ❌ No | `audio` | +| Synced Block | ❌ No | `synced_block` | +| Table Of Contents | ❌ No | `table_of_contents` | +| Embed | ❌ No | `embed` | +| Figma | ❌ No | `figma` | +| Google Maps | ❌ No | `maps` | +| Google Drive | ❌ No | `drive` | +| Tweet | ❌ No | `tweet` | +| PDF | ❌ No | `pdf` | +| File | ❌ No | `file` | +| Link | ❌ No | `text` (inline) | +| Page Link | ❌ No | `page` | +| External Page Link | ❌ No | `text` (inline) | +| Collections | ❌ No | - | +| Collection View | ❌ No | `collection_view` | +| Collection View Table | ❌ No | `collection_view` | +| Collection View Gallery | ❌ No | `collection_view` | +| Collection View Board | ❌ No | `collection_view` | +| Collection View List | ❌ No | `collection_view` | +| Collection View Calendar | ❌ No | `collection_view` | +| Collection View Page | ❌ No | `collection_view_page` | + +This list will be continuously updated, and we plan to support more block types in the future. + +## Using Props + +React-Notion-Custom provides Props types for each block type. These Props types can be imported directly from the library for use. + +For example, the Props for the Paragraph component can be imported and used as follows: + +```typescript +import { ParagraphProps } from "react-notion-custom"; + +// Structure of ParagraphProps +interface ParagraphProps { + type: "paragraph"; + paragraph: { + rich_text: Array<RichTextItemResponse>; + color: string; + }; + id: string; + has_children?: boolean; +} + +// Usage example +const MyParagraph: React.FC<ParagraphProps> = (props) => { + // Custom implementation +}; +``` + +Detailed structures of Props for each block type can be found in the respective block type documentation. + +## Applying Custom Components + +One of the major advantages of React-Notion-Custom is the ability to completely replace each block type with custom components. Here's how to apply custom components: + +```jsx +import { Notion } from "react-notion-custom"; +import CustomParagraph from "./CustomParagraph"; +import CustomHeading from "./CustomHeading"; + +const customComponents = { + paragraph: CustomParagraph, + heading_1: CustomHeading, + heading_2: CustomHeading, + heading_3: CustomHeading, + // ... custom components for other block types +}; + +function MyNotionPage({ blocks }) { + return ( + <Notion custom={customComponents}> + <Notion.Blocks blocks={blocks} /> + </Notion> + ); +} +``` + +The type of `components` injected into the `Notion` component is as follows: + +```typescript +type NotionComponents = Record<string, React.ComponentType<any>>; +``` + +Here, `key` is the type of Notion block (e.g., 'paragraph', 'heading_1', etc.), and `value` is the React component that will render that type. + +## Data Preprocessing + +React-Notion-Custom preprocesses block data fetched from the Notion API into a form optimized for UI rendering. In this process, it defines and uses a new type called `ContextedBlock`. + +``` ++----------------------+ +| Notion API | +| Block Data | +| +------------------+ | +| | Block | | +| | - id | | +| | - type | | +| | - has_children | | +| | - [type_specific]| | +| +------------------+ | ++----------------------+ + | + | (Input) + v ++---------------------------+ +| resolveToContextedBlocks | +| +-------------------------+ +| | For each block: | +| | +---------------------+ | +| | |resolveToContextedBlo| | +| | |ck (recursive) | | +| | +---------------------+ | +| | | | +| | v | +| | Map parent-child | +| | relationships | +| +-------------------------+ +| | +| v +| Map sibling relationships | ++---------------------------+ + | + | (Output) + v ++---------------------------+ +| ContextedBlocks | +| +-------------------------+ +| | ContextedBlock | +| | - ...Block properties | +| | - context: | +| | - previous (sibling) | +| | - next (sibling) | +| | - parent | +| | - blocks (children) | +| +-------------------------+ ++---------------------------+ + | + | (Used by) + v + +----------------+ + | Notion.Blocks | + | Component | + +----------------+ +``` + +`ContextedBlock` extends the existing `Block` type to include the following additional information: + +- Previous block (sibling relationship) +- Next block (sibling relationship) +- Parent block + +This additional information can be useful in specific block components. + +## Rendering Process + +The rendering process of React-Notion-Custom is as follows: + +``` ++-------------------+ +------------------+ +| Custom | | Default | +| Components | | Components | ++-------------------+ +------------------+ + | | + | +-----------------+ | + +-->| Notion |<--+ + | Component | + |(React.Provider) | + +-----------------+ + | | + +----------+ +------------+ + | | ++----------------+ +----------------+ +| Notion.Cover | | Notion.Body | ++----------------+ +----------------+ + | + +----------------+ + | Notion.Title | + +----------------+ + | + +----------------+ + | Notion.Blocks | + +----------------+ + | + v + +----------------+ + | Rendered | + | Notion Page | + +----------------+ +``` + +Through this process, custom components and default components work together to render the Notion page. + +## Styling + +The currently provided Notion components are implemented to closely resemble Notion's style. Each component is styled using CSS classes with the `notion-` prefix. + +## Future Plans + +- Add support for more block types +- Develop component sets with applied themes +- Expand advanced customization options + +In the following pages, you can find detailed descriptions, Props structures, usage examples, and customization options for each block type. diff --git a/packages/docs/content/guide/en/03. block-types/02. heading.md b/packages/docs/content/guide/en/03. block-types/02. heading.md new file mode 100644 index 0000000..0b7c5cc --- /dev/null +++ b/packages/docs/content/guide/en/03. block-types/02. heading.md @@ -0,0 +1,124 @@ +--- +group: "block-types" +order: 3 +title: "Heading Block" +slug: "notion-block-heading" +description: "Detailed explanation of the Heading block type in React-Notion-Custom." +--- + +# Heading Block + +Heading blocks are used to represent the structure of a document and support three levels (Heading 1, Heading 2, Heading 3). + +## Data Structure + +The Notion API data structure for a Heading block is as follows (example for Heading 1): + +```json +{ + "type": "heading_1", + "heading_1": { + "rich_text": [ + { + "type": "text", + "text": { + "content": "This is a heading", + "link": null + }, + "annotations": { + "bold": false, + "italic": false, + "strikethrough": false, + "underline": false, + "code": false, + "color": "default" + }, + "plain_text": "This is a heading", + "href": null + } + ], + "color": "default", + "is_toggleable": false + } +} +``` + +- `type`: Can be "heading_1", "heading_2", or "heading_3". +- `rich_text`: An array containing the content and style information of the heading text. +- `color`: Specifies the color of the heading. +- `is_toggleable`: Indicates whether the toggle feature is used. + +## React Component + +The component that renders the Heading block in React-Notion-Custom is as follows: + +```jsx +import React, { useMemo } from "react"; +import type { HeadingsProps, HeadingConfig } from "../types"; +import { getColorCss } from "../utils"; +import RichText from "./internal/rich-text"; + +const Headings: React.FC<HeadingsProps> = ({ children, type, ...props }) => { + const { + [type]: { color, rich_text: texts, is_toggleable }, + } = props; + + const { headingTag: HeadingTag, headingClassName } = + useMemo<HeadingConfig>(() => { + switch (type) { + case "heading_2": + return { headingTag: "h2", headingClassName: "notion-h2" }; + case "heading_3": + return { headingTag: "h3", headingClassName: "notion-h3" }; + default: + return { headingTag: "h1", headingClassName: "notion-h1" }; + } + }, [type]); + + return ( + <div + className={`notion-block ${headingClassName} ${getColorCss(color)}`} + > + <HeadingTag className={`notion-h-content ${headingClassName}-content`}> + <RichText props={texts} /> + </HeadingTag> + {children} + </div> + ); +}; + +export default Headings; +``` + +## Usage Example + +Here's an example of how to use the Heading block: + +```jsx +import { Notion } from "react-notion-custom"; + +function MyNotionPage({ blocks }) { + return ( + <Notion> + <Notion.Blocks blocks={blocks} /> + </Notion> + ); +} +``` + +Here, `blocks` is an array of block data received from the Notion API. + +## Styling + +The style of the Heading block can be customized through the following CSS classes: + +- `.notion-block`: Basic style applied to all Notion blocks +- `.notion-h1`, `.notion-h2`, `.notion-h3`: Specific styles for each heading level +- `.notion-h1-content`, `.notion-h2-content`, `.notion-h3-content`: Styles for the content of each heading + +If additional styling is needed, you can write CSS targeting these classes. + +## Notes + +- As Heading blocks are important elements representing the structure of a document, it's recommended to use appropriate heading levels. +- If the `is_toggleable` property is true, the heading will have a toggle functionality. In this case, additional logic may be required. diff --git a/packages/docs/content/guide/en/03. block-types/03. bulleted-list-item.md b/packages/docs/content/guide/en/03. block-types/03. bulleted-list-item.md new file mode 100644 index 0000000..6df31b4 --- /dev/null +++ b/packages/docs/content/guide/en/03. block-types/03. bulleted-list-item.md @@ -0,0 +1,134 @@ +--- +group: "block-types" +order: 4 +title: "Bulleted List Item Block" +slug: "notion-block-bulleted-list-item" +description: "Detailed explanation of the Bulleted List Item block type in React-Notion-Custom." +--- + +# Bulleted List Item Block + +The Bulleted List Item block is used to create unordered lists. Each item starts with a bullet point. + +## Data Structure + +The Notion API data structure for a Bulleted List Item block is as follows: + +```json +{ + "type": "bulleted_list_item", + "bulleted_list_item": { + "rich_text": [ + { + "type": "text", + "text": { + "content": "This is a bulleted list item", + "link": null + }, + "annotations": { + "bold": false, + "italic": false, + "strikethrough": false, + "underline": false, + "code": false, + "color": "default" + }, + "plain_text": "This is a bulleted list item", + "href": null + } + ], + "color": "default" + } +} +``` + +- `rich_text`: An array containing the text content and style information of the list item. +- `color`: Specifies the color of the text. + +## React Component + +The component that renders the Bulleted List Item block in React-Notion-Custom is as follows: + +```jsx +import React from "react"; +import type { BulletedListItemArgs } from "../types"; +import { bulletedListItemMarker, getColorCss } from "../utils"; +import RichText from "./internal/rich-text"; + +type BulletedListItemProps = { + children?: React.ReactNode; +} & BulletedListItemArgs; + +const BulletedListItem: React.FC<BulletedListItemProps> = ({ + children, + ...props +}) => { + const { + bulleted_list_item: { rich_text: texts, color }, + } = props; + const { marker, format } = bulletedListItemMarker.getMarker(props); + + return ( + <ul + data-notion-marker-format={format} + className={`notion-block notion-list-bulleted ${getColorCss(color)}`} + > + <li className="notion-display-contents"> + <div className="notion-list-bulleted-content"> + <span + data-notion-marker-format={format} + className="notion-list-marker" + > + {marker} + </span> + <p> + <RichText props={texts} /> + </p> + </div> + {children} + </li> + </ul> + ); +}; + +export default BulletedListItem; +``` + +## Usage Example + +Here's an example of how to use the Bulleted List Item block: + +```jsx +import { Notion } from "react-notion-custom"; + +function MyNotionPage({ blocks }) { + return ( + <Notion> + <Notion.Blocks blocks={blocks} /> + </Notion> + ); +} +``` + +Here, `blocks` is an array of block data received from the Notion API. + +## Styling + +The style of the Bulleted List Item block can be customized through the following CSS classes: + +- `.notion-block`: Basic style applied to all Notion blocks +- `.notion-list-bulleted`: Specific style for Bulleted List Item blocks +- `.notion-list-bulleted-content`: Style for the content of list items +- `.notion-list-marker`: Style for bullet points + +If additional styling is needed, you can write CSS targeting these classes. + +## Handling Nested Lists + +Bulleted List Items can be nested. Nested lists are handled through the `children` prop. The shape of the bullet point may change depending on the nesting level, which is handled by the `bulletedListItemMarker.getMarker` function. + +## Notes + +- When Bulleted List Item blocks are used consecutively, they are automatically grouped into a single list. +- When using nested lists, it's good to clearly express the hierarchical structure through appropriate indentation. +- The shape of the bullet points follows Notion's default style, but can be customized through CSS. diff --git a/packages/docs/content/guide/en/03. block-types/05. numbered-list-item.md b/packages/docs/content/guide/en/03. block-types/05. numbered-list-item.md new file mode 100644 index 0000000..82684bb --- /dev/null +++ b/packages/docs/content/guide/en/03. block-types/05. numbered-list-item.md @@ -0,0 +1,145 @@ +--- +group: "block-types" +order: 5 +title: "Numbered List Item Block" +slug: "numbered-list-item" +description: "Detailed explanation of the Numbered List Item block type in React-Notion-Custom." +--- + +# Numbered List Item Block + +The Numbered List Item block is used to create ordered lists. Each item starts with a sequential number. + +## Data Structure + +The Notion API data structure for a Numbered List Item block is as follows: + +```json +{ + "type": "numbered_list_item", + "numbered_list_item": { + "rich_text": [ + { + "type": "text", + "text": { + "content": "This is a numbered list item", + "link": null + }, + "annotations": { + "bold": false, + "italic": false, + "strikethrough": false, + "underline": false, + "code": false, + "color": "default" + }, + "plain_text": "This is a numbered list item", + "href": null + } + ], + "color": "default" + } +} +``` + +- `rich_text`: An array containing the text content and style information of the list item. +- `color`: Specifies the color of the text. + +## React Component + +The component that renders the Numbered List Item block in React-Notion-Custom is as follows: + +```jsx +import React from "react"; +import type { NumberedListItemArgs } from "../types"; +import { getColorCss, numberedListItemMarker } from "../utils"; +import RichText from "./internal/rich-text"; + +type NumberedListProps = { + children?: React.ReactNode; +} & NumberedListItemArgs; + +const NumberedListItem: React.FC<NumberedListProps> = ({ + children, + ...props +}) => { + const { + numbered_list_item: { rich_text: texts, color }, + } = props; + const { marker, format } = numberedListItemMarker.getMarker(props); + + return ( + <ol + data-notion-marker-format={format} + className={`notion-block notion-list-numbered ${getColorCss(color)}`} + > + <li className="notion-display-contents"> + <div className="notion-list-numbered-content"> + <span + data-notion-marker-format={format} + className="notion-list-marker" + > + {marker} + </span> + <p> + <RichText props={texts} /> + </p> + </div> + {children} + </li> + </ol> + ); +}; + +export default NumberedListItem; +``` + +## Usage Example + +Here's an example of how to use the Numbered List Item block: + +```jsx +import { Notion } from "react-notion-custom"; + +function MyNotionPage({ blocks }) { + return ( + <Notion> + <Notion.Blocks blocks={blocks} /> + </Notion> + ); +} +``` + +Here, `blocks` is an array of block data received from the Notion API. + +## Styling + +The style of the Numbered List Item block can be customized through the following CSS classes: + +- `.notion-block`: Basic style applied to all Notion blocks +- `.notion-list-numbered`: Specific style for Numbered List Item blocks +- `.notion-list-numbered-content`: Style for the content of list items +- `.notion-list-marker`: Style for numbers + +If additional styling is needed, you can write CSS targeting these classes. + +## Numbering System + +The numbering system for Numbered List Items is determined by the `numberedListItemMarker.getMarker` function. This function returns an appropriate number or character considering the current item's position and nesting level. By default, it follows this order: + +1. Top level: 1, 2, 3, ... +2. Second level: a, b, c, ... +3. Third level: i, ii, iii, ... + +This order can be customized as needed. + +## Handling Nested Lists + +Numbered List Items can also be nested. Nested lists are handled through the `children` prop. The numbering system may change depending on the nesting level, which is handled by the `numberedListItemMarker.getMarker` function. + +## Notes + +- When Numbered List Item blocks are used consecutively, they are automatically grouped into a single list. +- When using nested lists, it's good to clearly express the hierarchical structure through appropriate indentation. +- The numbering system follows Notion's default style, but can be customized through CSS. +- If a list is interrupted and then resumed, the numbering may start over, so care should be taken. diff --git a/packages/docs/content/guide/en/03. block-types/06. Quote.md b/packages/docs/content/guide/en/03. block-types/06. Quote.md new file mode 100644 index 0000000..dda363e --- /dev/null +++ b/packages/docs/content/guide/en/03. block-types/06. Quote.md @@ -0,0 +1,136 @@ +--- +group: "block-types" +order: 6 +title: "Quote Block" +slug: "quote" +description: "Detailed explanation of the Quote block type in React-Notion-Custom." +--- + +# Quote Block + +The Quote block is used to display text quoted from another source. It is typically represented in a visually distinct style. + +## Data Structure + +The Notion API data structure for a Quote block is as follows: + +```json +{ + "type": "quote", + "quote": { + "rich_text": [ + { + "type": "text", + "text": { + "content": "This is a quote", + "link": null + }, + "annotations": { + "bold": false, + "italic": false, + "strikethrough": false, + "underline": false, + "code": false, + "color": "default" + }, + "plain_text": "This is a quote", + "href": null + } + ], + "color": "default" + } +} +``` + +- `rich_text`: An array containing the text content and style information of the quote. +- `color`: Specifies the color of the quote. + +## React Component + +The component that renders the Quote block in React-Notion-Custom is as follows: + +```jsx +import React from "react"; +import type { QuoteArgs } from "../types"; +import { getColorCss } from "../utils"; +import RichText from "./internal/rich-text"; + +type QuoteProps = { + children?: React.ReactNode; +} & QuoteArgs; + +const Quote: React.FC<QuoteProps> = ({ children, ...props }) => { + const { + quote: { color, rich_text: texts }, + } = props; + + return ( + <div className={`notion-block notion-quote ${getColorCss(color)}`}> + <div className="notion-quote-content"> + <p> + <RichText props={texts} /> + </p> + {children} + </div> + </div> + ); +}; + +export default Quote; +``` + +## Usage Example + +Here's an example of how to use the Quote block: + +```jsx +import { Notion } from "react-notion-custom"; + +function MyNotionPage({ blocks }) { + return ( + <Notion> + <Notion.Blocks blocks={blocks} /> + </Notion> + ); +} +``` + +Here, `blocks` is an array of block data received from the Notion API. + +## Styling + +The style of the Quote block can be customized through the following CSS classes: + +- `.notion-block`: Basic style applied to all Notion blocks +- `.notion-quote`: Specific style for Quote blocks +- `.notion-quote-content`: Style for the content of the quote + +If additional styling is needed, you can write CSS targeting these classes. Typically, Quote blocks have the following style characteristics: + +- Thick border on the left or both sides +- Slight indentation +- Changed background color +- Changed font style (e.g., italics) + +For example, you can apply the following CSS: + +```css +.notion-quote { + border-left: 4px solid #ccc; + padding-left: 16px; + margin-left: 0; + margin-right: 0; + font-style: italic; + background-color: #f9f9f9; +} +``` + +## Nesting Support + +Quote blocks can contain other blocks. This is handled through the `children` prop. For example, you can include lists or other text blocks within a quote. + +## Notes + +- Quote blocks are generally used for short quotations. For longer quotes, it might be better to split them into multiple Quote blocks for readability. +- It's good practice to specify the source of the quote. This can be added as a separate Paragraph block below the Quote block. +- For accessibility, it's recommended to add appropriate ARIA attributes for screen reader users. For example, you can use the `role="blockquote"` attribute. diff --git a/packages/docs/content/guide/en/03. block-types/07. Paragraph.md b/packages/docs/content/guide/en/03. block-types/07. Paragraph.md new file mode 100644 index 0000000..8d2cdce --- /dev/null +++ b/packages/docs/content/guide/en/03. block-types/07. Paragraph.md @@ -0,0 +1,106 @@ +--- +group: "block-types" +order: 2 +title: "Paragraph Block" +slug: "paragraph" +description: "Detailed explanation of the Paragraph block type in React-Notion-Custom." +--- + +# Paragraph Block + +The Paragraph block is the most basic block type used to represent plain text. + +## Data Structure + +The Notion API data structure for a Paragraph block is as follows: + +```json +{ + "type": "paragraph", + "paragraph": { + "rich_text": [ + { + "type": "text", + "text": { + "content": "This is a paragraph.", + "link": null + }, + "annotations": { + "bold": false, + "italic": false, + "strikethrough": false, + "underline": false, + "code": false, + "color": "default" + }, + "plain_text": "This is a paragraph.", + "href": null + } + ], + "color": "default" + } +} +``` + +- `rich_text`: An array containing the content and style information of the text. +- `color`: Specifies the color of the text. + +## React Component + +The component that renders the Paragraph block in React-Notion-Custom is as follows: + +```jsx +import React from "react"; +import type { ParagraphArgs } from "../types"; +import { getColorCss } from "../utils"; +import RichText from "./internal/rich-text"; + +type ParagraphProps = { + children?: React.ReactNode; +} & ParagraphArgs; + +const Paragraph: React.FC<ParagraphProps> = ({ children, ...props }) => { + const { + paragraph: { color, rich_text: texts }, + } = props; + + return ( + <div className={`notion-block notion-paragraph ${getColorCss(color)}`}> + <p className="notion-paragraph-content"> + <RichText props={texts} /> + </p> + {children} + </div> + ); +}; + +export default Paragraph; +``` + +## Usage Example + +Here's an example of how to use the Paragraph block: + +```jsx +import { Notion } from "react-notion-custom"; + +function MyNotionPage({ blocks }) { + return ( + <Notion> + <Notion.Blocks blocks={blocks} /> + </Notion> + ); +} +``` + +Here, `blocks` is an array of block data received from the Notion API. + +## Styling + +The style of the Paragraph block can be customized through the following CSS classes: + +- `.notion-block`: Basic style applied to all Notion blocks +- `.notion-paragraph`: Specific style for Paragraph blocks +- `.notion-paragraph-content`: Style for the content of the paragraph + +If additional styling is needed, you can write CSS targeting these classes. diff --git a/packages/docs/content/guide/en/03. block-types/08. Callout.md b/packages/docs/content/guide/en/03. block-types/08. Callout.md new file mode 100644 index 0000000..0901433 --- /dev/null +++ b/packages/docs/content/guide/en/03. block-types/08. Callout.md @@ -0,0 +1,155 @@ +--- +group: "block-types" +order: 7 +title: "Callout Block" +slug: "callout" +description: "Detailed explanation of the Callout block type in React-Notion-Custom." +--- + +# Callout Block + +The Callout block is used to emphasize content that needs attention. It's typically displayed with an icon and a colored background to make it visually stand out. + +## Data Structure + +The Notion API data structure for a Callout block is as follows: + +```json +{ + "type": "callout", + "callout": { + "rich_text": [ + { + "type": "text", + "text": { + "content": "This is a callout", + "link": null + }, + "annotations": { + "bold": false, + "italic": false, + "strikethrough": false, + "underline": false, + "code": false, + "color": "default" + }, + "plain_text": "This is a callout", + "href": null + } + ], + "icon": { + "type": "emoji", + "emoji": "💡" + }, + "color": "gray_background" + } +} +``` + +- `rich_text`: An array containing the text content and style information of the callout. +- `icon`: Information about the icon to be displayed with the callout. It can be an emoji or an external image URL. +- `color`: Specifies the background color of the callout. + +## React Component + +The component that renders the Callout block in React-Notion-Custom is as follows: + +```jsx +import React from "react"; +import type { CalloutArgs } from "../types"; +import { getColorCss } from "../utils"; +import RichText from "./internal/rich-text"; +import Icon from "./internal/icon"; + +type CalloutProps = { + children?: React.ReactNode; +} & CalloutArgs; + +const Callout: React.FC<CalloutProps> = ({ children, ...props }) => { + const { callout } = props; + + return ( + <div + className={`notion-block notion-callout ${getColorCss(callout.color)}`} + > + <div className="notion-callout-content"> + <div className="notion-callout-icon" aria-hidden="true"> + <Icon icon={callout.icon} /> + </div> + <div className="notion-callout-text"> + <RichText props={callout.rich_text} /> + </div> + </div> + {children} + </div> + ); +}; + +export default Callout; +``` + +## Usage Example + +Here's an example of how to use the Callout block: + +```jsx +import { Notion } from "react-notion-custom"; + +function MyNotionPage({ blocks }) { + return ( + <Notion> + <Notion.Blocks blocks={blocks} /> + </Notion> + ); +} +``` + +Here, `blocks` is an array of block data received from the Notion API. + +## Icon Handling + +The icon for the Callout block is handled through the `Icon` component. This component supports both emojis and external image URLs. It selects the appropriate rendering method based on the icon type. + +## Styling + +The style of the Callout block can be customized through the following CSS classes: + +- `.notion-block`: Basic style applied to all Notion blocks +- `.notion-callout`: Specific style for Callout blocks +- `.notion-callout-content`: Style for the content of the callout +- `.notion-callout-icon`: Style for the callout icon +- `.notion-callout-text`: Style for the callout text + +If additional styling is needed, you can write CSS targeting these classes. For example: + +```css +.notion-callout { + border-radius: 4px; + padding: 16px; + display: flex; + align-items: center; +} + +.notion-callout-icon { + margin-right: 12px; + font-size: 24px; +} + +.notion-callout-text { + flex: 1; +} +``` + +## Color Handling + +The `color` property of the Callout determines the background color. The `getColorCss` utility function is used to generate the appropriate CSS class. This function converts Notion's color names into CSS classes. + +## Nesting Support + +Callout blocks can contain other blocks. This is handled through the `children` prop. For example, you can include lists or other text blocks within a callout. + +## Notes + +- Callout blocks are effective for emphasizing important information or warnings. However, excessive use can be distracting, so they should be used appropriately. +- Make sure the combination of icon and background color meets accessibility standards. In particular, maintain sufficient contrast between text and background. +- It's good practice to provide appropriate alternative text for icons to support screen reader users. diff --git a/packages/docs/content/guide/en/03. block-types/09. Toggle.md b/packages/docs/content/guide/en/03. block-types/09. Toggle.md new file mode 100644 index 0000000..1bee0f2 --- /dev/null +++ b/packages/docs/content/guide/en/03. block-types/09. Toggle.md @@ -0,0 +1,179 @@ +--- +group: "block-types" +order: 8 +title: "Toggle Block" +slug: "toggle" +description: "Detailed explanation of the Toggle block type in React-Notion-Custom." +--- + +# Toggle Block + +The Toggle block is used to create collapsible content. It's an interactive element where hidden content expands or collapses when the user clicks on the toggle. + +## Data Structure + +The Notion API data structure for a Toggle block is as follows: + +```json +{ + "type": "toggle", + "toggle": { + "rich_text": [ + { + "type": "text", + "text": { + "content": "Click to toggle", + "link": null + }, + "annotations": { + "bold": false, + "italic": false, + "strikethrough": false, + "underline": false, + "code": false, + "color": "default" + }, + "plain_text": "Click to toggle", + "href": null + } + ], + "color": "default", + "children": [ + // Nested blocks... + ] + } +} +``` + +- `rich_text`: An array containing the display text and style information of the toggle. +- `color`: Specifies the color of the toggle text. +- `children`: An array of nested blocks that will be displayed when the toggle is expanded. + +## React Component + +The component that renders the Toggle block in React-Notion-Custom is as follows: + +```jsx +import React, { useState, useCallback } from "react"; +import type { ToggleArgs } from "../types"; +import { getColorCss } from "../utils"; +import RichText from "./internal/rich-text"; + +type ToggleProps = { + children?: React.ReactNode; +} & ToggleArgs; + +const Toggle: React.FC<ToggleProps> = ({ children, ...props }) => { + const { + toggle: { color, rich_text: texts }, + } = props; + + const [open, setOpen] = useState(false); + + const toggleOpen = useCallback(() => setOpen((prevOpen) => !prevOpen), []); + + return ( + <div + className={`notion-block notion-toggle ${getColorCss(color)} ${ + open ? "notion-toggle-open" : "" + }`} + aria-expanded={open} + > + <div className="notion-toggle-content"> + <button onClick={toggleOpen} className="notion-toggle-button"> + <div + className={`notion-toggle-button-arrow ${ + open ? "notion-toggle-button-arrow-opened" : "" + }`} + /> + </button> + <p> + <RichText props={texts} /> + </p> + </div> + + {open && children} + </div> + ); +}; + +export default Toggle; +``` + +## Usage Example + +Here's an example of how to use the Toggle block: + +```jsx +import { Notion } from "react-notion-custom"; + +function MyNotionPage({ blocks }) { + return ( + <Notion> + <Notion.Blocks blocks={blocks} /> + </Notion> + ); +} +``` + +Here, `blocks` is an array of block data received from the Notion API. + +## State Management + +The Toggle block internally uses the `useState` hook to manage its open/closed state. The `open` state variable represents the current state of the toggle, and the `toggleOpen` function switches this state. + +## Styling + +The style of the Toggle block can be customized through the following CSS classes: + +- `.notion-block`: Basic style applied to all Notion blocks +- `.notion-toggle`: Specific style for Toggle blocks +- `.notion-toggle-open`: Style for when the toggle is in the open state +- `.notion-toggle-content`: Style for the toggle content +- `.notion-toggle-button`: Style for the toggle button +- `.notion-toggle-button-arrow`: Style for the toggle arrow + +If additional styling is needed, you can write CSS targeting these classes. For example: + +```css +.notion-toggle { + margin-bottom: 8px; +} + +.notion-toggle-button { + cursor: pointer; + background: none; + border: none; + padding: 0; + margin-right: 4px; +} + +.notion-toggle-button-arrow { + width: 0; + height: 0; + border-left: 5px solid transparent; + border-right: 5px solid transparent; + border-top: 5px solid #333; + transition: transform 0.3s ease; +} + +.notion-toggle-button-arrow-opened { + transform: rotate(180deg); +} +``` + +## Nesting Support + +Toggle blocks can contain other blocks. This is handled through the `children` prop. The `children` are only rendered when the toggle is in the open state. + +## Accessibility Considerations + +- The `aria-expanded` attribute is used to inform screen readers of the toggle's current state. +- It's good to add `tabIndex={0}` to the toggle button to support keyboard navigation. +- Providing an appropriate `aria-label` to the toggle button is recommended to clearly describe its function. + +## Notes + +- Including too much content inside a toggle can degrade the user experience, so it's best to include only an appropriate amount of content. +- The open/closed state of toggles is not stored on the server, so all toggles will return to their default state (closed) when the page is refreshed. +- Deeply nested toggles can confuse users, so it's best to limit the nesting levels as much as possible. diff --git a/packages/docs/content/guide/kr/01. getting-started/01. introduction.md b/packages/docs/content/guide/kr/01. getting-started/01. introduction.md new file mode 100644 index 0000000..25c08ef --- /dev/null +++ b/packages/docs/content/guide/kr/01. getting-started/01. introduction.md @@ -0,0 +1,102 @@ +--- +group: "getting-started" +order: 1 +title: "프로젝트 소개" +slug: "introduction" +description: "React-Notion-Custom 프로젝트의 개요, 주요 기능, 그리고 활용 가능성에 대해 소개합니다." +--- + +# React-Notion-Custom 프로젝트 소개 + +## React-Notion-Custom이란? + +React-Notion-Custom은 Notion의 강력한 콘텐츠 작성 기능을 웹사이트로 손쉽게 가져올 수 있게 해주는 도구입니다. 이 라이브러리를 사용하면 Notion에서 작성한 콘텐츠를 그대로 React 기반의 웹사이트에서 렌더링할 수 있습니다. 개발자에게는 빠르게 결과를 만들어낼 수 있는 도구가 되고, 콘텐츠 작성자에게는 편리한 관리 도구가 되는, 그야말로 "개발자와 작성자 모두를 위한 매직 포션" 같은 존재입니다. ✨ + +## 주요 기능 및 제공하는 가치 + +### 🛠️ Notion 데이터의 React 렌더링 + +Notion에서 작성한 글, 이미지, 리스트, 코드 블록 등 다양한 콘텐츠를 React 컴포넌트로 쉽게 변환할 수 있습니다. 다음과 같은 구조로 Notion 페이지가 웹에서 재탄생합니다: + +```jsx +import { Notion, type NotionPageData } from 'react-notion-custom'; + +interface NotionPageProps { + pageData: NotionPageData; +} + +function NotionPage({ pageData }: NotionPageProps) { + return ( + <Notion> + <Notion.Cover src={pageData.cover} /> + <Notion.Body> + <Notion.Title title={pageData.title} /> + <Notion.Blocks blocks={pageData.blocks} /> + </Notion.Body> + </Notion> + ); +} +``` + +이 예제에서 `NotionPageData` 타입은 Notion 페이지의 데이터 구조를 정의합니다. 이를 통해 TypeScript의 타입 체크 기능을 활용하여 더 안정적인 코드를 작성할 수 있습니다. + +### 🎨 커스터마이징의 자유 + +CSS만 건드려서 간단하게 테마를 바꾸거나, 기본 제공 컴포넌트를 부분적으로 수정해 필요한 부분만 커스터마이징할 수 있습니다. 심지어 완전히 새로운 컴포넌트를 만들어 React-Notion-Custom에 결합할 수도 있어요. 코드를 다루는 개발자에게는 무한한 가능성을 제공합니다. + +### 🚀 빠른 사이트 구축 + +Notion-dump CLI를 사용해 Notion 데이터를 쉽게 추출하고, Next.js와 같은 프레임워크를 통해 빠르게 정적 사이트를 구축할 수 있습니다. 몇 가지 명령어만으로 사이트를 구축하는 과정은 마치 텍스트에서 바로 웹사이트가 완성되는 느낌을 줍니다. + +```bash +npx notion-dump --page YOUR_PAGE_URL --auth YOUR_API_KEY +``` + +notion-dump를 실행하면 다음과 같은 폴더 구조로 Notion 데이터가 추출됩니다: + +``` +my-notion-project/ +├── notion-data/ +│ ├── page-id-1.json +│ ├── page-id-2.json +│ └── ... +└── public/ + └── notion-dadta/ + ├── page-id-1/ + │ ├── image1.png + │ └── image2.jpg + ├── page-id-2/ + │ └── image1.png + └── ... +``` + +- `notion-data/`: 각 Notion 페이지의 내용이 JSON 형식으로 저장됩니다. 파일명은 페이지의 slug를 기반으로 생성됩니다. +- `public/notion-data/`: Notion 페이지에서 사용된 이미지들이 각 페이지별로 분류되어 저장됩니다. + +이렇게 추출된 데이터는 React-Notion-Custom 컴포넌트와 쉽게 연동되어, Notion의 콘텐츠를 빠르게 웹사이트로 변환할 수 있습니다. JSON 파일의 내용을 React 컴포넌트에 전달하기만 하면, Notion 페이지의 구조와 스타일이 그대로 웹사이트에 반영됩니다. + +## 타겟 사용자와 그들의 활용 가능성 + +### 개발자 + +Notion에서 작성한 콘텐츠를 웹사이트에 쉽게 렌더링하고, 커스터마이징할 수 있는 자유가 주어집니다. 빠르게 웹사이트를 구축하고 싶거나, 상용 서비스의 한계를 넘어 원하는 스타일로 꾸미고 싶다면 React-Notion-Custom이 답입니다. + +### 디자이너 및 크리에이터 + +Notion에서 작성한 콘텐츠를 직접 웹에 반영할 수 있어 디자인에 일관성을 유지하면서 빠르게 콘텐츠를 업데이트할 수 있습니다. + +### 활용 가능성 예시 + +- **개인 블로그**: Notion에서 쉽게 글을 작성하고, 자동으로 웹사이트에 반영합니다. +- **회사 소개 및 채용 페이지**: Notion을 활용해 회사 프로필이나 채용 정보를 빠르게 업데이트할 수 있습니다. +- **포트폴리오 사이트**: 프로젝트와 작업물을 손쉽게 업데이트할 수 있어, 포트폴리오 관리가 한결 쉬워집니다. + +## React-Notion-Custom을 선택해야 하는 이유 + +1. **개발자 중심의 커스터마이징**: 다른 상용 서비스에서는 손댈 수 없는 세밀한 부분까지 수정할 수 있습니다. 코드를 통한 무한한 가능성, 이것이 React-Notion-Custom의 매력입니다. + +2. **오픈소스의 자유**: 모든 것이 투명하고, 여러분의 손에서 직접 다룰 수 있습니다. 필요하면 코드를 수정하고, 프로젝트를 자신의 서버에서 셀프 호스팅할 수 있어 비용 부담도 없습니다. + +3. **빠른 결과와 높은 효율성**: Notion에서 콘텐츠를 작성하고, 간단한 설정만으로 웹사이트에서 바로 사용할 수 있습니다. 콘텐츠 작성과 배포의 과정이 한층 더 간소화됩니다. + +React-Notion-Custom은 여러분의 Notion 페이지를 멋진 웹사이트로 바꾸는 다리입니다. 이제 복잡한 설정은 그만, Notion에서 바로 시작해 보세요! diff --git a/packages/docs/content/guide/kr/01. getting-started/02. quick-start.md b/packages/docs/content/guide/kr/01. getting-started/02. quick-start.md new file mode 100644 index 0000000..56fd351 --- /dev/null +++ b/packages/docs/content/guide/kr/01. getting-started/02. quick-start.md @@ -0,0 +1,216 @@ +--- +group: "getting-started" +order: 2 +title: "빠른 시작 가이드" +slug: "quick-start" +description: "React-Notion-Custom을 사용하여 Notion 콘텐츠를 웹사이트로 구축하는 과정을 단계별로 안내합니다." +--- + +# React-Notion-Custom 빠른 시작 가이드 + +## 2.1. 설치 및 셋업 + +### React-Notion-Custom 설치 + +React-Notion-Custom과 Notion 데이터를 추출할 CLI 도구인 Notion-dump를 설치합니다. + +```bash +npm install react-notion-custom notion-dump +``` + +### Next.js 프로젝트 설정 + +Next.js 프로젝트가 없는 경우 새로 생성합니다. Next.js는 정적 사이트 생성과 서버사이드 렌더링을 지원하여 최적의 성능을 제공합니다. + +```bash +npx create-next-app my-notion-blog +cd my-notion-blog +``` + +### Notion API 키 생성 + +1. Notion 개발자 포털(https://developers.notion.com/)에 접속합니다. +2. 새 통합(integration)을 생성합니다. +3. 생성된 API 키를 안전하게 보관합니다. + +## 2.2. Notion 데이터 추출하기 (notion-dump) + +Notion-dump CLI를 사용하여 Notion 데이터를 JSON 형식으로 추출하고, 이를 웹사이트에 사용할 수 있도록 준비합니다. + +### 데이터 추출 명령어 사용법 + +Notion 페이지 URL과 API 키를 사용해 데이터를 추출합니다. + +```bash +npx notion-dump --page YOUR_PAGE_URL --auth YOUR_API_KEY +``` + +### notion-dump 폴더 구조 및 옵션 + +notion-dump는 기본적으로 Next.js 프로젝트 구조에 맞춰져 있습니다. 추출된 데이터는 다음과 같은 구조로 저장됩니다: + +``` +my-notion-blog/ +├── notion-data/ +│ ├── page-id-1.json +│ ├── page-id-2.json +│ └── ... +└── public/ + └── images/ + ├── page-id-1/ + │ ├── image1.png + │ └── image2.jpg + ├── page-id-2/ + │ └── image1.png + └── ... +``` + +notion-dump의 주요 옵션: + +- `--dir`: JSON 파일이 저장될 디렉토리를 지정합니다. 기본값은 `./notion-data`입니다. +- `--image-dir`: 이미지 파일이 저장될 디렉토리를 지정합니다. 기본값은 `./public/images`입니다. + +예를 들어, 저장 위치를 변경하고 싶다면 다음과 같이 사용할 수 있습니다: + +```bash +npx notion-dump --page YOUR_PAGE_URL --auth YOUR_API_KEY --dir ./data --image-dir ./public/assets/images +``` + +옵션을 지정하지 않으면 기본값이 사용되며, 이는 Next.js 프로젝트 구조에 최적화되어 있습니다. + +## 2.3. React에서 Notion 데이터 렌더링 + +추출한 Notion 데이터를 React 컴포넌트를 사용하여 렌더링합니다. 이 단계에서는 기본적인 페이지 구조와 컴포넌트 사용법을 다룹니다. + +### 기본 렌더링 예제 코드 + +Notion 데이터를 React 컴포넌트로 렌더링하는 간단한 예제입니다. + +```jsx +import { Notion } from "react-notion-custom"; +import pageData from "./notion-data/your-page-id.json"; + +function NotionPage() { + return ( + <Notion data={pageData}> + <Notion.Cover src={pageData.cover} /> + <Notion.Body> + <Notion.Title title={pageData.title} /> + <Notion.Blocks blocks={pageData.blocks} /> + </Notion.Body> + </Notion> + ); +} + +export default NotionPage; +``` + +### 설명 + +- `<Notion>` 컴포넌트는 Notion 데이터를 받아 페이지를 구성합니다. +- `<Cover>`, `<Title>`, `<Blocks>` 등 Notion의 기본 요소를 그대로 웹에서 표현할 수 있습니다. + +## 2.4. Next.js 통합으로 첫 페이지 만들기 + +Next.js를 통해 동적 웹페이지를 생성하고, 정적 사이트 생성(SSG)을 활용해 성능을 최적화합니다. + +### Next.js 페이지 생성 및 동적 라우팅 설정 + +각 포스트 페이지를 동적 라우팅으로 생성하고, 정적 사이트 생성을 위해 `getStaticPaths`와 `getStaticProps`를 사용합니다. + +```jsx +// pages/[slug].js +import { Notion } from "react-notion-custom"; +import posts from "../posts"; + +export async function getStaticPaths() { + const paths = posts.map((post) => ({ + params: { slug: post.slug }, + })); + return { paths, fallback: false }; +} + +export async function getStaticProps({ params }) { + const post = posts.find((p) => p.slug === params.slug); + return { props: { post } }; +} + +export default function PostPage({ post }) { + return ( + <Notion> + <Notion.Blocks blocks={post.content.blocks} /> + </Notion> + ); +} +``` + +### 설명 + +- `getStaticPaths`는 포스트의 경로를 미리 생성합니다. +- `getStaticProps`는 각 포스트의 데이터를 받아와 렌더링합니다. + +## 2.5. 블로그 만들기 + +이 섹션에서는 블로그의 전체 구조를 설정하고, 각 포스트를 어떻게 관리하고 표시할 수 있는지에 대해 설명합니다. + +### posts/index.js 설정 + +블로그 포스트 데이터를 중앙에서 관리합니다. 각 포스트의 메타데이터와 콘텐츠를 배열로 정리하여 사용합니다. + +```javascript +// posts/index.js +import post1Content from "../notion-data/post1.json"; +import post2Content from "../notion-data/post2.json"; + +const posts = [ + { + slug: "post1", + title: "첫 번째 블로그 포스트", + date: "2023-10-01", + description: "첫 번째 블로그 포스트의 설명입니다.", + content: post1Content, + }, + { + slug: "post2", + title: "두 번째 블로그 포스트", + date: "2023-10-05", + description: "두 번째 블로그 포스트의 설명입니다.", + content: post2Content, + }, +]; + +export default posts; +``` + +### 데이터 사용하기: 블로그 메인 페이지 + +`posts` 배열을 사용하여 메인 페이지에서 포스트 목록을 렌더링합니다. + +```jsx +// pages/index.js +import Link from "next/link"; +import posts from "../posts"; + +export default function Home() { + return ( + <div> + <h1>블로그 포스트 목록</h1> + <ul> + {posts.map((post) => ( + <li key={post.slug}> + <Link href={`/${post.slug}`}> + <a>{post.title}</a> + </Link> + <p>{post.description}</p> + <small>{post.date}</small> + </li> + ))} + </ul> + </div> + ); +} +``` + +### 개별 포스트 페이지 생성 + +각 포스트 페이지는 동적 라우팅을 통해 자동 생성됩니다. Notion에서 작성한 글이 그대로 웹사이트에 반영되어, 블로그를 손쉽게 업데이트할 수 있습니다. diff --git a/packages/docs/content/guide/kr/02. customization-guide/01. structure-and-customization.md b/packages/docs/content/guide/kr/02. customization-guide/01. structure-and-customization.md new file mode 100644 index 0000000..06af0f8 --- /dev/null +++ b/packages/docs/content/guide/kr/02. customization-guide/01. structure-and-customization.md @@ -0,0 +1,175 @@ +--- +group: "customization-guide" +order: 1 +title: "React-Notion-Custom 구조 이해와 커스터마이징" +slug: "structure-and-customization" +description: "React-Notion-Custom의 데이터 구조를 이해하고 이를 바탕으로 효과적인 커스터마이징 방법을 학습합니다." +--- + +# React-Notion-Custom 구조 이해와 커스터마이징 + +React-Notion-Custom을 효과적으로 커스터마이징하기 위해서는 먼저 기본 구조와 데이터 흐름을 이해해야 합니다. 이 섹션에서는 Notion 데이터의 구조를 살펴보고, 이를 바탕으로 커스텀 컴포넌트를 만드는 방법을 설명합니다. + +## 데이터 구조 이해하기 + +React-Notion-Custom이 사용하는 Notion 데이터는 JSON 형식으로 구성되며, 페이지 정보와 각 블록의 세부 내용을 포함합니다. 이 구조를 이해하는 것이 효과적인 커스터마이징의 첫 걸음입니다. + +### 기본 데이터 구조 + +```json +{ + "object": "page", + "id": "page-id", + "created_time": "2024-09-20T13:54:00.000Z", + "last_edited_time": "2024-09-21T04:28:00.000Z", + "cover": null, + "icon": null, + "parent": { + "type": "page_id", + "page_id": "parent-page-id" + }, + "properties": { + "title": { + "id": "title", + "type": "title", + "title": [ + { + "type": "text", + "text": { + "content": "페이지 제목", + "link": null + }, + "annotations": { + "bold": false, + "italic": false, + "color": "default" + }, + "plain_text": "페이지 제목" + } + ] + } + }, + "blocks": [ + { + "object": "block", + "id": "block-id-1", + "type": "toggle", + "toggle": { + "rich_text": [ + { + "type": "text", + "text": { + "content": "토글 내용", + "link": null + }, + "annotations": { + "bold": false, + "italic": false, + "color": "default" + }, + "plain_text": "토글 내용" + } + ], + "color": "default" + }, + "blocks": [ + { + "object": "block", + "id": "block-id-2", + "type": "paragraph", + "paragraph": { + "rich_text": [ + { + "type": "text", + "text": { + "content": "토글 내부 단락", + "link": null + }, + "annotations": { + "bold": false, + "italic": false, + "color": "default" + }, + "plain_text": "토글 내부 단락" + } + ], + "color": "default" + } + } + ] + } + ] +} +``` + +### 주요 구조 요소 + +1. **페이지 메타데이터**: `id`, `created_time`, `last_edited_time`, `parent` 등 페이지의 기본 정보를 포함합니다. + +2. **cover와 icon**: 페이지의 커버 이미지와 아이콘 정보입니다. 커스텀 디자인을 적용할 때 이 정보를 활용할 수 있습니다. + +3. **properties**: 페이지의 속성 정보로, 주로 페이지 제목(`title`)이 포함됩니다. 커스텀 헤더를 만들 때 이 정보를 사용할 수 있습니다. + +4. **blocks**: 페이지 내용을 구성하는 블록들의 배열입니다. 각 블록은 다음 구조를 가집니다: + + - `object`: 항상 "block" + - `id`: 블록의 고유 식별자 + - `type`: 블록의 타입 (예: "toggle", "paragraph", "heading_1" 등) + - `[type]`: 블록 타입에 해당하는 키로, 해당 블록의 구체적인 내용을 포함합니다. + - `blocks`: 하위 블록들의 배열. 이를 통해 재귀적인 구조를 형성합니다. + +5. **재귀적 구조**: `blocks` 배열 내의 각 블록은 다시 `blocks` 배열을 가질 수 있어, 중첩된 구조를 표현할 수 있습니다. 이는 토글, 목록 등의 중첩된 콘텐츠를 커스터마이징할 때 중요합니다. + +6. **rich_text**: 대부분의 텍스트 콘텐츠는 `rich_text` 배열로 표현되며, 텍스트의 스타일 정보(`annotations`)를 포함합니다. 이를 통해 텍스트 스타일을 세밀하게 커스터마이징할 수 있습니다. + +## 커스터마이징을 위한 컴포넌트 구조 + +React-Notion-Custom의 기본적인 컴포넌트 구조는 다음과 같습니다: + +<pre style="font-family: monospace; line-height: 1.2; white-space: pre;"> +┌────────────────────────────────────────────────────────────────┐ +│ <Notion> │ +│ ┌────────────────────────────────────────────────────────┐ │ +│ │ <Cover /> │ │ +│ │ // Page cover image │ │ +│ └────────────────────────────────────────────────────────┘ │ +│ │ +│ ┌────────────────────────────┐ │ +│ │ <Body> │ │ +│ │ ┌──────────────────────┐ │ │ +│ │ │ <Title /> │ │ │ +│ │ │ // Page title │ │ │ +│ │ └──────────────────────┘ │ │ +│ │ ┌──────────────────────┐ │ │ +│ │ │ <Blocks> │ │ │ +│ │ │ ┌──────────────────┐ │ │ │ +│ │ │ │<Block type="...">│ │ │ │ +│ │ │ │ // e.g. Heading │ │ │ │ +│ │ │ └──────────────────┘ │ │ │ +│ │ │ ┌──────────────────┐ │ │ │ +│ │ │ │<Block type="...">│ │ │ │ +│ │ │ │ // e.g. Para │ │ │ │ +│ │ │ └──────────────────┘ │ │ │ +│ │ │ ... │ │ │ +│ │ │ // More blocks │ │ │ +│ │ └──────────────────────┘ │ │ +│ └────────────────────────────┘ │ +│ │ +└────────────────────────────────────────────────────────────────┘ +</pre> + +이 구조에서 `<Body>` 컴포넌트는 전체 너비의 약 60%를 차지하며 중앙에 정렬됩니다. 이 레이아웃을 기반으로 각 컴포넌트를 커스터마이징할 수 있습니다. + +## 커스터마이징 전략 + +1. **블록별 커스텀 컴포넌트**: 각 블록 타입(`paragraph`, `heading_1`, `toggle` 등)에 대해 커스텀 컴포넌트를 만들어 기본 스타일과 동작을 오버라이드할 수 있습니다. + +2. **스타일 커스터마이징**: CSS 변수와 클래스를 활용하여 전체적인 디자인을 조정할 수 있습니다. + +3. **레이아웃 수정**: `<Body>` 컴포넌트의 너비나 정렬을 조정하여 전체 레이아웃을 변경할 수 있습니다. + +4. **재귀적 렌더링 활용**: 중첩된 블록 구조를 활용하여 복잡한 레이아웃이나 인터랙션을 구현할 수 있습니다. + +5. **메타데이터 활용**: 페이지의 메타데이터, 커버 이미지, 아이콘 등을 활용하여 풍부한 UI를 구성할 수 있습니다. + +이러한 구조와 전략을 이해하면, React-Notion-Custom을 기반으로 고도로 커스터마이즈된 Notion 기반 웹사이트를 만들 수 있습니다. 다음 섹션에서는 각 요소별 구체적인 커스터마이징 방법을 살펴보겠습니다. diff --git a/packages/docs/content/guide/kr/02. customization-guide/02. custom-component.md b/packages/docs/content/guide/kr/02. customization-guide/02. custom-component.md new file mode 100644 index 0000000..4ff809c --- /dev/null +++ b/packages/docs/content/guide/kr/02. customization-guide/02. custom-component.md @@ -0,0 +1,215 @@ +--- +group: "customization-guide" +order: 2 +title: "커스텀 컴포넌트 사용법" +slug: "custom-component-usage" +description: "React-Notion-Custom의 컴포넌트를 커스터마이징하고 효과적으로 사용하는 방법을 설명합니다." +--- + +# 커스텀 컴포넌트 사용법 + +React-Notion-Custom은 다양한 Notion 블록 타입에 대한 기본 컴포넌트를 제공합니다. 이 섹션에서는 이러한 기본 컴포넌트를 사용하고 커스터마이징하는 방법, 그리고 새로운 커스텀 컴포넌트를 만드는 방법을 설명합니다. + +## 기본 컴포넌트 사용 예시 + +### Heading 컴포넌트 + +Heading 컴포넌트는 Notion의 제목 블록을 렌더링합니다. 다음은 Heading 컴포넌트를 커스터마이징하는 예시입니다: + +```tsx +import React from "react"; +import { Heading, type HeadingArgs } from "react-notion-custom"; + +const CustomHeading: React.FC<HeadingArgs> = (props) => { + // level 3 헤딩에 밑줄 추가 + if (props.type === "heading_3") { + return <Heading {...props} style={{ textDecoration: "underline" }} />; + } + return <Heading {...props} />; +}; + +export default CustomHeading; +``` + +이 예시에서는 `level` 속성을 확인하여 특정 레벨의 헤딩에 대해 스타일을 변경합니다. + +### Paragraph 컴포넌트 + +Paragraph 컴포넌트는 일반 텍스트 블록을 렌더링합니다. 다음은 Paragraph 컴포넌트를 커스터마이징하는 예시입니다: + +```tsx +import React from "react"; +import { Paragraph, type ParagraphArgs } from "react-notion-custom"; + +const CustomParagraph: React.FC<ParagraphArgs> = (props) => { + return ( + <Paragraph + {...props} + className="custom-paragraph" + style={{ lineHeight: "1.8", marginBottom: "1em" }} + /> + ); +}; + +export default CustomParagraph; +``` + +이 예시에서는 모든 단락에 대해 줄 간격과 하단 여백을 조정하고, 사용자 정의 클래스를 추가합니다. + +## Toggle 컴포넌트 커스터마이징 예시 + +Toggle 컴포넌트는 Notion의 토글 블록을 구현합니다. 다음은 Toggle 컴포넌트를 커스터마이징하는 고급 예시입니다: + +```tsx +import React, { useState } from "react"; +import { Toggle, type ToggleArgs } from "react-notion-custom"; +import { ChevronDown, ChevronRight } from "your-icon-library"; + +const CustomToggle: React.FC<ToggleArgs> = (props) => { + const [isOpen, setIsOpen] = useState(false); + + const handleChangeOpen = (open: boolean) => { + setIsOpen(open); + // 여기에 추가적인 로직을 넣을 수 있습니다. + console.log(`Toggle state changed to: ${open}`); + }; + + return ( + <Toggle {...props} isOpen={isOpen} onChangeOpen={handleChangeOpen}> + <Toggle.Icon> + {isOpen ? <ChevronDown size={20} /> : <ChevronRight size={20} />} + </Toggle.Icon> + <Toggle.Content>{props.children}</Toggle.Content> + </Toggle> + ); +}; + +export default CustomToggle; +``` + +이 예시에서는: + +1. `useState`를 사용하여 토글의 열림/닫힘 상태를 관리합니다. +2. `handleChangeOpen` 함수를 통해 상태 변경을 처리하며, 필요에 따라 추가 로직을 구현할 수 있습니다. +3. `Toggle` 컴포넌트에 `isOpen`과 `onChangeOpen` prop을 전달하여 상태를 직접 제어합니다. +4. `isOpen` 상태에 따라 다른 아이콘을 표시합니다. + +`<Toggle.Icon>`에 컨텐츠를 제공하지 않으면 기본 아이콘이 사용됩니다. + +## 커스텀 컴포넌트 적용하기 + +React-Notion-Custom에서 커스텀 컴포넌트를 사용하는 방법은 다음과 같습니다: + +```jsx +import { Notion } from "react-notion-custom"; +import CustomHeading from "./CustomHeading"; +import CustomParagraph from "./CustomParagraph"; +import CustomToggle from "./CustomToggle"; + +const MyNotionPage = ({ pageData }) => { + const customComponents = { + heading_1: CustomHeading, + heading_2: CustomHeading, + heading_3: CustomHeading, + paragraph: CustomParagraph, + toggle: CustomToggle, + }; + + return ( + <Notion custom={customComponents}> + <Notion.Body> + <Notion.Blocks blocks={pageData.blocks} /> + </Notion.Body> + </Notion> + ); +}; +``` + +이 예시에서 `custom` prop을 통해 커스텀 컴포넌트를 Notion 컴포넌트에 전달합니다. 각 블록 타입에 대해 커스텀 컴포넌트를 지정할 수 있습니다. + +## 완전한 커스텀 컴포넌트 만들기 + +React-Notion-Custom의 기본 컴포넌트를 사용하지 않고, 완전히 새로운 컴포넌트를 만들 수도 있습니다. 다음은 Todo 컴포넌트를 처음부터 만드는 예시입니다: + +```tsx +import React from "react"; +import { RichText, type TodoProps } from "react-notion-custom"; + +const Todo: React.FC<TodoProps> = ({ children, ...props }) => { + const { + to_do: { color, rich_text: texts, checked }, + } = props; + + const colorClass = color !== "default" ? `notion-${color}` : ""; + + return ( + <div className={`notion-block notion-to-do ${colorClass}`}> + <div className="notion-to-do-item"> + <div className="notion-to-do-item-content"> + <input + type="checkbox" + checked={checked} + readOnly + className="notion-to-do-checkbox" + /> + <div + className={`notion-to-do-text ${checked ? "notion-to-do-checked" : ""}`} + > + <RichText props={texts} /> + </div> + </div> + {children} + </div> + </div> + ); +}; + +export default Todo; +``` + +이 예시에서 주의할 점: + +- RichText 컴포넌트만 react-notion-custom에서 import합니다. 이는 복잡한 텍스트 렌더링을 처리하기 위함입니다. +- CSS 클래스 이름은 모두 notion-으로 시작합니다. 이는 React-Notion-Custom의 스타일 규칙을 따르는 것입니다. +- gnotion-${color} 형식의 클래스를 추가합니다. 기본 색상("default")의 경우 추가 클래스를 적용하지 않습니다. +- 컴포넌트는 children prop을 받아 하위 컨텐츠를 렌더링할 수 있게 합니다. 이는 중첩된 구조를 지원하기 위함입니다. + +## 컴포넌트 작성 시 주의사항 + +1. **타입 안정성**: TypeScript를 사용하여 prop 타입을 명확히 정의하세요. React-Notion-Custom에서 제공하는 타입(예: `HeadingArgs`, `ParagraphArgs`)을 활용하세요. +2. **prop 전달**: 커스텀 컴포넌트에서 기본 컴포넌트로 모든 prop을 전달하려면 `{...props}`를 사용하세요. 이는 타입 안정성을 유지하면서 불필요한 prop 나열을 방지합니다. +3. **스타일 오버라이딩**: 기존 스타일을 완전히 대체하지 않도록 주의하세요. 대신 `className`을 추가하거나 `style` prop을 확장하는 방식을 사용하세요. +4. **성능 고려**: 불필요한 리렌더링을 방지하기 위해 `React.memo`나 `useMemo`, `useCallback`을 적절히 사용하세요. +5. **접근성**: 커스텀 컴포넌트를 만들 때 접근성(a11y)을 고려하세요. 예를 들어, 토글 버튼에 적절한 `aria-` 속성을 추가하는 것이 좋습니다. +6. **테마 지원**: React-Notion-Custom의 테마 시스템과 호환되도록 커스텀 컴포넌트를 설계하세요. CSS 변수를 활용하면 다크 모드 등의 테마 전환을 쉽게 지원할 수 있습니다. + +## 커스텀 컴포넌트 적용하기 + +React-Notion-Custom에서 커스텀 컴포넌트를 사용하는 방법은 다음과 같습니다: + +```jsx +import { Notion, type NotionPage, type CustomComponents } from 'react-notion-custom'; +import CustomParagraph from './CustomParagraph'; +import CustomToggle from './CustomToggle'; + +const MyNotionPage = ({ pageData }: { pageData: NotionPage }) => { + const customComponents = useMemo<CustomComponents>(() => ({ + paragraph: CustomParagraph, + toggle: CustomToggle, + }), []); + + return ( + <Notion custom={customComponents}> + <Notion.Cover cover={pageData.cover} /> + <Notion.Body> + <Notion.Title title={pageData.title} /> + <Notion.Blocks blocks={pageData.blocks} /> + </Notion.Body> + </Notion> + ); +}; +``` + +이 예시에서 custom prop을 통해 커스텀 컴포넌트를 Notion 컴포넌트에 전달합니다. + +이러한 가이드라인을 따르면 React-Notion-Custom의 기능을 확장하면서도 일관성과 유지보수성을 유지할 수 있습니다. diff --git a/packages/docs/content/guide/kr/02. customization-guide/03. custom-style.md b/packages/docs/content/guide/kr/02. customization-guide/03. custom-style.md new file mode 100644 index 0000000..0e0abd0 --- /dev/null +++ b/packages/docs/content/guide/kr/02. customization-guide/03. custom-style.md @@ -0,0 +1,168 @@ +--- +group: "customization-guide" +order: 3 +title: "CSS 구조 및 스타일링 가이드" +slug: "css-structure-and-styling" +description: "React-Notion-Custom의 CSS 구조를 이해하고 효과적인 스타일링 방법을 학습합니다." +--- + +# CSS 구조 및 스타일링 가이드 + +React-Notion-Custom은 유연하고 확장 가능한 CSS 구조를 제공합니다. 이 섹션에서는 CSS 변수 시스템, 주요 스타일링 포인트, 그리고 Indent 적용 원리에 대해 설명합니다. + +## CSS 변수 시스템 + +React-Notion-Custom은 CSS 변수를 사용하여 전역 스타일을 정의합니다. 이를 통해 테마 변경이나 전체적인 스타일 조정을 쉽게 할 수 있습니다. + +주요 CSS 변수들은 다음과 같습니다: + +```css +.notion { + --notion-font: ui-sans-serif, system-ui, -apple-system, BlinkMacSystemFont, + "Segoe UI", Helvetica, "Apple Color Emoji", Arial, sans-serif, + "Segoe UI Emoji", "Segoe UI Symbol"; + --notion-max-width: 720px; + --notion-header-height: 45px; + --notion-indent: 27px; + + /* 색상 변수 */ + --fg-color: rgb(55, 53, 47); + --fg-color-0: rgba(55, 53, 47, 0.09); + --fg-color-1: rgba(55, 53, 47, 0.16); + --fg-color-2: rgba(55, 53, 47, 0.4); + --fg-color-3: rgba(55, 53, 47, 0.6); + --fg-color-4: #000; + --fg-color-5: rgb(233, 233, 231); + --fg-color-6: rgba(55, 53, 47, 0.8); + --fg-color-icon: var(--fg-color); + + --bg-color: #fff; + --bg-color-0: rgba(135, 131, 120, 0.15); + --bg-color-1: rgb(247, 246, 243); + --bg-color-2: rgba(135, 131, 120, 0.15); + + --select-color-0: rgb(46, 170, 220); + --select-color-1: rgba(35, 131, 226, 0.28); + --select-color-2: #d9eff8; +} +``` + +이러한 변수들을 재정의하여 전체적인 스타일을 쉽게 변경할 수 있습니다. + +## 주요 스타일링 포인트 + +1. **클래스 네이밍 규칙**: 모든 클래스는 `notion-`으로 시작합니다. 이는 스타일 충돌을 방지합니다. + + 예: `.notion-block`, `.notion-h1`, `.notion-toggle` + +2. **블록 레벨 스타일**: 모든 블록 레벨 요소에는 `notion-block` 클래스가 적용됩니다. + + ```css + .notion-block { + display: block; + } + ``` + +3. **색상 적용**: 색상은 CSS 변수를 통해 적용됩니다. + + ```css + .notion-text { + color: var(--fg-color); + } + ``` + +4. **반응형 디자인**: `--notion-max-width` 변수를 활용하여 반응형 레이아웃을 구현합니다. + + ```css + .notion-body { + width: 100%; + max-width: var(--notion-max-width); + margin: 0 auto; + } + ``` + +## Indent 적용 원리 + +React-Notion-Custom은 재귀적인 구조를 통해 들여쓰기를 구현합니다. 이는 CSS에서 다음과 같이 처리됩니다: + +```css +.notion-block > .notion-block { + margin-left: var(--notion-indent); +} + +.notion-block > .notion-display-contents > .notion-block { + margin-left: var(--notion-indent); +} +``` + +이 CSS 규칙은 중첩된 블록에 대해 자동으로 들여쓰기를 적용합니다. `--notion-indent` 변수를 조정하여 들여쓰기 정도를 쉽게 변경할 수 있습니다. + +## 다크 모드 및 커스텀 테마 + +React-Notion-Custom은 기본적으로 다크 모드를 지원합니다. 다크 모드는 다음과 같이 구현되어 있습니다: + +```css +[data-theme="dark"] .notion { + --fg-color: rgba(255, 255, 255, 0.9); + --bg-color: #2f3437; + /* 기타 다크 모드 관련 변수들... */ +} +``` + +또한, 커스텀 테마를 쉽게 적용할 수 있습니다. `data-theme` 속성을 사용하여 원하는 테마를 적용할 수 있습니다: + +```jsx +<Notion data-theme="custom">{/* Notion 컨텐츠 */}</Notion> +``` + +그리고 CSS에서 다음과 같이 스타일을 정의할 수 있습니다: + +```css +.notion[data-theme="custom"] { + --fg-color: #333; + --bg-color: #f4f4f4; + /* 기타 커스텀 테마 관련 변수들... */ +} +``` + +이 방식을 사용하면 선택자 우선순위를 활용하여 기본 스타일을 쉽게 오버라이딩할 수 있습니다. + +## 스타일 커스터마이징 예시 + +1. **글로벌 폰트 변경**: + + ```css + .notion[data-theme="custom"] { + --notion-font: "Roboto", sans-serif; + } + ``` + +2. **헤딩 스타일 변경**: + + ```css + .notion[data-theme="custom"] .notion-h1 { + font-size: 2.5em; + color: var(--select-color-0); + border-bottom: 2px solid var(--fg-color-1); + } + ``` + +3. **들여쓰기 조정**: + + ```css + .notion[data-theme="custom"] { + --notion-indent: 20px; + } + ``` + +4. **다크 모드 구현**: + + ```css + [data-theme="dark"] .notion[data-theme="custom"] { + --fg-color: rgba(255, 255, 255, 0.9); + --bg-color: #2f3437; + /* 다른 색상 변수들도 적절히 조정 */ + } + ``` + +이러한 CSS 구조와 스타일링 가이드를 따르면, React-Notion-Custom을 사용하여 일관성 있고 유지보수가 쉬운 스타일을 구현할 수 있습니다. 또한, 필요에 따라 세부적인 커스터마이징도 가능합니다. diff --git a/packages/docs/content/guide/kr/03. block-types/01. block-type-guide-intro.md b/packages/docs/content/guide/kr/03. block-types/01. block-type-guide-intro.md new file mode 100644 index 0000000..5146a53 --- /dev/null +++ b/packages/docs/content/guide/kr/03. block-types/01. block-type-guide-intro.md @@ -0,0 +1,242 @@ +--- +group: "block-types" +order: 1 +title: "블록 타입 가이드 소개" +slug: "block-type-guide-intro" +description: "React-Notion-Custom에서 지원하는 블록 타입들의 개요, 사용 방법, 커스터마이징 옵션, 그리고 내부 동작 원리를 소개합니다." +--- + +# 블록 타입 가이드 소개 + +React-Notion-Custom은 Notion의 다양한 블록 타입을 지원하며, 각 블록 타입에 대해 Notion과 최대한 유사한 스타일의 컴포넌트를 제공합니다. 이 가이드에서는 현재 지원되는 블록 타입들, 각 블록 타입의 Props, 커스터마이징 방법, 그리고 내부 동작 원리에 대해 설명합니다. + +## 지원되는 블록 타입 + +React-Notion-Custom에서 현재 지원되는 블록 타입과 지원 예정인 블록 타입들의 목록입니다: + +| Block Type | 지원 여부 | Block Type Enum | +| ------------------------ | --------- | ---------------------- | +| Paragraph | ✅ Yes | `paragraph` | +| Heading 1 | ✅ Yes | `heading_1` | +| Heading 2 | ✅ Yes | `heading_2` | +| Heading 3 | ✅ Yes | `heading_3` | +| Bulleted List Item | ✅ Yes | `bulleted_list_item` | +| Numbered List Item | ✅ Yes | `numbered_list_item` | +| To-do | ❌ No | `to_do` | +| Toggle | ✅ Yes | `toggle` | +| Quote | ✅ Yes | `quote` | +| Callout | ✅ Yes | `callout` | +| Equation | ❌ No | `equation` | +| Code | ❌ No | `code` | +| Image | ❌ No | `image` | +| Video | ❌ No | `video` | +| Bookmark | ❌ No | `bookmark` | +| Divider | ✅ Yes | `divider` | +| Table | ❌ No | `table` | +| Table Row | ❌ No | `table_row` | +| Column | ❌ No | `column` | +| Column List | ❌ No | `column_list` | +| Audio | ❌ No | `audio` | +| Synced Block | ❌ No | `synced_block` | +| Table Of Contents | ❌ No | `table_of_contents` | +| Embed | ❌ No | `embed` | +| Figma | ❌ No | `figma` | +| Google Maps | ❌ No | `maps` | +| Google Drive | ❌ No | `drive` | +| Tweet | ❌ No | `tweet` | +| PDF | ❌ No | `pdf` | +| File | ❌ No | `file` | +| Link | ❌ No | `text` (inline) | +| Page Link | ❌ No | `page` | +| External Page Link | ❌ No | `text` (inline) | +| Collections | ❌ No | - | +| Collection View | ❌ No | `collection_view` | +| Collection View Table | ❌ No | `collection_view` | +| Collection View Gallery | ❌ No | `collection_view` | +| Collection View Board | ❌ No | `collection_view` | +| Collection View List | ❌ No | `collection_view` | +| Collection View Calendar | ❌ No | `collection_view` | +| Collection View Page | ❌ No | `collection_view_page` | + +이 목록은 지속적으로 업데이트될 예정이며, 향후 더 많은 블록 타입을 지원할 계획입니다. + +## Props 사용법 + +React-Notion-Custom은 각 블록 타입에 대한 Props 타입을 제공합니다. 이러한 Props 타입들은 라이브러리에서 직접 import하여 사용할 수 있습니다. + +예를 들어, Paragraph 컴포넌트의 Props는 다음과 같이 import하고 사용할 수 있습니다: + +```typescript +import { ParagraphProps } from "react-notion-custom"; + +// ParagraphProps의 구조 +interface ParagraphProps { + type: "paragraph"; + paragraph: { + rich_text: Array<RichTextItemResponse>; + color: string; + }; + id: string; + has_children?: boolean; +} + +// 사용 예시 +const MyParagraph: React.FC<ParagraphProps> = (props) => { + // 커스텀 구현 +}; +``` + +각 블록 타입별 Props의 상세한 구조는 해당 블록 타입 문서에서 확인할 수 있습니다. + +## 커스텀 컴포넌트 적용 + +React-Notion-Custom의 큰 장점 중 하나는 각 블록 타입에 대해 커스텀 컴포넌트로 완전히 교체할 수 있다는 점입니다. 커스텀 컴포넌트를 적용하는 방법은 다음과 같습니다: + +```jsx +import { Notion } from "react-notion-custom"; +import CustomParagraph from "./CustomParagraph"; +import CustomHeading from "./CustomHeading"; + +const customComponents = { + paragraph: CustomParagraph, + heading_1: CustomHeading, + heading_2: CustomHeading, + heading_3: CustomHeading, + // ... 기타 블록 타입에 대한 커스텀 컴포넌트 +}; + +function MyNotionPage({ blocks }) { + return ( + <Notion custom={customComponents}> + <Notion.Blocks blocks={blocks} /> + </Notion> + ); +} +``` + +`Notion` 컴포넌트에 주입되는 `components`의 타입은 다음과 같습니다: + +```typescript +type NotionComponents = Record<string, React.ComponentType<any>>; +``` + +여기서 `key`는 Notion 블록의 타입(예: 'paragraph', 'heading_1' 등)이고, `value`는 해당 타입을 렌더링할 React 컴포넌트입니다. + +## 데이터 전처리 과정 + +React-Notion-Custom은 Notion API에서 가져온 블록 데이터를 UI 렌더링에 최적화된 형태로 전처리합니다. 이 과정에서 `ContextedBlock`이라는 새로운 타입을 정의하여 사용합니다. + +``` ++----------------------+ +| Notion API | +| Block Data | +| +------------------+ | +| | Block | | +| | - id | | +| | - type | | +| | - has_children | | +| | - [type_specific]| | +| +------------------+ | ++----------------------+ + | + | (Input) + v ++---------------------------+ +| resolveToContextedBlocks | +| +-------------------------+ +| | For each block: | +| | +---------------------+ | +| | |resolveToContextedBlo| | +| | |ck (recursive) | | +| | +---------------------+ | +| | | | +| | v | +| | Map parent-child | +| | relationships | +| +-------------------------+ +| | +| v +| Map sibling relationships | ++---------------------------+ + | + | (Output) + v ++---------------------------+ +| ContextedBlocks | +| +-------------------------+ +| | ContextedBlock | +| | - ...Block properties | +| | - context: | +| | - previous (sibling) | +| | - next (sibling) | +| | - parent | +| | - blocks (children) | +| +-------------------------+ ++---------------------------+ + | + | (Used by) + v + +----------------+ + | Notion.Blocks | + | Component | + +----------------+ +``` + +`ContextedBlock`은 기존 `Block` 타입을 확장하여 다음과 같은 추가 정보를 포함합니다: + +- 이전 블록 (형제 관계) +- 다음 블록 (형제 관계) +- 부모 블록 + +이러한 추가 정보는 특정 블록 컴포넌트에서 유용하게 사용될 수 있습니다. + +## 렌더링 프로세스 + +React-Notion-Custom의 렌더링 프로세스는 다음과 같습니다: + +``` ++-------------------+ +------------------+ +| Custom | | Default | +| Components | | Components | ++-------------------+ +------------------+ + | | + | +-----------------+ | + +-->| Notion |<--+ + | Component | + |(React.Provider) | + +-----------------+ + | | + +----------+ +------------+ + | | ++----------------+ +----------------+ +| Notion.Cover | | Notion.Body | ++----------------+ +----------------+ + | + +----------------+ + | Notion.Title | + +----------------+ + | + +----------------+ + | Notion.Blocks | + +----------------+ + | + v + +----------------+ + | Rendered | + | Notion Page | + +----------------+ +``` + +이 프로세스를 통해 커스텀 컴포넌트와 기본 컴포넌트가 함께 작동하여 Notion 페이지를 렌더링합니다. + +## 스타일링 + +현재 제공되는 Notion 컴포넌트들은 Notion의 스타일을 최대한 유사하게 구현했습니다. 각 컴포넌트는 `notion-` 접두사를 가진 CSS 클래스를 사용하여 스타일링되어 있습니다. + +## 향후 계획 + +- 더 많은 블록 타입 지원 추가 +- 테마를 적용한 컴포넌트 셋 개발 +- 고급 커스터마이징 옵션 확장 + +이어지는 페이지들에서 각 블록 타입에 대한 상세한 설명, Props 구조, 사용 예시, 그리고 커스터마이징 옵션을 확인할 수 있습니다. diff --git a/packages/docs/content/guide/kr/03. block-types/02. heading.md b/packages/docs/content/guide/kr/03. block-types/02. heading.md new file mode 100644 index 0000000..043bbc8 --- /dev/null +++ b/packages/docs/content/guide/kr/03. block-types/02. heading.md @@ -0,0 +1,124 @@ +--- +group: "block-types" +order: 3 +title: "Heading 블록" +slug: "notion-block-heading" +description: "React-Notion-Custom의 Heading 블록 타입에 대한 상세 설명입니다." +--- + +# Heading 블록 + +Heading 블록은 문서의 구조를 나타내는 데 사용되며, 세 가지 레벨(Heading 1, Heading 2, Heading 3)을 지원합니다. + +## 데이터 구조 + +Heading 블록의 Notion API 데이터 구조는 다음과 같습니다 (Heading 1의 예시): + +```json +{ + "type": "heading_1", + "heading_1": { + "rich_text": [ + { + "type": "text", + "text": { + "content": "This is a heading", + "link": null + }, + "annotations": { + "bold": false, + "italic": false, + "strikethrough": false, + "underline": false, + "code": false, + "color": "default" + }, + "plain_text": "This is a heading", + "href": null + } + ], + "color": "default", + "is_toggleable": false + } +} +``` + +- `type`: "heading_1", "heading_2", 또는 "heading_3" 중 하나입니다. +- `rich_text`: 헤딩 텍스트의 내용과 스타일 정보를 담고 있는 배열입니다. +- `color`: 헤딩의 색상을 지정합니다. +- `is_toggleable`: 토글 기능 사용 여부를 나타냅니다. + +## React 컴포넌트 + +React-Notion-Custom에서 Heading 블록을 렌더링하는 컴포넌트는 다음과 같습니다: + +```jsx +import React, { useMemo } from "react"; +import type { HeadingsProps, HeadingConfig } from "../types"; +import { getColorCss } from "../utils"; +import RichText from "./internal/rich-text"; + +const Headings: React.FC<HeadingsProps> = ({ children, type, ...props }) => { + const { + [type]: { color, rich_text: texts, is_toggleable }, + } = props; + + const { headingTag: HeadingTag, headingClassName } = + useMemo<HeadingConfig>(() => { + switch (type) { + case "heading_2": + return { headingTag: "h2", headingClassName: "notion-h2" }; + case "heading_3": + return { headingTag: "h3", headingClassName: "notion-h3" }; + default: + return { headingTag: "h1", headingClassName: "notion-h1" }; + } + }, [type]); + + return ( + <div + className={`notion-block ${headingClassName} ${getColorCss(color)}`} + > + <HeadingTag className={`notion-h-content ${headingClassName}-content`}> + <RichText props={texts} /> + </HeadingTag> + {children} + </div> + ); +}; + +export default Headings; +``` + +## 사용 예시 + +Heading 블록을 사용하는 예시는 다음과 같습니다: + +```jsx +import { Notion } from "react-notion-custom"; + +function MyNotionPage({ blocks }) { + return ( + <Notion> + <Notion.Blocks blocks={blocks} /> + </Notion> + ); +} +``` + +여기서 `blocks`는 Notion API로부터 받아온 블록 데이터 배열입니다. + +## 스타일링 + +Heading 블록의 스타일은 다음 CSS 클래스를 통해 커스터마이즈할 수 있습니다: + +- `.notion-block`: 모든 Notion 블록에 적용되는 기본 스타일 +- `.notion-h1`, `.notion-h2`, `.notion-h3`: 각 헤딩 레벨에 대한 특정 스타일 +- `.notion-h1-content`, `.notion-h2-content`, `.notion-h3-content`: 각 헤딩 내용의 스타일 + +추가적인 스타일링이 필요한 경우, 이 클래스들을 대상으로 CSS를 작성하면 됩니다. + +## 주의사항 + +- Heading 블록은 문서의 구조를 나타내는 중요한 요소이므로, 적절한 레벨의 헤딩을 사용하는 것이 좋습니다. +- `is_toggleable` 속성이 true인 경우, 해당 헤딩은 토글 기능을 가지게 됩니다. 이 경우 추가적인 로직이 필요할 수 있습니다. diff --git a/packages/docs/content/guide/kr/03. block-types/03. bulleted-list-item.md b/packages/docs/content/guide/kr/03. block-types/03. bulleted-list-item.md new file mode 100644 index 0000000..628cd0c --- /dev/null +++ b/packages/docs/content/guide/kr/03. block-types/03. bulleted-list-item.md @@ -0,0 +1,134 @@ +--- +group: "block-types" +order: 4 +title: "Bulleted List Item 블록" +slug: "notion-block-bulleted-list-item" +description: "React-Notion-Custom의 Bulleted List Item 블록 타입에 대한 상세 설명입니다." +--- + +# Bulleted List Item 블록 + +Bulleted List Item 블록은 순서가 없는 목록을 만드는 데 사용됩니다. 각 항목은 글머리 기호(bullet)로 시작합니다. + +## 데이터 구조 + +Bulleted List Item 블록의 Notion API 데이터 구조는 다음과 같습니다: + +```json +{ + "type": "bulleted_list_item", + "bulleted_list_item": { + "rich_text": [ + { + "type": "text", + "text": { + "content": "This is a bulleted list item", + "link": null + }, + "annotations": { + "bold": false, + "italic": false, + "strikethrough": false, + "underline": false, + "code": false, + "color": "default" + }, + "plain_text": "This is a bulleted list item", + "href": null + } + ], + "color": "default" + } +} +``` + +- `rich_text`: 리스트 항목의 텍스트 내용과 스타일 정보를 담고 있는 배열입니다. +- `color`: 텍스트의 색상을 지정합니다. + +## React 컴포넌트 + +React-Notion-Custom에서 Bulleted List Item 블록을 렌더링하는 컴포넌트는 다음과 같습니다: + +```jsx +import React from "react"; +import type { BulletedListItemArgs } from "../types"; +import { bulletedListItemMarker, getColorCss } from "../utils"; +import RichText from "./internal/rich-text"; + +type BulletedListItemProps = { + children?: React.ReactNode; +} & BulletedListItemArgs; + +const BulletedListItem: React.FC<BulletedListItemProps> = ({ + children, + ...props +}) => { + const { + bulleted_list_item: { rich_text: texts, color }, + } = props; + const { marker, format } = bulletedListItemMarker.getMarker(props); + + return ( + <ul + data-notion-marker-format={format} + className={`notion-block notion-list-bulleted ${getColorCss(color)}`} + > + <li className="notion-display-contents"> + <div className="notion-list-bulleted-content"> + <span + data-notion-marker-format={format} + className="notion-list-marker" + > + {marker} + </span> + <p> + <RichText props={texts} /> + </p> + </div> + {children} + </li> + </ul> + ); +}; + +export default BulletedListItem; +``` + +## 사용 예시 + +Bulleted List Item 블록을 사용하는 예시는 다음과 같습니다: + +```jsx +import { Notion } from "react-notion-custom"; + +function MyNotionPage({ blocks }) { + return ( + <Notion> + <Notion.Blocks blocks={blocks} /> + </Notion> + ); +} +``` + +여기서 `blocks`는 Notion API로부터 받아온 블록 데이터 배열입니다. + +## 스타일링 + +Bulleted List Item 블록의 스타일은 다음 CSS 클래스를 통해 커스터마이즈할 수 있습니다: + +- `.notion-block`: 모든 Notion 블록에 적용되는 기본 스타일 +- `.notion-list-bulleted`: Bulleted List Item 블록 특정 스타일 +- `.notion-list-bulleted-content`: 리스트 항목 내용의 스타일 +- `.notion-list-marker`: 글머리 기호의 스타일 + +추가적인 스타일링이 필요한 경우, 이 클래스들을 대상으로 CSS를 작성하면 됩니다. + +## 중첩된 리스트 처리 + +Bulleted List Item은 중첩될 수 있습니다. 중첩된 리스트는 `children` prop을 통해 처리됩니다. 중첩 레벨에 따라 글머리 기호의 모양이 달라질 수 있으며, 이는 `bulletedListItemMarker.getMarker` 함수에서 처리됩니다. + +## 주의사항 + +- Bulleted List Item 블록은 연속적으로 사용될 때 자동으로 하나의 리스트로 그룹화됩니다. +- 중첩된 리스트를 사용할 때는 적절한 들여쓰기를 통해 계층 구조를 명확히 표현하는 것이 좋습니다. +- 글머리 기호의 모양은 Notion의 기본 스타일을 따르지만, CSS를 통해 커스터마이즈할 수 있습니다. diff --git a/packages/docs/content/guide/kr/03. block-types/05. numbered-list-item.md b/packages/docs/content/guide/kr/03. block-types/05. numbered-list-item.md new file mode 100644 index 0000000..71c2afa --- /dev/null +++ b/packages/docs/content/guide/kr/03. block-types/05. numbered-list-item.md @@ -0,0 +1,145 @@ +--- +group: "block-types" +order: 5 +title: "Numbered List Item 블록" +slug: "numbered-list-item" +description: "React-Notion-Custom의 Numbered List Item 블록 타입에 대한 상세 설명입니다." +--- + +# Numbered List Item 블록 + +Numbered List Item 블록은 순서가 있는 목록을 만드는 데 사용됩니다. 각 항목은 순차적인 번호로 시작합니다. + +## 데이터 구조 + +Numbered List Item 블록의 Notion API 데이터 구조는 다음과 같습니다: + +```json +{ + "type": "numbered_list_item", + "numbered_list_item": { + "rich_text": [ + { + "type": "text", + "text": { + "content": "This is a numbered list item", + "link": null + }, + "annotations": { + "bold": false, + "italic": false, + "strikethrough": false, + "underline": false, + "code": false, + "color": "default" + }, + "plain_text": "This is a numbered list item", + "href": null + } + ], + "color": "default" + } +} +``` + +- `rich_text`: 리스트 항목의 텍스트 내용과 스타일 정보를 담고 있는 배열입니다. +- `color`: 텍스트의 색상을 지정합니다. + +## React 컴포넌트 + +React-Notion-Custom에서 Numbered List Item 블록을 렌더링하는 컴포넌트는 다음과 같습니다: + +```jsx +import React from "react"; +import type { NumberedListItemArgs } from "../types"; +import { getColorCss, numberedListItemMarker } from "../utils"; +import RichText from "./internal/rich-text"; + +type NumberedListProps = { + children?: React.ReactNode; +} & NumberedListItemArgs; + +const NumberedListItem: React.FC<NumberedListProps> = ({ + children, + ...props +}) => { + const { + numbered_list_item: { rich_text: texts, color }, + } = props; + const { marker, format } = numberedListItemMarker.getMarker(props); + + return ( + <ol + data-notion-marker-format={format} + className={`notion-block notion-list-numbered ${getColorCss(color)}`} + > + <li className="notion-display-contents"> + <div className="notion-list-numbered-content"> + <span + data-notion-marker-format={format} + className="notion-list-marker" + > + {marker} + </span> + <p> + <RichText props={texts} /> + </p> + </div> + {children} + </li> + </ol> + ); +}; + +export default NumberedListItem; +``` + +## 사용 예시 + +Numbered List Item 블록을 사용하는 예시는 다음과 같습니다: + +```jsx +import { Notion } from "react-notion-custom"; + +function MyNotionPage({ blocks }) { + return ( + <Notion> + <Notion.Blocks blocks={blocks} /> + </Notion> + ); +} +``` + +여기서 `blocks`는 Notion API로부터 받아온 블록 데이터 배열입니다. + +## 스타일링 + +Numbered List Item 블록의 스타일은 다음 CSS 클래스를 통해 커스터마이즈할 수 있습니다: + +- `.notion-block`: 모든 Notion 블록에 적용되는 기본 스타일 +- `.notion-list-numbered`: Numbered List Item 블록 특정 스타일 +- `.notion-list-numbered-content`: 리스트 항목 내용의 스타일 +- `.notion-list-marker`: 번호의 스타일 + +추가적인 스타일링이 필요한 경우, 이 클래스들을 대상으로 CSS를 작성하면 됩니다. + +## 번호 매기기 방식 + +Numbered List Item의 번호 매기기 방식은 `numberedListItemMarker.getMarker` 함수에 의해 결정됩니다. 이 함수는 현재 항목의 위치와 중첩 레벨을 고려하여 적절한 번호나 문자를 반환합니다. 기본적으로 다음과 같은 순서를 따릅니다: + +1. 최상위 레벨: 1, 2, 3, ... +2. 두 번째 레벨: a, b, c, ... +3. 세 번째 레벨: i, ii, iii, ... + +이 순서는 필요에 따라 커스터마이즈할 수 있습니다. + +## 중첩된 리스트 처리 + +Numbered List Item도 중첩될 수 있습니다. 중첩된 리스트는 `children` prop을 통해 처리됩니다. 중첩 레벨에 따라 번호 매기기 방식이 달라질 수 있으며, 이는 `numberedListItemMarker.getMarker` 함수에서 처리됩니다. + +## 주의사항 + +- Numbered List Item 블록은 연속적으로 사용될 때 자동으로 하나의 리스트로 그룹화됩니다. +- 중첩된 리스트를 사용할 때는 적절한 들여쓰기를 통해 계층 구조를 명확히 표현하는 것이 좋습니다. +- 번호 매기기 방식은 Notion의 기본 스타일을 따르지만, CSS를 통해 커스터마이즈할 수 있습니다. +- 리스트가 중단되었다가 다시 시작될 경우, 번호 매기기가 새로 시작될 수 있으므로 주의가 필요합니다. diff --git a/packages/docs/content/guide/kr/03. block-types/06. Quote.md b/packages/docs/content/guide/kr/03. block-types/06. Quote.md new file mode 100644 index 0000000..2c4d5b9 --- /dev/null +++ b/packages/docs/content/guide/kr/03. block-types/06. Quote.md @@ -0,0 +1,136 @@ +--- +group: "block-types" +order: 6 +title: "Quote 블록" +slug: "quote" +description: "React-Notion-Custom의 Quote 블록 타입에 대한 상세 설명입니다." +--- + +# Quote 블록 + +Quote 블록은 다른 출처에서 인용한 텍스트를 표시하는 데 사용됩니다. 일반적으로 시각적으로 구분되는 스타일로 표현됩니다. + +## 데이터 구조 + +Quote 블록의 Notion API 데이터 구조는 다음과 같습니다: + +```json +{ + "type": "quote", + "quote": { + "rich_text": [ + { + "type": "text", + "text": { + "content": "This is a quote", + "link": null + }, + "annotations": { + "bold": false, + "italic": false, + "strikethrough": false, + "underline": false, + "code": false, + "color": "default" + }, + "plain_text": "This is a quote", + "href": null + } + ], + "color": "default" + } +} +``` + +- `rich_text`: 인용문의 텍스트 내용과 스타일 정보를 담고 있는 배열입니다. +- `color`: 인용문의 색상을 지정합니다. + +## React 컴포넌트 + +React-Notion-Custom에서 Quote 블록을 렌더링하는 컴포넌트는 다음과 같습니다: + +```jsx +import React from "react"; +import type { QuoteArgs } from "../types"; +import { getColorCss } from "../utils"; +import RichText from "./internal/rich-text"; + +type QuoteProps = { + children?: React.ReactNode; +} & QuoteArgs; + +const Quote: React.FC<QuoteProps> = ({ children, ...props }) => { + const { + quote: { color, rich_text: texts }, + } = props; + + return ( + <div className={`notion-block notion-quote ${getColorCss(color)}`}> + <div className="notion-quote-content"> + <p> + <RichText props={texts} /> + </p> + {children} + </div> + </div> + ); +}; + +export default Quote; +``` + +## 사용 예시 + +Quote 블록을 사용하는 예시는 다음과 같습니다: + +```jsx +import { Notion } from "react-notion-custom"; + +function MyNotionPage({ blocks }) { + return ( + <Notion> + <Notion.Blocks blocks={blocks} /> + </Notion> + ); +} +``` + +여기서 `blocks`는 Notion API로부터 받아온 블록 데이터 배열입니다. + +## 스타일링 + +Quote 블록의 스타일은 다음 CSS 클래스를 통해 커스터마이즈할 수 있습니다: + +- `.notion-block`: 모든 Notion 블록에 적용되는 기본 스타일 +- `.notion-quote`: Quote 블록 특정 스타일 +- `.notion-quote-content`: 인용문 내용의 스타일 + +추가적인 스타일링이 필요한 경우, 이 클래스들을 대상으로 CSS를 작성하면 됩니다. 일반적으로 Quote 블록은 다음과 같은 스타일 특성을 가집니다: + +- 왼쪽 또는 양쪽에 두꺼운 테두리 +- 약간의 들여쓰기 +- 배경색 변경 +- 폰트 스타일 변경 (예: 기울임꼴) + +예를 들어, 다음과 같은 CSS를 적용할 수 있습니다: + +```css +.notion-quote { + border-left: 4px solid #ccc; + padding-left: 16px; + margin-left: 0; + margin-right: 0; + font-style: italic; + background-color: #f9f9f9; +} +``` + +## 중첩 지원 + +Quote 블록은 다른 블록을 포함할 수 있습니다. 이는 `children` prop을 통해 처리됩니다. 예를 들어, 인용문 안에 리스트나 다른 텍스트 블록을 포함할 수 있습니다. + +## 주의사항 + +- Quote 블록은 일반적으로 짧은 인용문에 사용됩니다. 긴 인용문의 경우, 가독성을 위해 여러 개의 Quote 블록으로 나누는 것이 좋을 수 있습니다. +- 인용문의 출처를 명시하는 것이 좋습니다. 이는 Quote 블록 아래에 별도의 Paragraph 블록으로 추가할 수 있습니다. +- 접근성을 고려하여, 스크린 리더 사용자를 위해 적절한 ARIA 속성을 추가하는 것이 좋습니다. 예를 들어, `role="blockquote"` 속성을 사용할 수 있습니다. diff --git a/packages/docs/content/guide/kr/03. block-types/07. Paragraph.md b/packages/docs/content/guide/kr/03. block-types/07. Paragraph.md new file mode 100644 index 0000000..be54ac6 --- /dev/null +++ b/packages/docs/content/guide/kr/03. block-types/07. Paragraph.md @@ -0,0 +1,106 @@ +--- +group: "block-types" +order: 2 +title: "Paragraph 블록" +slug: "paragraph" +description: "React-Notion-Custom의 Paragraph 블록 타입에 대한 상세 설명입니다." +--- + +# Paragraph 블록 + +Paragraph 블록은 일반 텍스트를 표현하는 가장 기본적인 블록 타입입니다. + +## 데이터 구조 + +Paragraph 블록의 Notion API 데이터 구조는 다음과 같습니다: + +```json +{ + "type": "paragraph", + "paragraph": { + "rich_text": [ + { + "type": "text", + "text": { + "content": "This is a paragraph.", + "link": null + }, + "annotations": { + "bold": false, + "italic": false, + "strikethrough": false, + "underline": false, + "code": false, + "color": "default" + }, + "plain_text": "This is a paragraph.", + "href": null + } + ], + "color": "default" + } +} +``` + +- `rich_text`: 텍스트의 내용과 스타일 정보를 담고 있는 배열입니다. +- `color`: 텍스트의 색상을 지정합니다. + +## React 컴포넌트 + +React-Notion-Custom에서 Paragraph 블록을 렌더링하는 컴포넌트는 다음과 같습니다: + +```jsx +import React from "react"; +import type { ParagraphArgs } from "../types"; +import { getColorCss } from "../utils"; +import RichText from "./internal/rich-text"; + +type ParagraphProps = { + children?: React.ReactNode; +} & ParagraphArgs; + +const Paragraph: React.FC<ParagraphProps> = ({ children, ...props }) => { + const { + paragraph: { color, rich_text: texts }, + } = props; + + return ( + <div className={`notion-block notion-paragraph ${getColorCss(color)}`}> + <p className="notion-paragraph-content"> + <RichText props={texts} /> + </p> + {children} + </div> + ); +}; + +export default Paragraph; +``` + +## 사용 예시 + +Paragraph 블록을 사용하는 예시는 다음과 같습니다: + +```jsx +import { Notion } from "react-notion-custom"; + +function MyNotionPage({ blocks }) { + return ( + <Notion> + <Notion.Blocks blocks={blocks} /> + </Notion> + ); +} +``` + +여기서 `blocks`는 Notion API로부터 받아온 블록 데이터 배열입니다. + +## 스타일링 + +Paragraph 블록의 스타일은 다음 CSS 클래스를 통해 커스터마이즈할 수 있습니다: + +- `.notion-block`: 모든 Notion 블록에 적용되는 기본 스타일 +- `.notion-paragraph`: Paragraph 블록 특정 스타일 +- `.notion-paragraph-content`: Paragraph 내용의 스타일 + +추가적인 스타일링이 필요한 경우, 이 클래스들을 대상으로 CSS를 작성하면 됩니다. diff --git a/packages/docs/content/guide/kr/03. block-types/08. Callout.md b/packages/docs/content/guide/kr/03. block-types/08. Callout.md new file mode 100644 index 0000000..349aa27 --- /dev/null +++ b/packages/docs/content/guide/kr/03. block-types/08. Callout.md @@ -0,0 +1,155 @@ +--- +group: "block-types" +order: 7 +title: "Callout 블록" +slug: "callout" +description: "React-Notion-Custom의 Callout 블록 타입에 대한 상세 설명입니다." +--- + +# Callout 블록 + +Callout 블록은 주목을 끌어야 하는 내용을 강조하는 데 사용됩니다. 주로 아이콘과 함께 색상이 있는 배경으로 표시되어 시각적으로 눈에 띄게 만듭니다. + +## 데이터 구조 + +Callout 블록의 Notion API 데이터 구조는 다음과 같습니다: + +```json +{ + "type": "callout", + "callout": { + "rich_text": [ + { + "type": "text", + "text": { + "content": "This is a callout", + "link": null + }, + "annotations": { + "bold": false, + "italic": false, + "strikethrough": false, + "underline": false, + "code": false, + "color": "default" + }, + "plain_text": "This is a callout", + "href": null + } + ], + "icon": { + "type": "emoji", + "emoji": "💡" + }, + "color": "gray_background" + } +} +``` + +- `rich_text`: Callout의 텍스트 내용과 스타일 정보를 담고 있는 배열입니다. +- `icon`: Callout과 함께 표시될 아이콘 정보입니다. 이모지나 외부 이미지 URL이 될 수 있습니다. +- `color`: Callout의 배경색을 지정합니다. + +## React 컴포넌트 + +React-Notion-Custom에서 Callout 블록을 렌더링하는 컴포넌트는 다음과 같습니다: + +```jsx +import React from "react"; +import type { CalloutArgs } from "../types"; +import { getColorCss } from "../utils"; +import RichText from "./internal/rich-text"; +import Icon from "./internal/icon"; + +type CalloutProps = { + children?: React.ReactNode; +} & CalloutArgs; + +const Callout: React.FC<CalloutProps> = ({ children, ...props }) => { + const { callout } = props; + + return ( + <div + className={`notion-block notion-callout ${getColorCss(callout.color)}`} + > + <div className="notion-callout-content"> + <div className="notion-callout-icon" aria-hidden="true"> + <Icon icon={callout.icon} /> + </div> + <div className="notion-callout-text"> + <RichText props={callout.rich_text} /> + </div> + </div> + {children} + </div> + ); +}; + +export default Callout; +``` + +## 사용 예시 + +Callout 블록을 사용하는 예시는 다음과 같습니다: + +```jsx +import { Notion } from "react-notion-custom"; + +function MyNotionPage({ blocks }) { + return ( + <Notion> + <Notion.Blocks blocks={blocks} /> + </Notion> + ); +} +``` + +여기서 `blocks`는 Notion API로부터 받아온 블록 데이터 배열입니다. + +## 아이콘 처리 + +Callout 블록의 아이콘은 `Icon` 컴포넌트를 통해 처리됩니다. 이 컴포넌트는 이모지와 외부 이미지 URL을 모두 지원합니다. 아이콘 타입에 따라 적절한 렌더링 방식을 선택합니다. + +## 스타일링 + +Callout 블록의 스타일은 다음 CSS 클래스를 통해 커스터마이즈할 수 있습니다: + +- `.notion-block`: 모든 Notion 블록에 적용되는 기본 스타일 +- `.notion-callout`: Callout 블록 특정 스타일 +- `.notion-callout-content`: Callout 내용의 스타일 +- `.notion-callout-icon`: Callout 아이콘의 스타일 +- `.notion-callout-text`: Callout 텍스트의 스타일 + +추가적인 스타일링이 필요한 경우, 이 클래스들을 대상으로 CSS를 작성하면 됩니다. 예를 들어: + +```css +.notion-callout { + border-radius: 4px; + padding: 16px; + display: flex; + align-items: center; +} + +.notion-callout-icon { + margin-right: 12px; + font-size: 24px; +} + +.notion-callout-text { + flex: 1; +} +``` + +## 색상 처리 + +Callout의 `color` 속성은 배경색을 결정합니다. `getColorCss` 유틸리티 함수를 사용하여 적절한 CSS 클래스를 생성합니다. 이 함수는 Notion의 색상 이름을 CSS 클래스로 변환합니다. + +## 중첩 지원 + +Callout 블록은 다른 블록을 포함할 수 있습니다. 이는 `children` prop을 통해 처리됩니다. 예를 들어, Callout 안에 리스트나 다른 텍스트 블록을 포함할 수 있습니다. + +## 주의사항 + +- Callout 블록은 주요 정보나 경고를 강조하는 데 효과적입니다. 그러나 과도한 사용은 오히려 주의를 분산시킬 수 있으므로 적절히 사용해야 합니다. +- 아이콘과 배경색의 조합이 접근성 기준을 충족하는지 확인해야 합니다. 특히 텍스트와 배경 간의 충분한 대비를 유지해야 합니다. +- 스크린 리더 사용자를 위해 아이콘에 적절한 대체 텍스트를 제공하는 것이 좋습니다. diff --git a/packages/docs/content/guide/kr/03. block-types/09. Toggle.md b/packages/docs/content/guide/kr/03. block-types/09. Toggle.md new file mode 100644 index 0000000..95feef5 --- /dev/null +++ b/packages/docs/content/guide/kr/03. block-types/09. Toggle.md @@ -0,0 +1,179 @@ +--- +group: "block-types" +order: 8 +title: "Toggle 블록" +slug: "toggle" +description: "React-Notion-Custom의 Toggle 블록 타입에 대한 상세 설명입니다." +--- + +# Toggle 블록 + +Toggle 블록은 접을 수 있는 콘텐츠를 생성하는 데 사용됩니다. 사용자가 토글을 클릭하면 숨겨진 내용이 펼쳐지거나 접히는 인터랙티브한 요소입니다. + +## 데이터 구조 + +Toggle 블록의 Notion API 데이터 구조는 다음과 같습니다: + +```json +{ + "type": "toggle", + "toggle": { + "rich_text": [ + { + "type": "text", + "text": { + "content": "Click to toggle", + "link": null + }, + "annotations": { + "bold": false, + "italic": false, + "strikethrough": false, + "underline": false, + "code": false, + "color": "default" + }, + "plain_text": "Click to toggle", + "href": null + } + ], + "color": "default", + "children": [ + // 중첩된 블록들... + ] + } +} +``` + +- `rich_text`: 토글의 표시 텍스트와 스타일 정보를 담고 있는 배열입니다. +- `color`: 토글 텍스트의 색상을 지정합니다. +- `children`: 토글을 펼쳤을 때 표시될 중첩된 블록들의 배열입니다. + +## React 컴포넌트 + +React-Notion-Custom에서 Toggle 블록을 렌더링하는 컴포넌트는 다음과 같습니다: + +```jsx +import React, { useState, useCallback } from "react"; +import type { ToggleArgs } from "../types"; +import { getColorCss } from "../utils"; +import RichText from "./internal/rich-text"; + +type ToggleProps = { + children?: React.ReactNode; +} & ToggleArgs; + +const Toggle: React.FC<ToggleProps> = ({ children, ...props }) => { + const { + toggle: { color, rich_text: texts }, + } = props; + + const [open, setOpen] = useState(false); + + const toggleOpen = useCallback(() => setOpen((prevOpen) => !prevOpen), []); + + return ( + <div + className={`notion-block notion-toggle ${getColorCss(color)} ${ + open ? "notion-toggle-open" : "" + }`} + aria-expanded={open} + > + <div className="notion-toggle-content"> + <button onClick={toggleOpen} className="notion-toggle-button"> + <div + className={`notion-toggle-button-arrow ${ + open ? "notion-toggle-button-arrow-opened" : "" + }`} + /> + </button> + <p> + <RichText props={texts} /> + </p> + </div> + + {open && children} + </div> + ); +}; + +export default Toggle; +``` + +## 사용 예시 + +Toggle 블록을 사용하는 예시는 다음과 같습니다: + +```jsx +import { Notion } from "react-notion-custom"; + +function MyNotionPage({ blocks }) { + return ( + <Notion> + <Notion.Blocks blocks={blocks} /> + </Notion> + ); +} +``` + +여기서 `blocks`는 Notion API로부터 받아온 블록 데이터 배열입니다. + +## 상태 관리 + +Toggle 블록은 내부적으로 `useState` 훅을 사용하여 열림/닫힘 상태를 관리합니다. `open` 상태 변수는 토글의 현재 상태를 나타내며, `toggleOpen` 함수는 이 상태를 전환합니다. + +## 스타일링 + +Toggle 블록의 스타일은 다음 CSS 클래스를 통해 커스터마이즈할 수 있습니다: + +- `.notion-block`: 모든 Notion 블록에 적용되는 기본 스타일 +- `.notion-toggle`: Toggle 블록 특정 스타일 +- `.notion-toggle-open`: 토글이 열린 상태일 때의 스타일 +- `.notion-toggle-content`: 토글 내용의 스타일 +- `.notion-toggle-button`: 토글 버튼의 스타일 +- `.notion-toggle-button-arrow`: 토글 화살표의 스타일 + +추가적인 스타일링이 필요한 경우, 이 클래스들을 대상으로 CSS를 작성하면 됩니다. 예를 들어: + +```css +.notion-toggle { + margin-bottom: 8px; +} + +.notion-toggle-button { + cursor: pointer; + background: none; + border: none; + padding: 0; + margin-right: 4px; +} + +.notion-toggle-button-arrow { + width: 0; + height: 0; + border-left: 5px solid transparent; + border-right: 5px solid transparent; + border-top: 5px solid #333; + transition: transform 0.3s ease; +} + +.notion-toggle-button-arrow-opened { + transform: rotate(180deg); +} +``` + +## 중첩 지원 + +Toggle 블록은 다른 블록을 포함할 수 있습니다. 이는 `children` prop을 통해 처리됩니다. 토글이 열린 상태일 때만 `children`이 렌더링됩니다. + +## 접근성 고려사항 + +- `aria-expanded` 속성을 사용하여 토글의 현재 상태를 스크린 리더에 알립니다. +- 키보드 네비게이션을 지원하기 위해 토글 버튼에 `tabIndex={0}`를 추가하는 것이 좋습니다. +- 토글 버튼에 적절한 `aria-label`을 제공하여 그 기능을 명확히 설명하는 것이 좋습니다. + +## 주의사항 + +- 토글 내부에 너무 많은 콘텐츠를 넣으면 사용자 경험이 저하될 수 있으므로 적절한 양의 콘텐츠만 포함하는 것이 좋습니다. +- 토글의 열림/닫힘 상태는 서버에 저장되지 않으므로, 페이지를 새로고침하면 모든 토글이 기본 상태(닫힘)로 돌아갑니다. +- 깊은 수준의 중첩된 토글은 사용자를 혼란스럽게 할 수 있으므로 가능한 한 중첩 수준을 제한하는 것이 좋습니다. diff --git a/packages/docs/hooks/useIsMobile.ts b/packages/docs/hooks/useIsMobile.ts new file mode 100644 index 0000000..a47e6d5 --- /dev/null +++ b/packages/docs/hooks/useIsMobile.ts @@ -0,0 +1,22 @@ +"use client"; + +import { useState, useEffect } from "react"; + +const useIsMobile = () => { + const [isMobile, setIsMobile] = useState(false); + + useEffect(() => { + const checkIsMobile = () => { + setIsMobile(window.innerWidth < 768); + }; + + checkIsMobile(); + window.addEventListener("resize", checkIsMobile); + + return () => window.removeEventListener("resize", checkIsMobile); + }, []); + + return isMobile; +}; + +export default useIsMobile; diff --git a/packages/docs/lib/generateTOC.ts b/packages/docs/lib/generateTOC.ts new file mode 100644 index 0000000..b2537d4 --- /dev/null +++ b/packages/docs/lib/generateTOC.ts @@ -0,0 +1,19 @@ +export interface TOCItem { + id: string; + title: string; +} + +export function generateTOC(content: string): TOCItem[] { + const lines = content.split("\n"); + const toc: TOCItem[] = []; + + lines.forEach((line) => { + if (line.startsWith("<h2>")) { + const title = line.slice(4, -5).trim(); + const id = title.toLowerCase().replace(/[^a-z0-9가-힣]+/g, "-"); + toc.push({ id, title }); + } + }); + + return toc; +} diff --git a/packages/docs/lib/mdx.ts b/packages/docs/lib/mdx.ts new file mode 100644 index 0000000..587d0c6 --- /dev/null +++ b/packages/docs/lib/mdx.ts @@ -0,0 +1,122 @@ +import fs from "fs"; +import path from "path"; +import matter from "gray-matter"; +import { remark } from "remark"; +import html from "remark-html"; +import { getGroupOrder, GroupId } from "@/constants/group"; + +const contentDirectory = path.join(process.cwd(), "content", "guide"); + +interface DocumentData { + slug: string; + content: string; + title: string; + group: string; + prevDocument: { slug: string; title: string; group: string } | null; + nextDocument: { slug: string; title: string; group: string } | null; +} + +export async function getDocumentBySlug( + lang: string, + group: string, + slug: string, +): Promise<DocumentData> { + const notFoundDocument: DocumentData = { + slug: "", + content: "<p>Document not found</p>", + title: "Not Found", + group: "", + prevDocument: null, + nextDocument: null, + }; + + if (!lang || !group || !slug) { + return notFoundDocument; + } + + const groupOrder = getGroupOrder(group as GroupId); + const groupDir = path.join(contentDirectory, lang, `${groupOrder}. ${group}`); + + if (!fs.existsSync(groupDir)) { + return notFoundDocument; + } + + const files = fs.readdirSync(groupDir); + let targetFile = null; + + for (const file of files) { + const filePath = path.join(groupDir, file); + const stat = fs.statSync(filePath); + + if (stat.isFile()) { + const fileContents = fs.readFileSync(filePath, "utf8"); + const { data } = matter(fileContents); + + if (data.slug === slug) { + targetFile = filePath; + break; + } + } + } + + if (!targetFile) { + return notFoundDocument; + } + + const fileContents = fs.readFileSync(targetFile, "utf8"); + const { data, content } = matter(fileContents); + const processedContent = await remark().use(html).process(content); + const contentHtml = processedContent.toString(); + + const allDocuments = getAllDocuments(lang); + const currentIndex = allDocuments.findIndex((doc) => doc.slug === slug); + const prevDocument = currentIndex > 0 ? allDocuments[currentIndex - 1] : null; + const nextDocument = + currentIndex < allDocuments.length - 1 + ? allDocuments[currentIndex + 1] + : null; + + return { + slug, + content: contentHtml, + title: data.title || "Untitled", + group: data.group || "", + prevDocument, + nextDocument, + }; +} + +function getAllDocumentsRecursive(dir: string): any[] { + if (!fs.existsSync(dir)) { + return []; + } + + const files = fs.readdirSync(dir); + let documents: any[] = []; + + files.forEach((file) => { + const filePath = path.join(dir, file); + const stat = fs.statSync(filePath); + + if (stat.isDirectory()) { + documents = documents.concat(getAllDocumentsRecursive(filePath)); + } else if (file.endsWith(".md")) { + const fileContents = fs.readFileSync(filePath, "utf8"); + const { data } = matter(fileContents); + documents.push({ + ...data, + slug: data.slug, + group: data.group, + }); + } + }); + + return documents; +} + +export function getAllDocuments(lang: string) { + const langDir = path.join(contentDirectory, lang); + const documents = getAllDocumentsRecursive(langDir); + + return documents; +} diff --git a/packages/docs/next.config.mjs b/packages/docs/next.config.mjs index 4678774..99cb634 100644 --- a/packages/docs/next.config.mjs +++ b/packages/docs/next.config.mjs @@ -1,4 +1,14 @@ /** @type {import('next').NextConfig} */ -const nextConfig = {}; +const nextConfig = { + async redirects() { + return [ + { + source: "/", + destination: "/en", + permanent: true, + }, + ]; + }, +}; export default nextConfig; diff --git a/packages/docs/package.json b/packages/docs/package.json index ac536f6..d89fc50 100644 --- a/packages/docs/package.json +++ b/packages/docs/package.json @@ -9,16 +9,26 @@ "lint": "next lint" }, "dependencies": { + "@heroicons/react": "2.1.5", + "@tailwindcss/typography": "0.5.15", + "@types/hast": "3.0.4", + "gray-matter": "4.0.3", + "next": "14.2.11", + "next-themes": "0.3.0", "react": "^18", "react-dom": "^18", - "next": "14.2.11" + "react-markdown": "9.0.1", + "rehype-highlight": "7.0.0", + "rehype-raw": "7.0.0", + "remark": "15.0.1", + "remark-html": "16.0.1" }, "devDependencies": { - "typescript": "^5", "@types/node": "^20", "@types/react": "^18", "@types/react-dom": "^18", "postcss": "^8", - "tailwindcss": "^3.4.1" + "tailwindcss": "^3.4.1", + "typescript": "^5" } } diff --git a/packages/docs/public/logo.svg b/packages/docs/public/logo.svg new file mode 100644 index 0000000..06077de --- /dev/null +++ b/packages/docs/public/logo.svg @@ -0,0 +1,11 @@ +<svg width="478" height="470" viewBox="0 0 478 470" fill="none" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink"> +<rect width="478" height="470" fill="url(#pattern0_103_15)"/> +<rect x="145.641" y="145.957" width="247.402" height="241.426" fill="white"/> +<path d="M371.278 367.419L317.045 368.805L266.6 283.567L255.432 277.1L261.366 264.694L272.863 264.401C281.86 264.171 288.87 261.24 293.895 255.61C298.919 249.98 301.304 242.166 301.048 232.17C300.75 220.507 297.342 211.091 290.823 203.922C284.467 196.582 275.29 193.065 263.294 193.372C255.797 193.564 248.859 196.075 242.48 200.906L246.205 346.609L274.15 353.897L266.568 370.346L178.097 372.608L177.739 358.612L200.727 338.268L197.181 199.564L169.242 192.525L177.074 176.07L218.56 175.009C233.056 174.639 247.288 173.775 261.258 172.417C271.238 171.495 278.727 170.97 283.725 170.842C296.554 170.514 307.944 172.557 317.893 176.971C328.005 181.214 335.826 187.183 341.358 194.877C347.057 202.568 350.029 211.244 350.276 220.908C350.575 232.571 345.686 243.533 335.612 253.793C325.538 264.054 313.15 271.873 298.449 277.251L318.59 282.488L352.562 339.638L378.86 350.97L371.278 367.419Z" fill="black"/> +<defs> +<pattern id="pattern0_103_15" patternContentUnits="objectBoundingBox" width="1" height="1"> +<use xlink:href="#image0_103_15" transform="matrix(0.00195312 0 0 0.00198637 0 -0.00851064)"/> +</pattern> +<image id="image0_103_15" width="512" height="512" xlink:href=""/> +</defs> +</svg> diff --git a/packages/docs/tailwind.config.ts b/packages/docs/tailwind.config.ts index d43da91..645372a 100644 --- a/packages/docs/tailwind.config.ts +++ b/packages/docs/tailwind.config.ts @@ -1,6 +1,7 @@ import type { Config } from "tailwindcss"; const config: Config = { + darkMode: "class", content: [ "./pages/**/*.{js,ts,jsx,tsx,mdx}", "./components/**/*.{js,ts,jsx,tsx,mdx}", @@ -8,12 +9,15 @@ const config: Config = { ], theme: { extend: { + fontFamily: { + sans: ["var(--font-pretendard)"], + }, colors: { background: "var(--background)", foreground: "var(--foreground)", }, }, }, - plugins: [], + plugins: [require("@tailwindcss/typography")], }; export default config;