From b7d81f425b896c7e0234ef43f069ee170e91cb11 Mon Sep 17 00:00:00 2001 From: kutta97 Date: Sat, 1 Jul 2023 01:59:16 +0900 Subject: [PATCH 01/87] init: initial project setting - ESLint(Airbnb style) & Prettier - Babel dependency - Webpack configuration - path alias setting(Webpack & Babel & VSCode) - husky pre-commit & pre-push rules --- .eslintrc.js | 13 + .gitignore | 23 + .husky/pre-commit | 4 + .husky/pre-push | 12 + .lintstagedrc.json | 3 + .prettierrc.js | 6 + babel.config.js | 17 + index.html | 11 + jsconfig.json | 11 + package.json | 38 + src/App.css | 8 + src/App.js | 9 + src/main.js | 7 + webpack.config.js | 54 + yarn.lock | 4943 ++++++++++++++++++++++++++++++++++++++++++++ 15 files changed, 5159 insertions(+) create mode 100644 .eslintrc.js create mode 100644 .gitignore create mode 100755 .husky/pre-commit create mode 100755 .husky/pre-push create mode 100644 .lintstagedrc.json create mode 100644 .prettierrc.js create mode 100644 babel.config.js create mode 100644 index.html create mode 100644 jsconfig.json create mode 100644 package.json create mode 100644 src/App.css create mode 100644 src/App.js create mode 100644 src/main.js create mode 100644 webpack.config.js create mode 100644 yarn.lock diff --git a/.eslintrc.js b/.eslintrc.js new file mode 100644 index 00000000..f11add17 --- /dev/null +++ b/.eslintrc.js @@ -0,0 +1,13 @@ +module.exports = { + env: { + browser: true, + es2021: true, + }, + extends: ['airbnb-base', 'prettier'], + parserOptions: { + ecmaVersion: 'latest', + sourceType: 'module', + }, + plugins: [], + rules: {}, +}; diff --git a/.gitignore b/.gitignore new file mode 100644 index 00000000..0e596cdc --- /dev/null +++ b/.gitignore @@ -0,0 +1,23 @@ +/dist +/node_modules +/test/js +/test/browsertest/js +/test/fixtures/temp-* +/test/temp +/test/ChangesAndRemovals +/test/**/dev-defaults.webpack.lock +/benchmark/js +/benchmark/fixtures +/examples/**/dist +/assembly/**/*.wat +/assembly/**/*.wasm +/coverage +/.nyc_output +/.jest-cache +.DS_Store +*.log +.idea +.vscode +.cache +.eslintcache +package-lock.json \ No newline at end of file diff --git a/.husky/pre-commit b/.husky/pre-commit new file mode 100755 index 00000000..5a182ef1 --- /dev/null +++ b/.husky/pre-commit @@ -0,0 +1,4 @@ +#!/usr/bin/env sh +. "$(dirname -- "$0")/_/husky.sh" + +yarn lint-staged diff --git a/.husky/pre-push b/.husky/pre-push new file mode 100755 index 00000000..820babe5 --- /dev/null +++ b/.husky/pre-push @@ -0,0 +1,12 @@ +#!/usr/bin/env sh +. "$(dirname -- "$0")/_/husky.sh" + +BRANCH="$(git rev-parse --abbrev-ref HEAD)" + +if [ "$BRANCH" != "4/#5_yanghyejin_working" ]; +then + echo "You can't push except for the branch you're working on" + exit 1 +fi + +exit 0 \ No newline at end of file diff --git a/.lintstagedrc.json b/.lintstagedrc.json new file mode 100644 index 00000000..9a4cee70 --- /dev/null +++ b/.lintstagedrc.json @@ -0,0 +1,3 @@ +{ + "**/*.{js,jsx}": ["yarn format:write", "yarn lint:fix"] +} diff --git a/.prettierrc.js b/.prettierrc.js new file mode 100644 index 00000000..f1284c64 --- /dev/null +++ b/.prettierrc.js @@ -0,0 +1,6 @@ +module.exports = { + trailingComma: 'es5', + tabWidth: 2, + semi: true, + singleQuote: true, +}; diff --git a/babel.config.js b/babel.config.js new file mode 100644 index 00000000..b7687a42 --- /dev/null +++ b/babel.config.js @@ -0,0 +1,17 @@ +module.exports = { + presets: ['@babel/preset-env'], + plugins: [ + [ + 'module-resolver', + { + root: ['./src'], + alias: { + '@components': './src/components', + '@pages': './src/pages', + '@consts': './src/consts', + '@utils': './src/utils', + }, + }, + ], + ], +}; diff --git a/index.html b/index.html new file mode 100644 index 00000000..f7a90332 --- /dev/null +++ b/index.html @@ -0,0 +1,11 @@ + + + + + + Notion with Vanilla JS + + +
+ + diff --git a/jsconfig.json b/jsconfig.json new file mode 100644 index 00000000..e11d9b86 --- /dev/null +++ b/jsconfig.json @@ -0,0 +1,11 @@ +{ + "compilerOptions": { + "baseUrl": ".", + "paths": { + "@utils/*": ["./src/utils/*"], + "@components/*": ["./src/components/*"], + "@pages/*": ["./src/pages/*"], + "@consts/*": ["./src/consts/*"] + } + } +} diff --git a/package.json b/package.json new file mode 100644 index 00000000..bc696a25 --- /dev/null +++ b/package.json @@ -0,0 +1,38 @@ +{ + "name": "FEDC4-5_Project_Notion_VanillaJS", + "version": "1.0.0", + "main": "main.js", + "scripts": { + "start": "yarn serve:dev", + "build": "yarn format:write && yarn lint:fix && yarn build:prod", + "build:dev": "webpack --mode=development", + "build:prod": "webpack --mode=production", + "serve:dev": "webpack serve --mode=development", + "serve:prod": "webpack serve --mode=production", + "lint:fix": "eslint --fix 'src/**/*.js'", + "format:write": "prettier --write 'src/**/*.js' 'src/**/*.css'" + }, + "repository": "https://github.com/prgrms-fe-devcourse/FEDC4-5_Project_Notion_VanillaJS", + "author": "kutta97 ", + "license": "MIT", + "private": true, + "devDependencies": { + "@babel/core": "^7.22.5", + "@babel/preset-env": "^7.22.5", + "babel-loader": "^9.1.2", + "babel-plugin-module-resolver": "^5.0.0", + "css-loader": "^6.8.1", + "eslint": "^8.43.0", + "eslint-config-airbnb-base": "^15.0.0", + "eslint-config-prettier": "^8.8.0", + "eslint-plugin-import": "^2.27.5", + "html-webpack-plugin": "^5.5.3", + "husky": "^8.0.0", + "lint-staged": "^13.2.3", + "prettier": "^2.8.8", + "style-loader": "^3.3.3", + "webpack": "^5.88.1", + "webpack-cli": "^5.1.4", + "webpack-dev-server": "^4.15.1" + } +} diff --git a/src/App.css b/src/App.css new file mode 100644 index 00000000..460f268a --- /dev/null +++ b/src/App.css @@ -0,0 +1,8 @@ +* { + box-sizing: border-box; +} +body { + background-color: #fff; + color: #37352f; + font-family: 'Pretendard'; +} diff --git a/src/App.js b/src/App.js new file mode 100644 index 00000000..a4490565 --- /dev/null +++ b/src/App.js @@ -0,0 +1,9 @@ +import './App.css'; + +export default class App { + constructor({ $target }) { + const $title = document.createElement('h1'); + $title.textContent = 'Notion with Vanilla JS'; + $target.appendChild($title); + } +} diff --git a/src/main.js b/src/main.js new file mode 100644 index 00000000..dd63fdb4 --- /dev/null +++ b/src/main.js @@ -0,0 +1,7 @@ +import App from './App'; + +const initApp = ($target) => new App({ $target }); + +const $app = document.querySelector(`#app`); + +initApp($app); diff --git a/webpack.config.js b/webpack.config.js new file mode 100644 index 00000000..6e03f094 --- /dev/null +++ b/webpack.config.js @@ -0,0 +1,54 @@ +const HtmlWebpackPlugin = require('html-webpack-plugin'); +const path = require('path'); + +module.exports = (_, argv) => { + const isDevelopment = argv.mode !== 'production'; + + return { + entry: './src/main.js', + + resolve: { + extensions: ['.js'], + alias: { + '@components': path.resolve(__dirname, 'src/components'), + '@consts': path.resolve(__dirname, 'src/consts'), + '@pages': path.resolve(__dirname, 'src/pages'), + '@utils': path.resolve(__dirname, 'src/utils'), + }, + }, + + output: { + path: path.resolve(__dirname, './dist'), + filename: 'index.js', + clean: true, + }, + + devServer: { + port: 3000, + hot: true, + }, + + devtool: isDevelopment ? 'eval-source-map' : 'source-map', + + module: { + rules: [ + { + test: /\.js$/i, + exclude: /node_modules/, + use: 'babel-loader', + }, + { + test: /\.css$/i, + use: ['style-loader', 'css-loader'], + }, + ], + }, + + plugins: [ + new HtmlWebpackPlugin({ + inject: 'body', + template: './index.html', + }), + ], + }; +}; diff --git a/yarn.lock b/yarn.lock new file mode 100644 index 00000000..79253558 --- /dev/null +++ b/yarn.lock @@ -0,0 +1,4943 @@ +# THIS IS AN AUTOGENERATED FILE. DO NOT EDIT THIS FILE DIRECTLY. +# yarn lockfile v1 + + +"@aashutoshrathi/word-wrap@^1.2.3": + version "1.2.6" + resolved "https://registry.yarnpkg.com/@aashutoshrathi/word-wrap/-/word-wrap-1.2.6.tgz#bd9154aec9983f77b3a034ecaa015c2e4201f6cf" + integrity sha512-1Yjs2SvM8TflER/OD3cOjhWWOZb58A2t7wpE2S9XfBYTiIl+XFhQG2bjy4Pu1I+EAlCNUzRDYDdFwFYUKvXcIA== + +"@ampproject/remapping@^2.2.0": + version "2.2.1" + resolved "https://registry.yarnpkg.com/@ampproject/remapping/-/remapping-2.2.1.tgz#99e8e11851128b8702cd57c33684f1d0f260b630" + integrity sha512-lFMjJTrFL3j7L9yBxwYfCq2k6qqwHyzuUl/XBnif78PWTJYyL/dfowQHWE3sp6U6ZzqWiiIZnpTMO96zhkjwtg== + dependencies: + "@jridgewell/gen-mapping" "^0.3.0" + "@jridgewell/trace-mapping" "^0.3.9" + +"@babel/code-frame@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/code-frame/-/code-frame-7.22.5.tgz#234d98e1551960604f1246e6475891a570ad5658" + integrity sha512-Xmwn266vad+6DAqEB2A6V/CcZVp62BbwVmcOJc2RPuwih1kw02TjQvWVWlcKGbBPd+8/0V5DEkOcizRGYsspYQ== + dependencies: + "@babel/highlight" "^7.22.5" + +"@babel/compat-data@^7.17.7", "@babel/compat-data@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/compat-data/-/compat-data-7.22.5.tgz#b1f6c86a02d85d2dd3368a2b67c09add8cd0c255" + integrity sha512-4Jc/YuIaYqKnDDz892kPIledykKg12Aw1PYX5i/TY28anJtacvM1Rrr8wbieB9GfEJwlzqT0hUEao0CxEebiDA== + +"@babel/core@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/core/-/core-7.22.5.tgz#d67d9747ecf26ee7ecd3ebae1ee22225fe902a89" + integrity sha512-SBuTAjg91A3eKOvD+bPEz3LlhHZRNu1nFOVts9lzDJTXshHTjII0BAtDS3Y2DAkdZdDKWVZGVwkDfc4Clxn1dg== + dependencies: + "@ampproject/remapping" "^2.2.0" + "@babel/code-frame" "^7.22.5" + "@babel/generator" "^7.22.5" + "@babel/helper-compilation-targets" "^7.22.5" + "@babel/helper-module-transforms" "^7.22.5" + "@babel/helpers" "^7.22.5" + "@babel/parser" "^7.22.5" + "@babel/template" "^7.22.5" + "@babel/traverse" "^7.22.5" + "@babel/types" "^7.22.5" + convert-source-map "^1.7.0" + debug "^4.1.0" + gensync "^1.0.0-beta.2" + json5 "^2.2.2" + semver "^6.3.0" + +"@babel/generator@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/generator/-/generator-7.22.5.tgz#1e7bf768688acfb05cf30b2369ef855e82d984f7" + integrity sha512-+lcUbnTRhd0jOewtFSedLyiPsD5tswKkbgcezOqqWFUVNEwoUTlpPOBmvhG7OXWLR4jMdv0czPGH5XbflnD1EA== + dependencies: + "@babel/types" "^7.22.5" + "@jridgewell/gen-mapping" "^0.3.2" + "@jridgewell/trace-mapping" "^0.3.17" + jsesc "^2.5.1" + +"@babel/helper-annotate-as-pure@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/helper-annotate-as-pure/-/helper-annotate-as-pure-7.22.5.tgz#e7f06737b197d580a01edf75d97e2c8be99d3882" + integrity sha512-LvBTxu8bQSQkcyKOU+a1btnNFQ1dMAd0R6PyW3arXes06F6QLWLIrd681bxRPIXlrMGR3XYnW9JyML7dP3qgxg== + dependencies: + "@babel/types" "^7.22.5" + +"@babel/helper-builder-binary-assignment-operator-visitor@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/helper-builder-binary-assignment-operator-visitor/-/helper-builder-binary-assignment-operator-visitor-7.22.5.tgz#a3f4758efdd0190d8927fcffd261755937c71878" + integrity sha512-m1EP3lVOPptR+2DwD125gziZNcmoNSHGmJROKoy87loWUQyJaVXDgpmruWqDARZSmtYQ+Dl25okU8+qhVzuykw== + dependencies: + "@babel/types" "^7.22.5" + +"@babel/helper-compilation-targets@^7.17.7", "@babel/helper-compilation-targets@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/helper-compilation-targets/-/helper-compilation-targets-7.22.5.tgz#fc7319fc54c5e2fa14b2909cf3c5fd3046813e02" + integrity sha512-Ji+ywpHeuqxB8WDxraCiqR0xfhYjiDE/e6k7FuIaANnoOFxAHskHChz4vA1mJC9Lbm01s1PVAGhQY4FUKSkGZw== + dependencies: + "@babel/compat-data" "^7.22.5" + "@babel/helper-validator-option" "^7.22.5" + browserslist "^4.21.3" + lru-cache "^5.1.1" + semver "^6.3.0" + +"@babel/helper-create-class-features-plugin@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/helper-create-class-features-plugin/-/helper-create-class-features-plugin-7.22.5.tgz#2192a1970ece4685fbff85b48da2c32fcb130b7c" + integrity sha512-xkb58MyOYIslxu3gKmVXmjTtUPvBU4odYzbiIQbWwLKIHCsx6UGZGX6F1IznMFVnDdirseUZopzN+ZRt8Xb33Q== + dependencies: + "@babel/helper-annotate-as-pure" "^7.22.5" + "@babel/helper-environment-visitor" "^7.22.5" + "@babel/helper-function-name" "^7.22.5" + "@babel/helper-member-expression-to-functions" "^7.22.5" + "@babel/helper-optimise-call-expression" "^7.22.5" + "@babel/helper-replace-supers" "^7.22.5" + "@babel/helper-skip-transparent-expression-wrappers" "^7.22.5" + "@babel/helper-split-export-declaration" "^7.22.5" + semver "^6.3.0" + +"@babel/helper-create-regexp-features-plugin@^7.18.6", "@babel/helper-create-regexp-features-plugin@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/helper-create-regexp-features-plugin/-/helper-create-regexp-features-plugin-7.22.5.tgz#bb2bf0debfe39b831986a4efbf4066586819c6e4" + integrity sha512-1VpEFOIbMRaXyDeUwUfmTIxExLwQ+zkW+Bh5zXpApA3oQedBx9v/updixWxnx/bZpKw7u8VxWjb/qWpIcmPq8A== + dependencies: + "@babel/helper-annotate-as-pure" "^7.22.5" + regexpu-core "^5.3.1" + semver "^6.3.0" + +"@babel/helper-define-polyfill-provider@^0.4.0": + version "0.4.0" + resolved "https://registry.yarnpkg.com/@babel/helper-define-polyfill-provider/-/helper-define-polyfill-provider-0.4.0.tgz#487053f103110f25b9755c5980e031e93ced24d8" + integrity sha512-RnanLx5ETe6aybRi1cO/edaRH+bNYWaryCEmjDDYyNr4wnSzyOp8T0dWipmqVHKEY3AbVKUom50AKSlj1zmKbg== + dependencies: + "@babel/helper-compilation-targets" "^7.17.7" + "@babel/helper-plugin-utils" "^7.16.7" + debug "^4.1.1" + lodash.debounce "^4.0.8" + resolve "^1.14.2" + semver "^6.1.2" + +"@babel/helper-environment-visitor@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/helper-environment-visitor/-/helper-environment-visitor-7.22.5.tgz#f06dd41b7c1f44e1f8da6c4055b41ab3a09a7e98" + integrity sha512-XGmhECfVA/5sAt+H+xpSg0mfrHq6FzNr9Oxh7PSEBBRUb/mL7Kz3NICXb194rCqAEdxkhPT1a88teizAFyvk8Q== + +"@babel/helper-function-name@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/helper-function-name/-/helper-function-name-7.22.5.tgz#ede300828905bb15e582c037162f99d5183af1be" + integrity sha512-wtHSq6jMRE3uF2otvfuD3DIvVhOsSNshQl0Qrd7qC9oQJzHvOL4qQXlQn2916+CXGywIjpGuIkoyZRRxHPiNQQ== + dependencies: + "@babel/template" "^7.22.5" + "@babel/types" "^7.22.5" + +"@babel/helper-hoist-variables@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/helper-hoist-variables/-/helper-hoist-variables-7.22.5.tgz#c01a007dac05c085914e8fb652b339db50d823bb" + integrity sha512-wGjk9QZVzvknA6yKIUURb8zY3grXCcOZt+/7Wcy8O2uctxhplmUPkOdlgoNhmdVee2c92JXbf1xpMtVNbfoxRw== + dependencies: + "@babel/types" "^7.22.5" + +"@babel/helper-member-expression-to-functions@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/helper-member-expression-to-functions/-/helper-member-expression-to-functions-7.22.5.tgz#0a7c56117cad3372fbf8d2fb4bf8f8d64a1e76b2" + integrity sha512-aBiH1NKMG0H2cGZqspNvsaBe6wNGjbJjuLy29aU+eDZjSbbN53BaxlpB02xm9v34pLTZ1nIQPFYn2qMZoa5BQQ== + dependencies: + "@babel/types" "^7.22.5" + +"@babel/helper-module-imports@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/helper-module-imports/-/helper-module-imports-7.22.5.tgz#1a8f4c9f4027d23f520bd76b364d44434a72660c" + integrity sha512-8Dl6+HD/cKifutF5qGd/8ZJi84QeAKh+CEe1sBzz8UayBBGg1dAIJrdHOcOM5b2MpzWL2yuotJTtGjETq0qjXg== + dependencies: + "@babel/types" "^7.22.5" + +"@babel/helper-module-transforms@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/helper-module-transforms/-/helper-module-transforms-7.22.5.tgz#0f65daa0716961b6e96b164034e737f60a80d2ef" + integrity sha512-+hGKDt/Ze8GFExiVHno/2dvG5IdstpzCq0y4Qc9OJ25D4q3pKfiIP/4Vp3/JvhDkLKsDK2api3q3fpIgiIF5bw== + dependencies: + "@babel/helper-environment-visitor" "^7.22.5" + "@babel/helper-module-imports" "^7.22.5" + "@babel/helper-simple-access" "^7.22.5" + "@babel/helper-split-export-declaration" "^7.22.5" + "@babel/helper-validator-identifier" "^7.22.5" + "@babel/template" "^7.22.5" + "@babel/traverse" "^7.22.5" + "@babel/types" "^7.22.5" + +"@babel/helper-optimise-call-expression@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/helper-optimise-call-expression/-/helper-optimise-call-expression-7.22.5.tgz#f21531a9ccbff644fdd156b4077c16ff0c3f609e" + integrity sha512-HBwaojN0xFRx4yIvpwGqxiV2tUfl7401jlok564NgB9EHS1y6QT17FmKWm4ztqjeVdXLuC4fSvHc5ePpQjoTbw== + dependencies: + "@babel/types" "^7.22.5" + +"@babel/helper-plugin-utils@^7.0.0", "@babel/helper-plugin-utils@^7.10.4", "@babel/helper-plugin-utils@^7.12.13", "@babel/helper-plugin-utils@^7.14.5", "@babel/helper-plugin-utils@^7.16.7", "@babel/helper-plugin-utils@^7.18.6", "@babel/helper-plugin-utils@^7.22.5", "@babel/helper-plugin-utils@^7.8.0", "@babel/helper-plugin-utils@^7.8.3": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/helper-plugin-utils/-/helper-plugin-utils-7.22.5.tgz#dd7ee3735e8a313b9f7b05a773d892e88e6d7295" + integrity sha512-uLls06UVKgFG9QD4OeFYLEGteMIAa5kpTPcFL28yuCIIzsf6ZyKZMllKVOCZFhiZ5ptnwX4mtKdWCBE/uT4amg== + +"@babel/helper-remap-async-to-generator@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/helper-remap-async-to-generator/-/helper-remap-async-to-generator-7.22.5.tgz#14a38141a7bf2165ad38da61d61cf27b43015da2" + integrity sha512-cU0Sq1Rf4Z55fgz7haOakIyM7+x/uCFwXpLPaeRzfoUtAEAuUZjZvFPjL/rk5rW693dIgn2hng1W7xbT7lWT4g== + dependencies: + "@babel/helper-annotate-as-pure" "^7.22.5" + "@babel/helper-environment-visitor" "^7.22.5" + "@babel/helper-wrap-function" "^7.22.5" + "@babel/types" "^7.22.5" + +"@babel/helper-replace-supers@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/helper-replace-supers/-/helper-replace-supers-7.22.5.tgz#71bc5fb348856dea9fdc4eafd7e2e49f585145dc" + integrity sha512-aLdNM5I3kdI/V9xGNyKSF3X/gTyMUBohTZ+/3QdQKAA9vxIiy12E+8E2HoOP1/DjeqU+g6as35QHJNMDDYpuCg== + dependencies: + "@babel/helper-environment-visitor" "^7.22.5" + "@babel/helper-member-expression-to-functions" "^7.22.5" + "@babel/helper-optimise-call-expression" "^7.22.5" + "@babel/template" "^7.22.5" + "@babel/traverse" "^7.22.5" + "@babel/types" "^7.22.5" + +"@babel/helper-simple-access@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/helper-simple-access/-/helper-simple-access-7.22.5.tgz#4938357dc7d782b80ed6dbb03a0fba3d22b1d5de" + integrity sha512-n0H99E/K+Bika3++WNL17POvo4rKWZ7lZEp1Q+fStVbUi8nxPQEBOlTmCOxW/0JsS56SKKQ+ojAe2pHKJHN35w== + dependencies: + "@babel/types" "^7.22.5" + +"@babel/helper-skip-transparent-expression-wrappers@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/helper-skip-transparent-expression-wrappers/-/helper-skip-transparent-expression-wrappers-7.22.5.tgz#007f15240b5751c537c40e77abb4e89eeaaa8847" + integrity sha512-tK14r66JZKiC43p8Ki33yLBVJKlQDFoA8GYN67lWCDCqoL6EMMSuM9b+Iff2jHaM/RRFYl7K+iiru7hbRqNx8Q== + dependencies: + "@babel/types" "^7.22.5" + +"@babel/helper-split-export-declaration@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/helper-split-export-declaration/-/helper-split-export-declaration-7.22.5.tgz#88cf11050edb95ed08d596f7a044462189127a08" + integrity sha512-thqK5QFghPKWLhAV321lxF95yCg2K3Ob5yw+M3VHWfdia0IkPXUtoLH8x/6Fh486QUvzhb8YOWHChTVen2/PoQ== + dependencies: + "@babel/types" "^7.22.5" + +"@babel/helper-string-parser@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/helper-string-parser/-/helper-string-parser-7.22.5.tgz#533f36457a25814cf1df6488523ad547d784a99f" + integrity sha512-mM4COjgZox8U+JcXQwPijIZLElkgEpO5rsERVDJTc2qfCDfERyob6k5WegS14SX18IIjv+XD+GrqNumY5JRCDw== + +"@babel/helper-validator-identifier@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/helper-validator-identifier/-/helper-validator-identifier-7.22.5.tgz#9544ef6a33999343c8740fa51350f30eeaaaf193" + integrity sha512-aJXu+6lErq8ltp+JhkJUfk1MTGyuA4v7f3pA+BJ5HLfNC6nAQ0Cpi9uOquUj8Hehg0aUiHzWQbOVJGao6ztBAQ== + +"@babel/helper-validator-option@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/helper-validator-option/-/helper-validator-option-7.22.5.tgz#de52000a15a177413c8234fa3a8af4ee8102d0ac" + integrity sha512-R3oB6xlIVKUnxNUxbmgq7pKjxpru24zlimpE8WK47fACIlM0II/Hm1RS8IaOI7NgCr6LNS+jl5l75m20npAziw== + +"@babel/helper-wrap-function@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/helper-wrap-function/-/helper-wrap-function-7.22.5.tgz#44d205af19ed8d872b4eefb0d2fa65f45eb34f06" + integrity sha512-bYqLIBSEshYcYQyfks8ewYA8S30yaGSeRslcvKMvoUk6HHPySbxHq9YRi6ghhzEU+yhQv9bP/jXnygkStOcqZw== + dependencies: + "@babel/helper-function-name" "^7.22.5" + "@babel/template" "^7.22.5" + "@babel/traverse" "^7.22.5" + "@babel/types" "^7.22.5" + +"@babel/helpers@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/helpers/-/helpers-7.22.5.tgz#74bb4373eb390d1ceed74a15ef97767e63120820" + integrity sha512-pSXRmfE1vzcUIDFQcSGA5Mr+GxBV9oiRKDuDxXvWQQBCh8HoIjs/2DlDB7H8smac1IVrB9/xdXj2N3Wol9Cr+Q== + dependencies: + "@babel/template" "^7.22.5" + "@babel/traverse" "^7.22.5" + "@babel/types" "^7.22.5" + +"@babel/highlight@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/highlight/-/highlight-7.22.5.tgz#aa6c05c5407a67ebce408162b7ede789b4d22031" + integrity sha512-BSKlD1hgnedS5XRnGOljZawtag7H1yPfQp0tdNJCHoH6AZ+Pcm9VvkrK59/Yy593Ypg0zMxH2BxD1VPYUQ7UIw== + dependencies: + "@babel/helper-validator-identifier" "^7.22.5" + chalk "^2.0.0" + js-tokens "^4.0.0" + +"@babel/parser@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/parser/-/parser-7.22.5.tgz#721fd042f3ce1896238cf1b341c77eb7dee7dbea" + integrity sha512-DFZMC9LJUG9PLOclRC32G63UXwzqS2koQC8dkx+PLdmt1xSePYpbT/NbsrJy8Q/muXz7o/h/d4A7Fuyixm559Q== + +"@babel/plugin-bugfix-safari-id-destructuring-collision-in-function-expression@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/plugin-bugfix-safari-id-destructuring-collision-in-function-expression/-/plugin-bugfix-safari-id-destructuring-collision-in-function-expression-7.22.5.tgz#87245a21cd69a73b0b81bcda98d443d6df08f05e" + integrity sha512-NP1M5Rf+u2Gw9qfSO4ihjcTGW5zXTi36ITLd4/EoAcEhIZ0yjMqmftDNl3QC19CX7olhrjpyU454g/2W7X0jvQ== + dependencies: + "@babel/helper-plugin-utils" "^7.22.5" + +"@babel/plugin-bugfix-v8-spread-parameters-in-optional-chaining@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/plugin-bugfix-v8-spread-parameters-in-optional-chaining/-/plugin-bugfix-v8-spread-parameters-in-optional-chaining-7.22.5.tgz#fef09f9499b1f1c930da8a0c419db42167d792ca" + integrity sha512-31Bb65aZaUwqCbWMnZPduIZxCBngHFlzyN6Dq6KAJjtx+lx6ohKHubc61OomYi7XwVD4Ol0XCVz4h+pYFR048g== + dependencies: + "@babel/helper-plugin-utils" "^7.22.5" + "@babel/helper-skip-transparent-expression-wrappers" "^7.22.5" + "@babel/plugin-transform-optional-chaining" "^7.22.5" + +"@babel/plugin-proposal-private-property-in-object@7.21.0-placeholder-for-preset-env.2": + version "7.21.0-placeholder-for-preset-env.2" + resolved "https://registry.yarnpkg.com/@babel/plugin-proposal-private-property-in-object/-/plugin-proposal-private-property-in-object-7.21.0-placeholder-for-preset-env.2.tgz#7844f9289546efa9febac2de4cfe358a050bd703" + integrity sha512-SOSkfJDddaM7mak6cPEpswyTRnuRltl429hMraQEglW+OkovnCzsiszTmsrlY//qLFjCpQDFRvjdm2wA5pPm9w== + +"@babel/plugin-proposal-unicode-property-regex@^7.4.4": + version "7.18.6" + resolved "https://registry.yarnpkg.com/@babel/plugin-proposal-unicode-property-regex/-/plugin-proposal-unicode-property-regex-7.18.6.tgz#af613d2cd5e643643b65cded64207b15c85cb78e" + integrity sha512-2BShG/d5yoZyXZfVePH91urL5wTG6ASZU9M4o03lKK8u8UW1y08OMttBSOADTcJrnPMpvDXRG3G8fyLh4ovs8w== + dependencies: + "@babel/helper-create-regexp-features-plugin" "^7.18.6" + "@babel/helper-plugin-utils" "^7.18.6" + +"@babel/plugin-syntax-async-generators@^7.8.4": + version "7.8.4" + resolved "https://registry.yarnpkg.com/@babel/plugin-syntax-async-generators/-/plugin-syntax-async-generators-7.8.4.tgz#a983fb1aeb2ec3f6ed042a210f640e90e786fe0d" + integrity sha512-tycmZxkGfZaxhMRbXlPXuVFpdWlXpir2W4AMhSJgRKzk/eDlIXOhb2LHWoLpDF7TEHylV5zNhykX6KAgHJmTNw== + dependencies: + "@babel/helper-plugin-utils" "^7.8.0" + +"@babel/plugin-syntax-class-properties@^7.12.13": + version "7.12.13" + resolved "https://registry.yarnpkg.com/@babel/plugin-syntax-class-properties/-/plugin-syntax-class-properties-7.12.13.tgz#b5c987274c4a3a82b89714796931a6b53544ae10" + integrity sha512-fm4idjKla0YahUNgFNLCB0qySdsoPiZP3iQE3rky0mBUtMZ23yDJ9SJdg6dXTSDnulOVqiF3Hgr9nbXvXTQZYA== + dependencies: + "@babel/helper-plugin-utils" "^7.12.13" + +"@babel/plugin-syntax-class-static-block@^7.14.5": + version "7.14.5" + resolved "https://registry.yarnpkg.com/@babel/plugin-syntax-class-static-block/-/plugin-syntax-class-static-block-7.14.5.tgz#195df89b146b4b78b3bf897fd7a257c84659d406" + integrity sha512-b+YyPmr6ldyNnM6sqYeMWE+bgJcJpO6yS4QD7ymxgH34GBPNDM/THBh8iunyvKIZztiwLH4CJZ0RxTk9emgpjw== + dependencies: + "@babel/helper-plugin-utils" "^7.14.5" + +"@babel/plugin-syntax-dynamic-import@^7.8.3": + version "7.8.3" + resolved "https://registry.yarnpkg.com/@babel/plugin-syntax-dynamic-import/-/plugin-syntax-dynamic-import-7.8.3.tgz#62bf98b2da3cd21d626154fc96ee5b3cb68eacb3" + integrity sha512-5gdGbFon+PszYzqs83S3E5mpi7/y/8M9eC90MRTZfduQOYW76ig6SOSPNe41IG5LoP3FGBn2N0RjVDSQiS94kQ== + dependencies: + "@babel/helper-plugin-utils" "^7.8.0" + +"@babel/plugin-syntax-export-namespace-from@^7.8.3": + version "7.8.3" + resolved "https://registry.yarnpkg.com/@babel/plugin-syntax-export-namespace-from/-/plugin-syntax-export-namespace-from-7.8.3.tgz#028964a9ba80dbc094c915c487ad7c4e7a66465a" + integrity sha512-MXf5laXo6c1IbEbegDmzGPwGNTsHZmEy6QGznu5Sh2UCWvueywb2ee+CCE4zQiZstxU9BMoQO9i6zUFSY0Kj0Q== + dependencies: + "@babel/helper-plugin-utils" "^7.8.3" + +"@babel/plugin-syntax-import-assertions@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/plugin-syntax-import-assertions/-/plugin-syntax-import-assertions-7.22.5.tgz#07d252e2aa0bc6125567f742cd58619cb14dce98" + integrity sha512-rdV97N7KqsRzeNGoWUOK6yUsWarLjE5Su/Snk9IYPU9CwkWHs4t+rTGOvffTR8XGkJMTAdLfO0xVnXm8wugIJg== + dependencies: + "@babel/helper-plugin-utils" "^7.22.5" + +"@babel/plugin-syntax-import-attributes@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/plugin-syntax-import-attributes/-/plugin-syntax-import-attributes-7.22.5.tgz#ab840248d834410b829f569f5262b9e517555ecb" + integrity sha512-KwvoWDeNKPETmozyFE0P2rOLqh39EoQHNjqizrI5B8Vt0ZNS7M56s7dAiAqbYfiAYOuIzIh96z3iR2ktgu3tEg== + dependencies: + "@babel/helper-plugin-utils" "^7.22.5" + +"@babel/plugin-syntax-import-meta@^7.10.4": + version "7.10.4" + resolved "https://registry.yarnpkg.com/@babel/plugin-syntax-import-meta/-/plugin-syntax-import-meta-7.10.4.tgz#ee601348c370fa334d2207be158777496521fd51" + integrity sha512-Yqfm+XDx0+Prh3VSeEQCPU81yC+JWZ2pDPFSS4ZdpfZhp4MkFMaDC1UqseovEKwSUpnIL7+vK+Clp7bfh0iD7g== + dependencies: + "@babel/helper-plugin-utils" "^7.10.4" + +"@babel/plugin-syntax-json-strings@^7.8.3": + version "7.8.3" + resolved "https://registry.yarnpkg.com/@babel/plugin-syntax-json-strings/-/plugin-syntax-json-strings-7.8.3.tgz#01ca21b668cd8218c9e640cb6dd88c5412b2c96a" + integrity sha512-lY6kdGpWHvjoe2vk4WrAapEuBR69EMxZl+RoGRhrFGNYVK8mOPAW8VfbT/ZgrFbXlDNiiaxQnAtgVCZ6jv30EA== + dependencies: + "@babel/helper-plugin-utils" "^7.8.0" + +"@babel/plugin-syntax-logical-assignment-operators@^7.10.4": + version "7.10.4" + resolved "https://registry.yarnpkg.com/@babel/plugin-syntax-logical-assignment-operators/-/plugin-syntax-logical-assignment-operators-7.10.4.tgz#ca91ef46303530448b906652bac2e9fe9941f699" + integrity sha512-d8waShlpFDinQ5MtvGU9xDAOzKH47+FFoney2baFIoMr952hKOLp1HR7VszoZvOsV/4+RRszNY7D17ba0te0ig== + dependencies: + "@babel/helper-plugin-utils" "^7.10.4" + +"@babel/plugin-syntax-nullish-coalescing-operator@^7.8.3": + version "7.8.3" + resolved "https://registry.yarnpkg.com/@babel/plugin-syntax-nullish-coalescing-operator/-/plugin-syntax-nullish-coalescing-operator-7.8.3.tgz#167ed70368886081f74b5c36c65a88c03b66d1a9" + integrity sha512-aSff4zPII1u2QD7y+F8oDsz19ew4IGEJg9SVW+bqwpwtfFleiQDMdzA/R+UlWDzfnHFCxxleFT0PMIrR36XLNQ== + dependencies: + "@babel/helper-plugin-utils" "^7.8.0" + +"@babel/plugin-syntax-numeric-separator@^7.10.4": + version "7.10.4" + resolved "https://registry.yarnpkg.com/@babel/plugin-syntax-numeric-separator/-/plugin-syntax-numeric-separator-7.10.4.tgz#b9b070b3e33570cd9fd07ba7fa91c0dd37b9af97" + integrity sha512-9H6YdfkcK/uOnY/K7/aA2xpzaAgkQn37yzWUMRK7OaPOqOpGS1+n0H5hxT9AUw9EsSjPW8SVyMJwYRtWs3X3ug== + dependencies: + "@babel/helper-plugin-utils" "^7.10.4" + +"@babel/plugin-syntax-object-rest-spread@^7.8.3": + version "7.8.3" + resolved "https://registry.yarnpkg.com/@babel/plugin-syntax-object-rest-spread/-/plugin-syntax-object-rest-spread-7.8.3.tgz#60e225edcbd98a640332a2e72dd3e66f1af55871" + integrity sha512-XoqMijGZb9y3y2XskN+P1wUGiVwWZ5JmoDRwx5+3GmEplNyVM2s2Dg8ILFQm8rWM48orGy5YpI5Bl8U1y7ydlA== + dependencies: + "@babel/helper-plugin-utils" "^7.8.0" + +"@babel/plugin-syntax-optional-catch-binding@^7.8.3": + version "7.8.3" + resolved "https://registry.yarnpkg.com/@babel/plugin-syntax-optional-catch-binding/-/plugin-syntax-optional-catch-binding-7.8.3.tgz#6111a265bcfb020eb9efd0fdfd7d26402b9ed6c1" + integrity sha512-6VPD0Pc1lpTqw0aKoeRTMiB+kWhAoT24PA+ksWSBrFtl5SIRVpZlwN3NNPQjehA2E/91FV3RjLWoVTglWcSV3Q== + dependencies: + "@babel/helper-plugin-utils" "^7.8.0" + +"@babel/plugin-syntax-optional-chaining@^7.8.3": + version "7.8.3" + resolved "https://registry.yarnpkg.com/@babel/plugin-syntax-optional-chaining/-/plugin-syntax-optional-chaining-7.8.3.tgz#4f69c2ab95167e0180cd5336613f8c5788f7d48a" + integrity sha512-KoK9ErH1MBlCPxV0VANkXW2/dw4vlbGDrFgz8bmUsBGYkFRcbRwMh6cIJubdPrkxRwuGdtCk0v/wPTKbQgBjkg== + dependencies: + "@babel/helper-plugin-utils" "^7.8.0" + +"@babel/plugin-syntax-private-property-in-object@^7.14.5": + version "7.14.5" + resolved "https://registry.yarnpkg.com/@babel/plugin-syntax-private-property-in-object/-/plugin-syntax-private-property-in-object-7.14.5.tgz#0dc6671ec0ea22b6e94a1114f857970cd39de1ad" + integrity sha512-0wVnp9dxJ72ZUJDV27ZfbSj6iHLoytYZmh3rFcxNnvsJF3ktkzLDZPy/mA17HGsaQT3/DQsWYX1f1QGWkCoVUg== + dependencies: + "@babel/helper-plugin-utils" "^7.14.5" + +"@babel/plugin-syntax-top-level-await@^7.14.5": + version "7.14.5" + resolved "https://registry.yarnpkg.com/@babel/plugin-syntax-top-level-await/-/plugin-syntax-top-level-await-7.14.5.tgz#c1cfdadc35a646240001f06138247b741c34d94c" + integrity sha512-hx++upLv5U1rgYfwe1xBQUhRmU41NEvpUvrp8jkrSCdvGSnM5/qdRMtylJ6PG5OFkBaHkbTAKTnd3/YyESRHFw== + dependencies: + "@babel/helper-plugin-utils" "^7.14.5" + +"@babel/plugin-syntax-unicode-sets-regex@^7.18.6": + version "7.18.6" + resolved "https://registry.yarnpkg.com/@babel/plugin-syntax-unicode-sets-regex/-/plugin-syntax-unicode-sets-regex-7.18.6.tgz#d49a3b3e6b52e5be6740022317580234a6a47357" + integrity sha512-727YkEAPwSIQTv5im8QHz3upqp92JTWhidIC81Tdx4VJYIte/VndKf1qKrfnnhPLiPghStWfvC/iFaMCQu7Nqg== + dependencies: + "@babel/helper-create-regexp-features-plugin" "^7.18.6" + "@babel/helper-plugin-utils" "^7.18.6" + +"@babel/plugin-transform-arrow-functions@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-arrow-functions/-/plugin-transform-arrow-functions-7.22.5.tgz#e5ba566d0c58a5b2ba2a8b795450641950b71958" + integrity sha512-26lTNXoVRdAnsaDXPpvCNUq+OVWEVC6bx7Vvz9rC53F2bagUWW4u4ii2+h8Fejfh7RYqPxn+libeFBBck9muEw== + dependencies: + "@babel/helper-plugin-utils" "^7.22.5" + +"@babel/plugin-transform-async-generator-functions@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-async-generator-functions/-/plugin-transform-async-generator-functions-7.22.5.tgz#7336356d23380eda9a56314974f053a020dab0c3" + integrity sha512-gGOEvFzm3fWoyD5uZq7vVTD57pPJ3PczPUD/xCFGjzBpUosnklmXyKnGQbbbGs1NPNPskFex0j93yKbHt0cHyg== + dependencies: + "@babel/helper-environment-visitor" "^7.22.5" + "@babel/helper-plugin-utils" "^7.22.5" + "@babel/helper-remap-async-to-generator" "^7.22.5" + "@babel/plugin-syntax-async-generators" "^7.8.4" + +"@babel/plugin-transform-async-to-generator@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-async-to-generator/-/plugin-transform-async-to-generator-7.22.5.tgz#c7a85f44e46f8952f6d27fe57c2ed3cc084c3775" + integrity sha512-b1A8D8ZzE/VhNDoV1MSJTnpKkCG5bJo+19R4o4oy03zM7ws8yEMK755j61Dc3EyvdysbqH5BOOTquJ7ZX9C6vQ== + dependencies: + "@babel/helper-module-imports" "^7.22.5" + "@babel/helper-plugin-utils" "^7.22.5" + "@babel/helper-remap-async-to-generator" "^7.22.5" + +"@babel/plugin-transform-block-scoped-functions@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-block-scoped-functions/-/plugin-transform-block-scoped-functions-7.22.5.tgz#27978075bfaeb9fa586d3cb63a3d30c1de580024" + integrity sha512-tdXZ2UdknEKQWKJP1KMNmuF5Lx3MymtMN/pvA+p/VEkhK8jVcQ1fzSy8KM9qRYhAf2/lV33hoMPKI/xaI9sADA== + dependencies: + "@babel/helper-plugin-utils" "^7.22.5" + +"@babel/plugin-transform-block-scoping@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-block-scoping/-/plugin-transform-block-scoping-7.22.5.tgz#8bfc793b3a4b2742c0983fadc1480d843ecea31b" + integrity sha512-EcACl1i5fSQ6bt+YGuU/XGCeZKStLmyVGytWkpyhCLeQVA0eu6Wtiw92V+I1T/hnezUv7j74dA/Ro69gWcU+hg== + dependencies: + "@babel/helper-plugin-utils" "^7.22.5" + +"@babel/plugin-transform-class-properties@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-class-properties/-/plugin-transform-class-properties-7.22.5.tgz#97a56e31ad8c9dc06a0b3710ce7803d5a48cca77" + integrity sha512-nDkQ0NfkOhPTq8YCLiWNxp1+f9fCobEjCb0n8WdbNUBc4IB5V7P1QnX9IjpSoquKrXF5SKojHleVNs2vGeHCHQ== + dependencies: + "@babel/helper-create-class-features-plugin" "^7.22.5" + "@babel/helper-plugin-utils" "^7.22.5" + +"@babel/plugin-transform-class-static-block@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-class-static-block/-/plugin-transform-class-static-block-7.22.5.tgz#3e40c46f048403472d6f4183116d5e46b1bff5ba" + integrity sha512-SPToJ5eYZLxlnp1UzdARpOGeC2GbHvr9d/UV0EukuVx8atktg194oe+C5BqQ8jRTkgLRVOPYeXRSBg1IlMoVRA== + dependencies: + "@babel/helper-create-class-features-plugin" "^7.22.5" + "@babel/helper-plugin-utils" "^7.22.5" + "@babel/plugin-syntax-class-static-block" "^7.14.5" + +"@babel/plugin-transform-classes@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-classes/-/plugin-transform-classes-7.22.5.tgz#635d4e98da741fad814984639f4c0149eb0135e1" + integrity sha512-2edQhLfibpWpsVBx2n/GKOz6JdGQvLruZQfGr9l1qes2KQaWswjBzhQF7UDUZMNaMMQeYnQzxwOMPsbYF7wqPQ== + dependencies: + "@babel/helper-annotate-as-pure" "^7.22.5" + "@babel/helper-compilation-targets" "^7.22.5" + "@babel/helper-environment-visitor" "^7.22.5" + "@babel/helper-function-name" "^7.22.5" + "@babel/helper-optimise-call-expression" "^7.22.5" + "@babel/helper-plugin-utils" "^7.22.5" + "@babel/helper-replace-supers" "^7.22.5" + "@babel/helper-split-export-declaration" "^7.22.5" + globals "^11.1.0" + +"@babel/plugin-transform-computed-properties@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-computed-properties/-/plugin-transform-computed-properties-7.22.5.tgz#cd1e994bf9f316bd1c2dafcd02063ec261bb3869" + integrity sha512-4GHWBgRf0krxPX+AaPtgBAlTgTeZmqDynokHOX7aqqAB4tHs3U2Y02zH6ETFdLZGcg9UQSD1WCmkVrE9ErHeOg== + dependencies: + "@babel/helper-plugin-utils" "^7.22.5" + "@babel/template" "^7.22.5" + +"@babel/plugin-transform-destructuring@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-destructuring/-/plugin-transform-destructuring-7.22.5.tgz#d3aca7438f6c26c78cdd0b0ba920a336001b27cc" + integrity sha512-GfqcFuGW8vnEqTUBM7UtPd5A4q797LTvvwKxXTgRsFjoqaJiEg9deBG6kWeQYkVEL569NpnmpC0Pkr/8BLKGnQ== + dependencies: + "@babel/helper-plugin-utils" "^7.22.5" + +"@babel/plugin-transform-dotall-regex@^7.22.5", "@babel/plugin-transform-dotall-regex@^7.4.4": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-dotall-regex/-/plugin-transform-dotall-regex-7.22.5.tgz#dbb4f0e45766eb544e193fb00e65a1dd3b2a4165" + integrity sha512-5/Yk9QxCQCl+sOIB1WelKnVRxTJDSAIxtJLL2/pqL14ZVlbH0fUQUZa/T5/UnQtBNgghR7mfB8ERBKyKPCi7Vw== + dependencies: + "@babel/helper-create-regexp-features-plugin" "^7.22.5" + "@babel/helper-plugin-utils" "^7.22.5" + +"@babel/plugin-transform-duplicate-keys@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-duplicate-keys/-/plugin-transform-duplicate-keys-7.22.5.tgz#b6e6428d9416f5f0bba19c70d1e6e7e0b88ab285" + integrity sha512-dEnYD+9BBgld5VBXHnF/DbYGp3fqGMsyxKbtD1mDyIA7AkTSpKXFhCVuj/oQVOoALfBs77DudA0BE4d5mcpmqw== + dependencies: + "@babel/helper-plugin-utils" "^7.22.5" + +"@babel/plugin-transform-dynamic-import@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-dynamic-import/-/plugin-transform-dynamic-import-7.22.5.tgz#d6908a8916a810468c4edff73b5b75bda6ad393e" + integrity sha512-0MC3ppTB1AMxd8fXjSrbPa7LT9hrImt+/fcj+Pg5YMD7UQyWp/02+JWpdnCymmsXwIx5Z+sYn1bwCn4ZJNvhqQ== + dependencies: + "@babel/helper-plugin-utils" "^7.22.5" + "@babel/plugin-syntax-dynamic-import" "^7.8.3" + +"@babel/plugin-transform-exponentiation-operator@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-exponentiation-operator/-/plugin-transform-exponentiation-operator-7.22.5.tgz#402432ad544a1f9a480da865fda26be653e48f6a" + integrity sha512-vIpJFNM/FjZ4rh1myqIya9jXwrwwgFRHPjT3DkUA9ZLHuzox8jiXkOLvwm1H+PQIP3CqfC++WPKeuDi0Sjdj1g== + dependencies: + "@babel/helper-builder-binary-assignment-operator-visitor" "^7.22.5" + "@babel/helper-plugin-utils" "^7.22.5" + +"@babel/plugin-transform-export-namespace-from@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-export-namespace-from/-/plugin-transform-export-namespace-from-7.22.5.tgz#57c41cb1d0613d22f548fddd8b288eedb9973a5b" + integrity sha512-X4hhm7FRnPgd4nDA4b/5V280xCx6oL7Oob5+9qVS5C13Zq4bh1qq7LU0GgRU6b5dBWBvhGaXYVB4AcN6+ol6vg== + dependencies: + "@babel/helper-plugin-utils" "^7.22.5" + "@babel/plugin-syntax-export-namespace-from" "^7.8.3" + +"@babel/plugin-transform-for-of@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-for-of/-/plugin-transform-for-of-7.22.5.tgz#ab1b8a200a8f990137aff9a084f8de4099ab173f" + integrity sha512-3kxQjX1dU9uudwSshyLeEipvrLjBCVthCgeTp6CzE/9JYrlAIaeekVxRpCWsDDfYTfRZRoCeZatCQvwo+wvK8A== + dependencies: + "@babel/helper-plugin-utils" "^7.22.5" + +"@babel/plugin-transform-function-name@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-function-name/-/plugin-transform-function-name-7.22.5.tgz#935189af68b01898e0d6d99658db6b164205c143" + integrity sha512-UIzQNMS0p0HHiQm3oelztj+ECwFnj+ZRV4KnguvlsD2of1whUeM6o7wGNj6oLwcDoAXQ8gEqfgC24D+VdIcevg== + dependencies: + "@babel/helper-compilation-targets" "^7.22.5" + "@babel/helper-function-name" "^7.22.5" + "@babel/helper-plugin-utils" "^7.22.5" + +"@babel/plugin-transform-json-strings@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-json-strings/-/plugin-transform-json-strings-7.22.5.tgz#14b64352fdf7e1f737eed68de1a1468bd2a77ec0" + integrity sha512-DuCRB7fu8MyTLbEQd1ew3R85nx/88yMoqo2uPSjevMj3yoN7CDM8jkgrY0wmVxfJZyJ/B9fE1iq7EQppWQmR5A== + dependencies: + "@babel/helper-plugin-utils" "^7.22.5" + "@babel/plugin-syntax-json-strings" "^7.8.3" + +"@babel/plugin-transform-literals@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-literals/-/plugin-transform-literals-7.22.5.tgz#e9341f4b5a167952576e23db8d435849b1dd7920" + integrity sha512-fTLj4D79M+mepcw3dgFBTIDYpbcB9Sm0bpm4ppXPaO+U+PKFFyV9MGRvS0gvGw62sd10kT5lRMKXAADb9pWy8g== + dependencies: + "@babel/helper-plugin-utils" "^7.22.5" + +"@babel/plugin-transform-logical-assignment-operators@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-logical-assignment-operators/-/plugin-transform-logical-assignment-operators-7.22.5.tgz#66ae5f068fd5a9a5dc570df16f56c2a8462a9d6c" + integrity sha512-MQQOUW1KL8X0cDWfbwYP+TbVbZm16QmQXJQ+vndPtH/BoO0lOKpVoEDMI7+PskYxH+IiE0tS8xZye0qr1lGzSA== + dependencies: + "@babel/helper-plugin-utils" "^7.22.5" + "@babel/plugin-syntax-logical-assignment-operators" "^7.10.4" + +"@babel/plugin-transform-member-expression-literals@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-member-expression-literals/-/plugin-transform-member-expression-literals-7.22.5.tgz#4fcc9050eded981a468347dd374539ed3e058def" + integrity sha512-RZEdkNtzzYCFl9SE9ATaUMTj2hqMb4StarOJLrZRbqqU4HSBE7UlBw9WBWQiDzrJZJdUWiMTVDI6Gv/8DPvfew== + dependencies: + "@babel/helper-plugin-utils" "^7.22.5" + +"@babel/plugin-transform-modules-amd@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-modules-amd/-/plugin-transform-modules-amd-7.22.5.tgz#4e045f55dcf98afd00f85691a68fc0780704f526" + integrity sha512-R+PTfLTcYEmb1+kK7FNkhQ1gP4KgjpSO6HfH9+f8/yfp2Nt3ggBjiVpRwmwTlfqZLafYKJACy36yDXlEmI9HjQ== + dependencies: + "@babel/helper-module-transforms" "^7.22.5" + "@babel/helper-plugin-utils" "^7.22.5" + +"@babel/plugin-transform-modules-commonjs@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-modules-commonjs/-/plugin-transform-modules-commonjs-7.22.5.tgz#7d9875908d19b8c0536085af7b053fd5bd651bfa" + integrity sha512-B4pzOXj+ONRmuaQTg05b3y/4DuFz3WcCNAXPLb2Q0GT0TrGKGxNKV4jwsXts+StaM0LQczZbOpj8o1DLPDJIiA== + dependencies: + "@babel/helper-module-transforms" "^7.22.5" + "@babel/helper-plugin-utils" "^7.22.5" + "@babel/helper-simple-access" "^7.22.5" + +"@babel/plugin-transform-modules-systemjs@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-modules-systemjs/-/plugin-transform-modules-systemjs-7.22.5.tgz#18c31410b5e579a0092638f95c896c2a98a5d496" + integrity sha512-emtEpoaTMsOs6Tzz+nbmcePl6AKVtS1yC4YNAeMun9U8YCsgadPNxnOPQ8GhHFB2qdx+LZu9LgoC0Lthuu05DQ== + dependencies: + "@babel/helper-hoist-variables" "^7.22.5" + "@babel/helper-module-transforms" "^7.22.5" + "@babel/helper-plugin-utils" "^7.22.5" + "@babel/helper-validator-identifier" "^7.22.5" + +"@babel/plugin-transform-modules-umd@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-modules-umd/-/plugin-transform-modules-umd-7.22.5.tgz#4694ae40a87b1745e3775b6a7fe96400315d4f98" + integrity sha512-+S6kzefN/E1vkSsKx8kmQuqeQsvCKCd1fraCM7zXm4SFoggI099Tr4G8U81+5gtMdUeMQ4ipdQffbKLX0/7dBQ== + dependencies: + "@babel/helper-module-transforms" "^7.22.5" + "@babel/helper-plugin-utils" "^7.22.5" + +"@babel/plugin-transform-named-capturing-groups-regex@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-named-capturing-groups-regex/-/plugin-transform-named-capturing-groups-regex-7.22.5.tgz#67fe18ee8ce02d57c855185e27e3dc959b2e991f" + integrity sha512-YgLLKmS3aUBhHaxp5hi1WJTgOUb/NCuDHzGT9z9WTt3YG+CPRhJs6nprbStx6DnWM4dh6gt7SU3sZodbZ08adQ== + dependencies: + "@babel/helper-create-regexp-features-plugin" "^7.22.5" + "@babel/helper-plugin-utils" "^7.22.5" + +"@babel/plugin-transform-new-target@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-new-target/-/plugin-transform-new-target-7.22.5.tgz#1b248acea54ce44ea06dfd37247ba089fcf9758d" + integrity sha512-AsF7K0Fx/cNKVyk3a+DW0JLo+Ua598/NxMRvxDnkpCIGFh43+h/v2xyhRUYf6oD8gE4QtL83C7zZVghMjHd+iw== + dependencies: + "@babel/helper-plugin-utils" "^7.22.5" + +"@babel/plugin-transform-nullish-coalescing-operator@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-nullish-coalescing-operator/-/plugin-transform-nullish-coalescing-operator-7.22.5.tgz#f8872c65776e0b552e0849d7596cddd416c3e381" + integrity sha512-6CF8g6z1dNYZ/VXok5uYkkBBICHZPiGEl7oDnAx2Mt1hlHVHOSIKWJaXHjQJA5VB43KZnXZDIexMchY4y2PGdA== + dependencies: + "@babel/helper-plugin-utils" "^7.22.5" + "@babel/plugin-syntax-nullish-coalescing-operator" "^7.8.3" + +"@babel/plugin-transform-numeric-separator@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-numeric-separator/-/plugin-transform-numeric-separator-7.22.5.tgz#57226a2ed9e512b9b446517ab6fa2d17abb83f58" + integrity sha512-NbslED1/6M+sXiwwtcAB/nieypGw02Ejf4KtDeMkCEpP6gWFMX1wI9WKYua+4oBneCCEmulOkRpwywypVZzs/g== + dependencies: + "@babel/helper-plugin-utils" "^7.22.5" + "@babel/plugin-syntax-numeric-separator" "^7.10.4" + +"@babel/plugin-transform-object-rest-spread@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-object-rest-spread/-/plugin-transform-object-rest-spread-7.22.5.tgz#9686dc3447df4753b0b2a2fae7e8bc33cdc1f2e1" + integrity sha512-Kk3lyDmEslH9DnvCDA1s1kkd3YWQITiBOHngOtDL9Pt6BZjzqb6hiOlb8VfjiiQJ2unmegBqZu0rx5RxJb5vmQ== + dependencies: + "@babel/compat-data" "^7.22.5" + "@babel/helper-compilation-targets" "^7.22.5" + "@babel/helper-plugin-utils" "^7.22.5" + "@babel/plugin-syntax-object-rest-spread" "^7.8.3" + "@babel/plugin-transform-parameters" "^7.22.5" + +"@babel/plugin-transform-object-super@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-object-super/-/plugin-transform-object-super-7.22.5.tgz#794a8d2fcb5d0835af722173c1a9d704f44e218c" + integrity sha512-klXqyaT9trSjIUrcsYIfETAzmOEZL3cBYqOYLJxBHfMFFggmXOv+NYSX/Jbs9mzMVESw/WycLFPRx8ba/b2Ipw== + dependencies: + "@babel/helper-plugin-utils" "^7.22.5" + "@babel/helper-replace-supers" "^7.22.5" + +"@babel/plugin-transform-optional-catch-binding@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-optional-catch-binding/-/plugin-transform-optional-catch-binding-7.22.5.tgz#842080be3076703be0eaf32ead6ac8174edee333" + integrity sha512-pH8orJahy+hzZje5b8e2QIlBWQvGpelS76C63Z+jhZKsmzfNaPQ+LaW6dcJ9bxTpo1mtXbgHwy765Ro3jftmUg== + dependencies: + "@babel/helper-plugin-utils" "^7.22.5" + "@babel/plugin-syntax-optional-catch-binding" "^7.8.3" + +"@babel/plugin-transform-optional-chaining@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-optional-chaining/-/plugin-transform-optional-chaining-7.22.5.tgz#1003762b9c14295501beb41be72426736bedd1e0" + integrity sha512-AconbMKOMkyG+xCng2JogMCDcqW8wedQAqpVIL4cOSescZ7+iW8utC6YDZLMCSUIReEA733gzRSaOSXMAt/4WQ== + dependencies: + "@babel/helper-plugin-utils" "^7.22.5" + "@babel/helper-skip-transparent-expression-wrappers" "^7.22.5" + "@babel/plugin-syntax-optional-chaining" "^7.8.3" + +"@babel/plugin-transform-parameters@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-parameters/-/plugin-transform-parameters-7.22.5.tgz#c3542dd3c39b42c8069936e48717a8d179d63a18" + integrity sha512-AVkFUBurORBREOmHRKo06FjHYgjrabpdqRSwq6+C7R5iTCZOsM4QbcB27St0a4U6fffyAOqh3s/qEfybAhfivg== + dependencies: + "@babel/helper-plugin-utils" "^7.22.5" + +"@babel/plugin-transform-private-methods@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-private-methods/-/plugin-transform-private-methods-7.22.5.tgz#21c8af791f76674420a147ae62e9935d790f8722" + integrity sha512-PPjh4gyrQnGe97JTalgRGMuU4icsZFnWkzicB/fUtzlKUqvsWBKEpPPfr5a2JiyirZkHxnAqkQMO5Z5B2kK3fA== + dependencies: + "@babel/helper-create-class-features-plugin" "^7.22.5" + "@babel/helper-plugin-utils" "^7.22.5" + +"@babel/plugin-transform-private-property-in-object@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-private-property-in-object/-/plugin-transform-private-property-in-object-7.22.5.tgz#07a77f28cbb251546a43d175a1dda4cf3ef83e32" + integrity sha512-/9xnaTTJcVoBtSSmrVyhtSvO3kbqS2ODoh2juEU72c3aYonNF0OMGiaz2gjukyKM2wBBYJP38S4JiE0Wfb5VMQ== + dependencies: + "@babel/helper-annotate-as-pure" "^7.22.5" + "@babel/helper-create-class-features-plugin" "^7.22.5" + "@babel/helper-plugin-utils" "^7.22.5" + "@babel/plugin-syntax-private-property-in-object" "^7.14.5" + +"@babel/plugin-transform-property-literals@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-property-literals/-/plugin-transform-property-literals-7.22.5.tgz#b5ddabd73a4f7f26cd0e20f5db48290b88732766" + integrity sha512-TiOArgddK3mK/x1Qwf5hay2pxI6wCZnvQqrFSqbtg1GLl2JcNMitVH/YnqjP+M31pLUeTfzY1HAXFDnUBV30rQ== + dependencies: + "@babel/helper-plugin-utils" "^7.22.5" + +"@babel/plugin-transform-regenerator@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-regenerator/-/plugin-transform-regenerator-7.22.5.tgz#cd8a68b228a5f75fa01420e8cc2fc400f0fc32aa" + integrity sha512-rR7KePOE7gfEtNTh9Qw+iO3Q/e4DEsoQ+hdvM6QUDH7JRJ5qxq5AA52ZzBWbI5i9lfNuvySgOGP8ZN7LAmaiPw== + dependencies: + "@babel/helper-plugin-utils" "^7.22.5" + regenerator-transform "^0.15.1" + +"@babel/plugin-transform-reserved-words@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-reserved-words/-/plugin-transform-reserved-words-7.22.5.tgz#832cd35b81c287c4bcd09ce03e22199641f964fb" + integrity sha512-DTtGKFRQUDm8svigJzZHzb/2xatPc6TzNvAIJ5GqOKDsGFYgAskjRulbR/vGsPKq3OPqtexnz327qYpP57RFyA== + dependencies: + "@babel/helper-plugin-utils" "^7.22.5" + +"@babel/plugin-transform-shorthand-properties@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-shorthand-properties/-/plugin-transform-shorthand-properties-7.22.5.tgz#6e277654be82b5559fc4b9f58088507c24f0c624" + integrity sha512-vM4fq9IXHscXVKzDv5itkO1X52SmdFBFcMIBZ2FRn2nqVYqw6dBexUgMvAjHW+KXpPPViD/Yo3GrDEBaRC0QYA== + dependencies: + "@babel/helper-plugin-utils" "^7.22.5" + +"@babel/plugin-transform-spread@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-spread/-/plugin-transform-spread-7.22.5.tgz#6487fd29f229c95e284ba6c98d65eafb893fea6b" + integrity sha512-5ZzDQIGyvN4w8+dMmpohL6MBo+l2G7tfC/O2Dg7/hjpgeWvUx8FzfeOKxGog9IimPa4YekaQ9PlDqTLOljkcxg== + dependencies: + "@babel/helper-plugin-utils" "^7.22.5" + "@babel/helper-skip-transparent-expression-wrappers" "^7.22.5" + +"@babel/plugin-transform-sticky-regex@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-sticky-regex/-/plugin-transform-sticky-regex-7.22.5.tgz#295aba1595bfc8197abd02eae5fc288c0deb26aa" + integrity sha512-zf7LuNpHG0iEeiyCNwX4j3gDg1jgt1k3ZdXBKbZSoA3BbGQGvMiSvfbZRR3Dr3aeJe3ooWFZxOOG3IRStYp2Bw== + dependencies: + "@babel/helper-plugin-utils" "^7.22.5" + +"@babel/plugin-transform-template-literals@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-template-literals/-/plugin-transform-template-literals-7.22.5.tgz#8f38cf291e5f7a8e60e9f733193f0bcc10909bff" + integrity sha512-5ciOehRNf+EyUeewo8NkbQiUs4d6ZxiHo6BcBcnFlgiJfu16q0bQUw9Jvo0b0gBKFG1SMhDSjeKXSYuJLeFSMA== + dependencies: + "@babel/helper-plugin-utils" "^7.22.5" + +"@babel/plugin-transform-typeof-symbol@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-typeof-symbol/-/plugin-transform-typeof-symbol-7.22.5.tgz#5e2ba478da4b603af8673ff7c54f75a97b716b34" + integrity sha512-bYkI5lMzL4kPii4HHEEChkD0rkc+nvnlR6+o/qdqR6zrm0Sv/nodmyLhlq2DO0YKLUNd2VePmPRjJXSBh9OIdA== + dependencies: + "@babel/helper-plugin-utils" "^7.22.5" + +"@babel/plugin-transform-unicode-escapes@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-unicode-escapes/-/plugin-transform-unicode-escapes-7.22.5.tgz#ce0c248522b1cb22c7c992d88301a5ead70e806c" + integrity sha512-biEmVg1IYB/raUO5wT1tgfacCef15Fbzhkx493D3urBI++6hpJ+RFG4SrWMn0NEZLfvilqKf3QDrRVZHo08FYg== + dependencies: + "@babel/helper-plugin-utils" "^7.22.5" + +"@babel/plugin-transform-unicode-property-regex@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-unicode-property-regex/-/plugin-transform-unicode-property-regex-7.22.5.tgz#098898f74d5c1e86660dc112057b2d11227f1c81" + integrity sha512-HCCIb+CbJIAE6sXn5CjFQXMwkCClcOfPCzTlilJ8cUatfzwHlWQkbtV0zD338u9dZskwvuOYTuuaMaA8J5EI5A== + dependencies: + "@babel/helper-create-regexp-features-plugin" "^7.22.5" + "@babel/helper-plugin-utils" "^7.22.5" + +"@babel/plugin-transform-unicode-regex@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-unicode-regex/-/plugin-transform-unicode-regex-7.22.5.tgz#ce7e7bb3ef208c4ff67e02a22816656256d7a183" + integrity sha512-028laaOKptN5vHJf9/Arr/HiJekMd41hOEZYvNsrsXqJ7YPYuX2bQxh31fkZzGmq3YqHRJzYFFAVYvKfMPKqyg== + dependencies: + "@babel/helper-create-regexp-features-plugin" "^7.22.5" + "@babel/helper-plugin-utils" "^7.22.5" + +"@babel/plugin-transform-unicode-sets-regex@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-unicode-sets-regex/-/plugin-transform-unicode-sets-regex-7.22.5.tgz#77788060e511b708ffc7d42fdfbc5b37c3004e91" + integrity sha512-lhMfi4FC15j13eKrh3DnYHjpGj6UKQHtNKTbtc1igvAhRy4+kLhV07OpLcsN0VgDEw/MjAvJO4BdMJsHwMhzCg== + dependencies: + "@babel/helper-create-regexp-features-plugin" "^7.22.5" + "@babel/helper-plugin-utils" "^7.22.5" + +"@babel/preset-env@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/preset-env/-/preset-env-7.22.5.tgz#3da66078b181f3d62512c51cf7014392c511504e" + integrity sha512-fj06hw89dpiZzGZtxn+QybifF07nNiZjZ7sazs2aVDcysAZVGjW7+7iFYxg6GLNM47R/thYfLdrXc+2f11Vi9A== + dependencies: + "@babel/compat-data" "^7.22.5" + "@babel/helper-compilation-targets" "^7.22.5" + "@babel/helper-plugin-utils" "^7.22.5" + "@babel/helper-validator-option" "^7.22.5" + "@babel/plugin-bugfix-safari-id-destructuring-collision-in-function-expression" "^7.22.5" + "@babel/plugin-bugfix-v8-spread-parameters-in-optional-chaining" "^7.22.5" + "@babel/plugin-proposal-private-property-in-object" "7.21.0-placeholder-for-preset-env.2" + "@babel/plugin-syntax-async-generators" "^7.8.4" + "@babel/plugin-syntax-class-properties" "^7.12.13" + "@babel/plugin-syntax-class-static-block" "^7.14.5" + "@babel/plugin-syntax-dynamic-import" "^7.8.3" + "@babel/plugin-syntax-export-namespace-from" "^7.8.3" + "@babel/plugin-syntax-import-assertions" "^7.22.5" + "@babel/plugin-syntax-import-attributes" "^7.22.5" + "@babel/plugin-syntax-import-meta" "^7.10.4" + "@babel/plugin-syntax-json-strings" "^7.8.3" + "@babel/plugin-syntax-logical-assignment-operators" "^7.10.4" + "@babel/plugin-syntax-nullish-coalescing-operator" "^7.8.3" + "@babel/plugin-syntax-numeric-separator" "^7.10.4" + "@babel/plugin-syntax-object-rest-spread" "^7.8.3" + "@babel/plugin-syntax-optional-catch-binding" "^7.8.3" + "@babel/plugin-syntax-optional-chaining" "^7.8.3" + "@babel/plugin-syntax-private-property-in-object" "^7.14.5" + "@babel/plugin-syntax-top-level-await" "^7.14.5" + "@babel/plugin-syntax-unicode-sets-regex" "^7.18.6" + "@babel/plugin-transform-arrow-functions" "^7.22.5" + "@babel/plugin-transform-async-generator-functions" "^7.22.5" + "@babel/plugin-transform-async-to-generator" "^7.22.5" + "@babel/plugin-transform-block-scoped-functions" "^7.22.5" + "@babel/plugin-transform-block-scoping" "^7.22.5" + "@babel/plugin-transform-class-properties" "^7.22.5" + "@babel/plugin-transform-class-static-block" "^7.22.5" + "@babel/plugin-transform-classes" "^7.22.5" + "@babel/plugin-transform-computed-properties" "^7.22.5" + "@babel/plugin-transform-destructuring" "^7.22.5" + "@babel/plugin-transform-dotall-regex" "^7.22.5" + "@babel/plugin-transform-duplicate-keys" "^7.22.5" + "@babel/plugin-transform-dynamic-import" "^7.22.5" + "@babel/plugin-transform-exponentiation-operator" "^7.22.5" + "@babel/plugin-transform-export-namespace-from" "^7.22.5" + "@babel/plugin-transform-for-of" "^7.22.5" + "@babel/plugin-transform-function-name" "^7.22.5" + "@babel/plugin-transform-json-strings" "^7.22.5" + "@babel/plugin-transform-literals" "^7.22.5" + "@babel/plugin-transform-logical-assignment-operators" "^7.22.5" + "@babel/plugin-transform-member-expression-literals" "^7.22.5" + "@babel/plugin-transform-modules-amd" "^7.22.5" + "@babel/plugin-transform-modules-commonjs" "^7.22.5" + "@babel/plugin-transform-modules-systemjs" "^7.22.5" + "@babel/plugin-transform-modules-umd" "^7.22.5" + "@babel/plugin-transform-named-capturing-groups-regex" "^7.22.5" + "@babel/plugin-transform-new-target" "^7.22.5" + "@babel/plugin-transform-nullish-coalescing-operator" "^7.22.5" + "@babel/plugin-transform-numeric-separator" "^7.22.5" + "@babel/plugin-transform-object-rest-spread" "^7.22.5" + "@babel/plugin-transform-object-super" "^7.22.5" + "@babel/plugin-transform-optional-catch-binding" "^7.22.5" + "@babel/plugin-transform-optional-chaining" "^7.22.5" + "@babel/plugin-transform-parameters" "^7.22.5" + "@babel/plugin-transform-private-methods" "^7.22.5" + "@babel/plugin-transform-private-property-in-object" "^7.22.5" + "@babel/plugin-transform-property-literals" "^7.22.5" + "@babel/plugin-transform-regenerator" "^7.22.5" + "@babel/plugin-transform-reserved-words" "^7.22.5" + "@babel/plugin-transform-shorthand-properties" "^7.22.5" + "@babel/plugin-transform-spread" "^7.22.5" + "@babel/plugin-transform-sticky-regex" "^7.22.5" + "@babel/plugin-transform-template-literals" "^7.22.5" + "@babel/plugin-transform-typeof-symbol" "^7.22.5" + "@babel/plugin-transform-unicode-escapes" "^7.22.5" + "@babel/plugin-transform-unicode-property-regex" "^7.22.5" + "@babel/plugin-transform-unicode-regex" "^7.22.5" + "@babel/plugin-transform-unicode-sets-regex" "^7.22.5" + "@babel/preset-modules" "^0.1.5" + "@babel/types" "^7.22.5" + babel-plugin-polyfill-corejs2 "^0.4.3" + babel-plugin-polyfill-corejs3 "^0.8.1" + babel-plugin-polyfill-regenerator "^0.5.0" + core-js-compat "^3.30.2" + semver "^6.3.0" + +"@babel/preset-modules@^0.1.5": + version "0.1.5" + resolved "https://registry.yarnpkg.com/@babel/preset-modules/-/preset-modules-0.1.5.tgz#ef939d6e7f268827e1841638dc6ff95515e115d9" + integrity sha512-A57th6YRG7oR3cq/yt/Y84MvGgE0eJG2F1JLhKuyG+jFxEgrd/HAMJatiFtmOiZurz+0DkrvbheCLaV5f2JfjA== + dependencies: + "@babel/helper-plugin-utils" "^7.0.0" + "@babel/plugin-proposal-unicode-property-regex" "^7.4.4" + "@babel/plugin-transform-dotall-regex" "^7.4.4" + "@babel/types" "^7.4.4" + esutils "^2.0.2" + +"@babel/regjsgen@^0.8.0": + version "0.8.0" + resolved "https://registry.yarnpkg.com/@babel/regjsgen/-/regjsgen-0.8.0.tgz#f0ba69b075e1f05fb2825b7fad991e7adbb18310" + integrity sha512-x/rqGMdzj+fWZvCOYForTghzbtqPDZ5gPwaoNGHdgDfF2QA/XZbCBp4Moo5scrkAMPhB7z26XM/AaHuIJdgauA== + +"@babel/runtime@^7.8.4": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/runtime/-/runtime-7.22.5.tgz#8564dd588182ce0047d55d7a75e93921107b57ec" + integrity sha512-ecjvYlnAaZ/KVneE/OdKYBYfgXV3Ptu6zQWmgEF7vwKhQnvVS6bjMD2XYgj+SNvQ1GfK/pjgokfPkC/2CO8CuA== + dependencies: + regenerator-runtime "^0.13.11" + +"@babel/template@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/template/-/template-7.22.5.tgz#0c8c4d944509875849bd0344ff0050756eefc6ec" + integrity sha512-X7yV7eiwAxdj9k94NEylvbVHLiVG1nvzCV2EAowhxLTwODV1jl9UzZ48leOC0sH7OnuHrIkllaBgneUykIcZaw== + dependencies: + "@babel/code-frame" "^7.22.5" + "@babel/parser" "^7.22.5" + "@babel/types" "^7.22.5" + +"@babel/traverse@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/traverse/-/traverse-7.22.5.tgz#44bd276690db6f4940fdb84e1cb4abd2f729ccd1" + integrity sha512-7DuIjPgERaNo6r+PZwItpjCZEa5vyw4eJGufeLxrPdBXBoLcCJCIasvK6pK/9DVNrLZTLFhUGqaC6X/PA007TQ== + dependencies: + "@babel/code-frame" "^7.22.5" + "@babel/generator" "^7.22.5" + "@babel/helper-environment-visitor" "^7.22.5" + "@babel/helper-function-name" "^7.22.5" + "@babel/helper-hoist-variables" "^7.22.5" + "@babel/helper-split-export-declaration" "^7.22.5" + "@babel/parser" "^7.22.5" + "@babel/types" "^7.22.5" + debug "^4.1.0" + globals "^11.1.0" + +"@babel/types@^7.22.5", "@babel/types@^7.4.4": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/types/-/types-7.22.5.tgz#cd93eeaab025880a3a47ec881f4b096a5b786fbe" + integrity sha512-zo3MIHGOkPOfoRXitsgHLjEXmlDaD/5KU1Uzuc9GNiZPhSqVxVRtxuPaSBZDsYZ9qV88AjtMtWW7ww98loJ9KA== + dependencies: + "@babel/helper-string-parser" "^7.22.5" + "@babel/helper-validator-identifier" "^7.22.5" + to-fast-properties "^2.0.0" + +"@discoveryjs/json-ext@^0.5.0": + version "0.5.7" + resolved "https://registry.yarnpkg.com/@discoveryjs/json-ext/-/json-ext-0.5.7.tgz#1d572bfbbe14b7704e0ba0f39b74815b84870d70" + integrity sha512-dBVuXR082gk3jsFp7Rd/JI4kytwGHecnCoTtXFb7DB6CNHp4rg5k1bhg0nWdLGLnOV71lmDzGQaLMy8iPLY0pw== + +"@eslint-community/eslint-utils@^4.2.0": + version "4.4.0" + resolved "https://registry.yarnpkg.com/@eslint-community/eslint-utils/-/eslint-utils-4.4.0.tgz#a23514e8fb9af1269d5f7788aa556798d61c6b59" + integrity sha512-1/sA4dwrzBAyeUoQ6oxahHKmrZvsnLCg4RfxW3ZFGGmQkSNQPFNLV9CUEFQP1x9EYXHTo5p6xdhZM1Ne9p/AfA== + dependencies: + eslint-visitor-keys "^3.3.0" + +"@eslint-community/regexpp@^4.4.0": + version "4.5.1" + resolved "https://registry.yarnpkg.com/@eslint-community/regexpp/-/regexpp-4.5.1.tgz#cdd35dce4fa1a89a4fd42b1599eb35b3af408884" + integrity sha512-Z5ba73P98O1KUYCCJTUeVpja9RcGoMdncZ6T49FCUl2lN38JtCJ+3WgIDBv0AuY4WChU5PmtJmOCTlN6FZTFKQ== + +"@eslint/eslintrc@^2.0.3": + version "2.0.3" + resolved "https://registry.yarnpkg.com/@eslint/eslintrc/-/eslintrc-2.0.3.tgz#4910db5505f4d503f27774bf356e3704818a0331" + integrity sha512-+5gy6OQfk+xx3q0d6jGZZC3f3KzAkXc/IanVxd1is/VIIziRqqt3ongQz0FiTUXqTk0c7aDB3OaFuKnuSoJicQ== + dependencies: + ajv "^6.12.4" + debug "^4.3.2" + espree "^9.5.2" + globals "^13.19.0" + ignore "^5.2.0" + import-fresh "^3.2.1" + js-yaml "^4.1.0" + minimatch "^3.1.2" + strip-json-comments "^3.1.1" + +"@eslint/js@8.43.0": + version "8.43.0" + resolved "https://registry.yarnpkg.com/@eslint/js/-/js-8.43.0.tgz#559ca3d9ddbd6bf907ad524320a0d14b85586af0" + integrity sha512-s2UHCoiXfxMvmfzqoN+vrQ84ahUSYde9qNO1MdxmoEhyHWsfmwOpFlwYV+ePJEVc7gFnATGUi376WowX1N7tFg== + +"@humanwhocodes/config-array@^0.11.10": + version "0.11.10" + resolved "https://registry.yarnpkg.com/@humanwhocodes/config-array/-/config-array-0.11.10.tgz#5a3ffe32cc9306365fb3fd572596cd602d5e12d2" + integrity sha512-KVVjQmNUepDVGXNuoRRdmmEjruj0KfiGSbS8LVc12LMsWDQzRXJ0qdhN8L8uUigKpfEHRhlaQFY0ib1tnUbNeQ== + dependencies: + "@humanwhocodes/object-schema" "^1.2.1" + debug "^4.1.1" + minimatch "^3.0.5" + +"@humanwhocodes/module-importer@^1.0.1": + version "1.0.1" + resolved "https://registry.yarnpkg.com/@humanwhocodes/module-importer/-/module-importer-1.0.1.tgz#af5b2691a22b44be847b0ca81641c5fb6ad0172c" + integrity sha512-bxveV4V8v5Yb4ncFTT3rPSgZBOpCkjfK0y4oVVVJwIuDVBRMDXrPyXRL988i5ap9m9bnyEEjWfm5WkBmtffLfA== + +"@humanwhocodes/object-schema@^1.2.1": + version "1.2.1" + resolved "https://registry.yarnpkg.com/@humanwhocodes/object-schema/-/object-schema-1.2.1.tgz#b520529ec21d8e5945a1851dfd1c32e94e39ff45" + integrity sha512-ZnQMnLV4e7hDlUvw8H+U8ASL02SS2Gn6+9Ac3wGGLIe7+je2AeAOxPY+izIPJDfFDb7eDjev0Us8MO1iFRN8hA== + +"@jridgewell/gen-mapping@^0.3.0", "@jridgewell/gen-mapping@^0.3.2": + version "0.3.3" + resolved "https://registry.yarnpkg.com/@jridgewell/gen-mapping/-/gen-mapping-0.3.3.tgz#7e02e6eb5df901aaedb08514203b096614024098" + integrity sha512-HLhSWOLRi875zjjMG/r+Nv0oCW8umGb0BgEhyX3dDX3egwZtB8PqLnjz3yedt8R5StBrzcg4aBpnh8UA9D1BoQ== + dependencies: + "@jridgewell/set-array" "^1.0.1" + "@jridgewell/sourcemap-codec" "^1.4.10" + "@jridgewell/trace-mapping" "^0.3.9" + +"@jridgewell/resolve-uri@3.1.0": + version "3.1.0" + resolved "https://registry.yarnpkg.com/@jridgewell/resolve-uri/-/resolve-uri-3.1.0.tgz#2203b118c157721addfe69d47b70465463066d78" + integrity sha512-F2msla3tad+Mfht5cJq7LSXcdudKTWCVYUgw6pLFOOHSTtZlj6SWNYAp+AhuqLmWdBO2X5hPrLcu8cVP8fy28w== + +"@jridgewell/set-array@^1.0.1": + version "1.1.2" + resolved "https://registry.yarnpkg.com/@jridgewell/set-array/-/set-array-1.1.2.tgz#7c6cf998d6d20b914c0a55a91ae928ff25965e72" + integrity sha512-xnkseuNADM0gt2bs+BvhO0p78Mk762YnZdsuzFV018NoG1Sj1SCQvpSqa7XUaTam5vAGasABV9qXASMKnFMwMw== + +"@jridgewell/source-map@^0.3.3": + version "0.3.4" + resolved "https://registry.yarnpkg.com/@jridgewell/source-map/-/source-map-0.3.4.tgz#856a142864530d4059dda415659b48d37db2d556" + integrity sha512-KE/SxsDqNs3rrWwFHcRh15ZLVFrI0YoZtgAdIyIq9k5hUNmiWRXXThPomIxHuL20sLdgzbDFyvkUMna14bvtrw== + +"@jridgewell/sourcemap-codec@1.4.14": + version "1.4.14" + resolved "https://registry.yarnpkg.com/@jridgewell/sourcemap-codec/-/sourcemap-codec-1.4.14.tgz#add4c98d341472a289190b424efbdb096991bb24" + integrity sha512-XPSJHWmi394fuUuzDnGz1wiKqWfo1yXecHQMRf2l6hztTO+nPru658AyDngaBe7isIxEkRsPR3FZh+s7iVa4Uw== + +"@jridgewell/sourcemap-codec@^1.4.10": + version "1.4.15" + resolved "https://registry.yarnpkg.com/@jridgewell/sourcemap-codec/-/sourcemap-codec-1.4.15.tgz#d7c6e6755c78567a951e04ab52ef0fd26de59f32" + integrity sha512-eF2rxCRulEKXHTRiDrDy6erMYWqNw4LPdQ8UQA4huuxaQsVeRPFl2oM8oDGxMFhJUWZf9McpLtJasDDZb/Bpeg== + +"@jridgewell/trace-mapping@^0.3.17", "@jridgewell/trace-mapping@^0.3.9": + version "0.3.18" + resolved "https://registry.yarnpkg.com/@jridgewell/trace-mapping/-/trace-mapping-0.3.18.tgz#25783b2086daf6ff1dcb53c9249ae480e4dd4cd6" + integrity sha512-w+niJYzMHdd7USdiH2U6869nqhD2nbfZXND5Yp93qIbEmnDNk7PD48o+YchRVpzMU7M6jVCbenTR7PA1FLQ9pA== + dependencies: + "@jridgewell/resolve-uri" "3.1.0" + "@jridgewell/sourcemap-codec" "1.4.14" + +"@leichtgewicht/ip-codec@^2.0.1": + version "2.0.4" + resolved "https://registry.yarnpkg.com/@leichtgewicht/ip-codec/-/ip-codec-2.0.4.tgz#b2ac626d6cb9c8718ab459166d4bb405b8ffa78b" + integrity sha512-Hcv+nVC0kZnQ3tD9GVu5xSMR4VVYOteQIr/hwFPVEvPdlXqgGEuRjiheChHgdM+JyqdgNcmzZOX/tnl0JOiI7A== + +"@nodelib/fs.scandir@2.1.5": + version "2.1.5" + resolved "https://registry.yarnpkg.com/@nodelib/fs.scandir/-/fs.scandir-2.1.5.tgz#7619c2eb21b25483f6d167548b4cfd5a7488c3d5" + integrity sha512-vq24Bq3ym5HEQm2NKCr3yXDwjc7vTsEThRDnkp2DK9p1uqLR+DHurm/NOTo0KG7HYHU7eppKZj3MyqYuMBf62g== + dependencies: + "@nodelib/fs.stat" "2.0.5" + run-parallel "^1.1.9" + +"@nodelib/fs.stat@2.0.5": + version "2.0.5" + resolved "https://registry.yarnpkg.com/@nodelib/fs.stat/-/fs.stat-2.0.5.tgz#5bd262af94e9d25bd1e71b05deed44876a222e8b" + integrity sha512-RkhPPp2zrqDAQA/2jNhnztcPAlv64XdhIp7a7454A5ovI7Bukxgt7MX7udwAu3zg1DcpPU0rz3VV1SeaqvY4+A== + +"@nodelib/fs.walk@^1.2.8": + version "1.2.8" + resolved "https://registry.yarnpkg.com/@nodelib/fs.walk/-/fs.walk-1.2.8.tgz#e95737e8bb6746ddedf69c556953494f196fe69a" + integrity sha512-oGB+UxlgWcgQkgwo8GcEGwemoTFt3FIO9ababBmaGwXIoBKZ+GTy0pP185beGg7Llih/NSHSV2XAs1lnznocSg== + dependencies: + "@nodelib/fs.scandir" "2.1.5" + fastq "^1.6.0" + +"@types/body-parser@*": + version "1.19.2" + resolved "https://registry.yarnpkg.com/@types/body-parser/-/body-parser-1.19.2.tgz#aea2059e28b7658639081347ac4fab3de166e6f0" + integrity sha512-ALYone6pm6QmwZoAgeyNksccT9Q4AWZQ6PvfwR37GT6r6FWUPguq6sUmNGSMV2Wr761oQoBxwGGa6DR5o1DC9g== + dependencies: + "@types/connect" "*" + "@types/node" "*" + +"@types/bonjour@^3.5.9": + version "3.5.10" + resolved "https://registry.yarnpkg.com/@types/bonjour/-/bonjour-3.5.10.tgz#0f6aadfe00ea414edc86f5d106357cda9701e275" + integrity sha512-p7ienRMiS41Nu2/igbJxxLDWrSZ0WxM8UQgCeO9KhoVF7cOVFkrKsiDr1EsJIla8vV3oEEjGcz11jc5yimhzZw== + dependencies: + "@types/node" "*" + +"@types/connect-history-api-fallback@^1.3.5": + version "1.5.0" + resolved "https://registry.yarnpkg.com/@types/connect-history-api-fallback/-/connect-history-api-fallback-1.5.0.tgz#9fd20b3974bdc2bcd4ac6567e2e0f6885cb2cf41" + integrity sha512-4x5FkPpLipqwthjPsF7ZRbOv3uoLUFkTA9G9v583qi4pACvq0uTELrB8OLUzPWUI4IJIyvM85vzkV1nyiI2Lig== + dependencies: + "@types/express-serve-static-core" "*" + "@types/node" "*" + +"@types/connect@*": + version "3.4.35" + resolved "https://registry.yarnpkg.com/@types/connect/-/connect-3.4.35.tgz#5fcf6ae445e4021d1fc2219a4873cc73a3bb2ad1" + integrity sha512-cdeYyv4KWoEgpBISTxWvqYsVy444DOqehiF3fM3ne10AmJ62RSyNkUnxMJXHQWRQQX2eR94m5y1IZyDwBjV9FQ== + dependencies: + "@types/node" "*" + +"@types/eslint-scope@^3.7.3": + version "3.7.4" + resolved "https://registry.yarnpkg.com/@types/eslint-scope/-/eslint-scope-3.7.4.tgz#37fc1223f0786c39627068a12e94d6e6fc61de16" + integrity sha512-9K4zoImiZc3HlIp6AVUDE4CWYx22a+lhSZMYNpbjW04+YF0KWj4pJXnEMjdnFTiQibFFmElcsasJXDbdI/EPhA== + dependencies: + "@types/eslint" "*" + "@types/estree" "*" + +"@types/eslint@*": + version "8.40.2" + resolved "https://registry.yarnpkg.com/@types/eslint/-/eslint-8.40.2.tgz#2833bc112d809677864a4b0e7d1de4f04d7dac2d" + integrity sha512-PRVjQ4Eh9z9pmmtaq8nTjZjQwKFk7YIHIud3lRoKRBgUQjgjRmoGxxGEPXQkF+lH7QkHJRNr5F4aBgYCW0lqpQ== + dependencies: + "@types/estree" "*" + "@types/json-schema" "*" + +"@types/estree@*", "@types/estree@^1.0.0": + version "1.0.1" + resolved "https://registry.yarnpkg.com/@types/estree/-/estree-1.0.1.tgz#aa22750962f3bf0e79d753d3cc067f010c95f194" + integrity sha512-LG4opVs2ANWZ1TJoKc937iMmNstM/d0ae1vNbnBvBhqCSezgVUOzcLCqbI5elV8Vy6WKwKjaqR+zO9VKirBBCA== + +"@types/express-serve-static-core@*", "@types/express-serve-static-core@^4.17.33": + version "4.17.35" + resolved "https://registry.yarnpkg.com/@types/express-serve-static-core/-/express-serve-static-core-4.17.35.tgz#c95dd4424f0d32e525d23812aa8ab8e4d3906c4f" + integrity sha512-wALWQwrgiB2AWTT91CB62b6Yt0sNHpznUXeZEcnPU3DRdlDIz74x8Qg1UUYKSVFi+va5vKOLYRBI1bRKiLLKIg== + dependencies: + "@types/node" "*" + "@types/qs" "*" + "@types/range-parser" "*" + "@types/send" "*" + +"@types/express@*", "@types/express@^4.17.13": + version "4.17.17" + resolved "https://registry.yarnpkg.com/@types/express/-/express-4.17.17.tgz#01d5437f6ef9cfa8668e616e13c2f2ac9a491ae4" + integrity sha512-Q4FmmuLGBG58btUnfS1c1r/NQdlp3DMfGDGig8WhfpA2YRUtEkxAjkZb0yvplJGYdF1fsQ81iMDcH24sSCNC/Q== + dependencies: + "@types/body-parser" "*" + "@types/express-serve-static-core" "^4.17.33" + "@types/qs" "*" + "@types/serve-static" "*" + +"@types/html-minifier-terser@^6.0.0": + version "6.1.0" + resolved "https://registry.yarnpkg.com/@types/html-minifier-terser/-/html-minifier-terser-6.1.0.tgz#4fc33a00c1d0c16987b1a20cf92d20614c55ac35" + integrity sha512-oh/6byDPnL1zeNXFrDXFLyZjkr1MsBG667IM792caf1L2UPOOMf65NFzjUH/ltyfwjAGfs1rsX1eftK0jC/KIg== + +"@types/http-errors@*": + version "2.0.1" + resolved "https://registry.yarnpkg.com/@types/http-errors/-/http-errors-2.0.1.tgz#20172f9578b225f6c7da63446f56d4ce108d5a65" + integrity sha512-/K3ds8TRAfBvi5vfjuz8y6+GiAYBZ0x4tXv1Av6CWBWn0IlADc+ZX9pMq7oU0fNQPnBwIZl3rmeLp6SBApbxSQ== + +"@types/http-proxy@^1.17.8": + version "1.17.11" + resolved "https://registry.yarnpkg.com/@types/http-proxy/-/http-proxy-1.17.11.tgz#0ca21949a5588d55ac2b659b69035c84bd5da293" + integrity sha512-HC8G7c1WmaF2ekqpnFq626xd3Zz0uvaqFmBJNRZCGEZCXkvSdJoNFn/8Ygbd9fKNQj8UzLdCETaI0UWPAjK7IA== + dependencies: + "@types/node" "*" + +"@types/json-schema@*", "@types/json-schema@^7.0.8", "@types/json-schema@^7.0.9": + version "7.0.12" + resolved "https://registry.yarnpkg.com/@types/json-schema/-/json-schema-7.0.12.tgz#d70faba7039d5fca54c83c7dbab41051d2b6f6cb" + integrity sha512-Hr5Jfhc9eYOQNPYO5WLDq/n4jqijdHNlDXjuAQkkt+mWdQR+XJToOHrsD4cPaMXpn6KO7y2+wM8AZEs8VpBLVA== + +"@types/json5@^0.0.29": + version "0.0.29" + resolved "https://registry.yarnpkg.com/@types/json5/-/json5-0.0.29.tgz#ee28707ae94e11d2b827bcbe5270bcea7f3e71ee" + integrity sha512-dRLjCWHYg4oaA77cxO64oO+7JwCwnIzkZPdrrC71jQmQtlhM556pwKo5bUzqvZndkVbeFLIIi+9TC40JNF5hNQ== + +"@types/mime@*": + version "3.0.1" + resolved "https://registry.yarnpkg.com/@types/mime/-/mime-3.0.1.tgz#5f8f2bca0a5863cb69bc0b0acd88c96cb1d4ae10" + integrity sha512-Y4XFY5VJAuw0FgAqPNd6NNoV44jbq9Bz2L7Rh/J6jLTiHBSBJa9fxqQIvkIld4GsoDOcCbvzOUAbLPsSKKg+uA== + +"@types/mime@^1": + version "1.3.2" + resolved "https://registry.yarnpkg.com/@types/mime/-/mime-1.3.2.tgz#93e25bf9ee75fe0fd80b594bc4feb0e862111b5a" + integrity sha512-YATxVxgRqNH6nHEIsvg6k2Boc1JHI9ZbH5iWFFv/MTkchz3b1ieGDa5T0a9RznNdI0KhVbdbWSN+KWWrQZRxTw== + +"@types/node@*": + version "20.3.2" + resolved "https://registry.yarnpkg.com/@types/node/-/node-20.3.2.tgz#fa6a90f2600e052a03c18b8cb3fd83dd4e599898" + integrity sha512-vOBLVQeCQfIcF/2Y7eKFTqrMnizK5lRNQ7ykML/5RuwVXVWxYkgwS7xbt4B6fKCUPgbSL5FSsjHQpaGQP/dQmw== + +"@types/qs@*": + version "6.9.7" + resolved "https://registry.yarnpkg.com/@types/qs/-/qs-6.9.7.tgz#63bb7d067db107cc1e457c303bc25d511febf6cb" + integrity sha512-FGa1F62FT09qcrueBA6qYTrJPVDzah9a+493+o2PCXsesWHIn27G98TsSMs3WPNbZIEj4+VJf6saSFpvD+3Zsw== + +"@types/range-parser@*": + version "1.2.4" + resolved "https://registry.yarnpkg.com/@types/range-parser/-/range-parser-1.2.4.tgz#cd667bcfdd025213aafb7ca5915a932590acdcdc" + integrity sha512-EEhsLsD6UsDM1yFhAvy0Cjr6VwmpMWqFBCb9w07wVugF7w9nfajxLuVmngTIpgS6svCnm6Vaw+MZhoDCKnOfsw== + +"@types/retry@0.12.0": + version "0.12.0" + resolved "https://registry.yarnpkg.com/@types/retry/-/retry-0.12.0.tgz#2b35eccfcee7d38cd72ad99232fbd58bffb3c84d" + integrity sha512-wWKOClTTiizcZhXnPY4wikVAwmdYHp8q6DmC+EJUzAMsycb7HB32Kh9RN4+0gExjmPmZSAQjgURXIGATPegAvA== + +"@types/send@*": + version "0.17.1" + resolved "https://registry.yarnpkg.com/@types/send/-/send-0.17.1.tgz#ed4932b8a2a805f1fe362a70f4e62d0ac994e301" + integrity sha512-Cwo8LE/0rnvX7kIIa3QHCkcuF21c05Ayb0ZfxPiv0W8VRiZiNW/WuRupHKpqqGVGf7SUA44QSOUKaEd9lIrd/Q== + dependencies: + "@types/mime" "^1" + "@types/node" "*" + +"@types/serve-index@^1.9.1": + version "1.9.1" + resolved "https://registry.yarnpkg.com/@types/serve-index/-/serve-index-1.9.1.tgz#1b5e85370a192c01ec6cec4735cf2917337a6278" + integrity sha512-d/Hs3nWDxNL2xAczmOVZNj92YZCS6RGxfBPjKzuu/XirCgXdpKEb88dYNbrYGint6IVWLNP+yonwVAuRC0T2Dg== + dependencies: + "@types/express" "*" + +"@types/serve-static@*", "@types/serve-static@^1.13.10": + version "1.15.2" + resolved "https://registry.yarnpkg.com/@types/serve-static/-/serve-static-1.15.2.tgz#3e5419ecd1e40e7405d34093f10befb43f63381a" + integrity sha512-J2LqtvFYCzaj8pVYKw8klQXrLLk7TBZmQ4ShlcdkELFKGwGMfevMLneMMRkMgZxotOD9wg497LpC7O8PcvAmfw== + dependencies: + "@types/http-errors" "*" + "@types/mime" "*" + "@types/node" "*" + +"@types/sockjs@^0.3.33": + version "0.3.33" + resolved "https://registry.yarnpkg.com/@types/sockjs/-/sockjs-0.3.33.tgz#570d3a0b99ac995360e3136fd6045113b1bd236f" + integrity sha512-f0KEEe05NvUnat+boPTZ0dgaLZ4SfSouXUgv5noUiefG2ajgKjmETo9ZJyuqsl7dfl2aHlLJUiki6B4ZYldiiw== + dependencies: + "@types/node" "*" + +"@types/ws@^8.5.5": + version "8.5.5" + resolved "https://registry.yarnpkg.com/@types/ws/-/ws-8.5.5.tgz#af587964aa06682702ee6dcbc7be41a80e4b28eb" + integrity sha512-lwhs8hktwxSjf9UaZ9tG5M03PGogvFaH8gUgLNbN9HKIg0dvv6q+gkSuJ8HN4/VbyxkuLzCjlN7GquQ0gUJfIg== + dependencies: + "@types/node" "*" + +"@webassemblyjs/ast@1.11.6", "@webassemblyjs/ast@^1.11.5": + version "1.11.6" + resolved "https://registry.yarnpkg.com/@webassemblyjs/ast/-/ast-1.11.6.tgz#db046555d3c413f8966ca50a95176a0e2c642e24" + integrity sha512-IN1xI7PwOvLPgjcf180gC1bqn3q/QaOCwYUahIOhbYUu8KA/3tw2RT/T0Gidi1l7Hhj5D/INhJxiICObqpMu4Q== + dependencies: + "@webassemblyjs/helper-numbers" "1.11.6" + "@webassemblyjs/helper-wasm-bytecode" "1.11.6" + +"@webassemblyjs/floating-point-hex-parser@1.11.6": + version "1.11.6" + resolved "https://registry.yarnpkg.com/@webassemblyjs/floating-point-hex-parser/-/floating-point-hex-parser-1.11.6.tgz#dacbcb95aff135c8260f77fa3b4c5fea600a6431" + integrity sha512-ejAj9hfRJ2XMsNHk/v6Fu2dGS+i4UaXBXGemOfQ/JfQ6mdQg/WXtwleQRLLS4OvfDhv8rYnVwH27YJLMyYsxhw== + +"@webassemblyjs/helper-api-error@1.11.6": + version "1.11.6" + resolved "https://registry.yarnpkg.com/@webassemblyjs/helper-api-error/-/helper-api-error-1.11.6.tgz#6132f68c4acd59dcd141c44b18cbebbd9f2fa768" + integrity sha512-o0YkoP4pVu4rN8aTJgAyj9hC2Sv5UlkzCHhxqWj8butaLvnpdc2jOwh4ewE6CX0txSfLn/UYaV/pheS2Txg//Q== + +"@webassemblyjs/helper-buffer@1.11.6": + version "1.11.6" + resolved "https://registry.yarnpkg.com/@webassemblyjs/helper-buffer/-/helper-buffer-1.11.6.tgz#b66d73c43e296fd5e88006f18524feb0f2c7c093" + integrity sha512-z3nFzdcp1mb8nEOFFk8DrYLpHvhKC3grJD2ardfKOzmbmJvEf/tPIqCY+sNcwZIY8ZD7IkB2l7/pqhUhqm7hLA== + +"@webassemblyjs/helper-numbers@1.11.6": + version "1.11.6" + resolved "https://registry.yarnpkg.com/@webassemblyjs/helper-numbers/-/helper-numbers-1.11.6.tgz#cbce5e7e0c1bd32cf4905ae444ef64cea919f1b5" + integrity sha512-vUIhZ8LZoIWHBohiEObxVm6hwP034jwmc9kuq5GdHZH0wiLVLIPcMCdpJzG4C11cHoQ25TFIQj9kaVADVX7N3g== + dependencies: + "@webassemblyjs/floating-point-hex-parser" "1.11.6" + "@webassemblyjs/helper-api-error" "1.11.6" + "@xtuc/long" "4.2.2" + +"@webassemblyjs/helper-wasm-bytecode@1.11.6": + version "1.11.6" + resolved "https://registry.yarnpkg.com/@webassemblyjs/helper-wasm-bytecode/-/helper-wasm-bytecode-1.11.6.tgz#bb2ebdb3b83aa26d9baad4c46d4315283acd51e9" + integrity sha512-sFFHKwcmBprO9e7Icf0+gddyWYDViL8bpPjJJl0WHxCdETktXdmtWLGVzoHbqUcY4Be1LkNfwTmXOJUFZYSJdA== + +"@webassemblyjs/helper-wasm-section@1.11.6": + version "1.11.6" + resolved "https://registry.yarnpkg.com/@webassemblyjs/helper-wasm-section/-/helper-wasm-section-1.11.6.tgz#ff97f3863c55ee7f580fd5c41a381e9def4aa577" + integrity sha512-LPpZbSOwTpEC2cgn4hTydySy1Ke+XEu+ETXuoyvuyezHO3Kjdu90KK95Sh9xTbmjrCsUwvWwCOQQNta37VrS9g== + dependencies: + "@webassemblyjs/ast" "1.11.6" + "@webassemblyjs/helper-buffer" "1.11.6" + "@webassemblyjs/helper-wasm-bytecode" "1.11.6" + "@webassemblyjs/wasm-gen" "1.11.6" + +"@webassemblyjs/ieee754@1.11.6": + version "1.11.6" + resolved "https://registry.yarnpkg.com/@webassemblyjs/ieee754/-/ieee754-1.11.6.tgz#bb665c91d0b14fffceb0e38298c329af043c6e3a" + integrity sha512-LM4p2csPNvbij6U1f19v6WR56QZ8JcHg3QIJTlSwzFcmx6WSORicYj6I63f9yU1kEUtrpG+kjkiIAkevHpDXrg== + dependencies: + "@xtuc/ieee754" "^1.2.0" + +"@webassemblyjs/leb128@1.11.6": + version "1.11.6" + resolved "https://registry.yarnpkg.com/@webassemblyjs/leb128/-/leb128-1.11.6.tgz#70e60e5e82f9ac81118bc25381a0b283893240d7" + integrity sha512-m7a0FhE67DQXgouf1tbN5XQcdWoNgaAuoULHIfGFIEVKA6tu/edls6XnIlkmS6FrXAquJRPni3ZZKjw6FSPjPQ== + dependencies: + "@xtuc/long" "4.2.2" + +"@webassemblyjs/utf8@1.11.6": + version "1.11.6" + resolved "https://registry.yarnpkg.com/@webassemblyjs/utf8/-/utf8-1.11.6.tgz#90f8bc34c561595fe156603be7253cdbcd0fab5a" + integrity sha512-vtXf2wTQ3+up9Zsg8sa2yWiQpzSsMyXj0qViVP6xKGCUT8p8YJ6HqI7l5eCnWx1T/FYdsv07HQs2wTFbbof/RA== + +"@webassemblyjs/wasm-edit@^1.11.5": + version "1.11.6" + resolved "https://registry.yarnpkg.com/@webassemblyjs/wasm-edit/-/wasm-edit-1.11.6.tgz#c72fa8220524c9b416249f3d94c2958dfe70ceab" + integrity sha512-Ybn2I6fnfIGuCR+Faaz7YcvtBKxvoLV3Lebn1tM4o/IAJzmi9AWYIPWpyBfU8cC+JxAO57bk4+zdsTjJR+VTOw== + dependencies: + "@webassemblyjs/ast" "1.11.6" + "@webassemblyjs/helper-buffer" "1.11.6" + "@webassemblyjs/helper-wasm-bytecode" "1.11.6" + "@webassemblyjs/helper-wasm-section" "1.11.6" + "@webassemblyjs/wasm-gen" "1.11.6" + "@webassemblyjs/wasm-opt" "1.11.6" + "@webassemblyjs/wasm-parser" "1.11.6" + "@webassemblyjs/wast-printer" "1.11.6" + +"@webassemblyjs/wasm-gen@1.11.6": + version "1.11.6" + resolved "https://registry.yarnpkg.com/@webassemblyjs/wasm-gen/-/wasm-gen-1.11.6.tgz#fb5283e0e8b4551cc4e9c3c0d7184a65faf7c268" + integrity sha512-3XOqkZP/y6B4F0PBAXvI1/bky7GryoogUtfwExeP/v7Nzwo1QLcq5oQmpKlftZLbT+ERUOAZVQjuNVak6UXjPA== + dependencies: + "@webassemblyjs/ast" "1.11.6" + "@webassemblyjs/helper-wasm-bytecode" "1.11.6" + "@webassemblyjs/ieee754" "1.11.6" + "@webassemblyjs/leb128" "1.11.6" + "@webassemblyjs/utf8" "1.11.6" + +"@webassemblyjs/wasm-opt@1.11.6": + version "1.11.6" + resolved "https://registry.yarnpkg.com/@webassemblyjs/wasm-opt/-/wasm-opt-1.11.6.tgz#d9a22d651248422ca498b09aa3232a81041487c2" + integrity sha512-cOrKuLRE7PCe6AsOVl7WasYf3wbSo4CeOk6PkrjS7g57MFfVUF9u6ysQBBODX0LdgSvQqRiGz3CXvIDKcPNy4g== + dependencies: + "@webassemblyjs/ast" "1.11.6" + "@webassemblyjs/helper-buffer" "1.11.6" + "@webassemblyjs/wasm-gen" "1.11.6" + "@webassemblyjs/wasm-parser" "1.11.6" + +"@webassemblyjs/wasm-parser@1.11.6", "@webassemblyjs/wasm-parser@^1.11.5": + version "1.11.6" + resolved "https://registry.yarnpkg.com/@webassemblyjs/wasm-parser/-/wasm-parser-1.11.6.tgz#bb85378c527df824004812bbdb784eea539174a1" + integrity sha512-6ZwPeGzMJM3Dqp3hCsLgESxBGtT/OeCvCZ4TA1JUPYgmhAx38tTPR9JaKy0S5H3evQpO/h2uWs2j6Yc/fjkpTQ== + dependencies: + "@webassemblyjs/ast" "1.11.6" + "@webassemblyjs/helper-api-error" "1.11.6" + "@webassemblyjs/helper-wasm-bytecode" "1.11.6" + "@webassemblyjs/ieee754" "1.11.6" + "@webassemblyjs/leb128" "1.11.6" + "@webassemblyjs/utf8" "1.11.6" + +"@webassemblyjs/wast-printer@1.11.6": + version "1.11.6" + resolved "https://registry.yarnpkg.com/@webassemblyjs/wast-printer/-/wast-printer-1.11.6.tgz#a7bf8dd7e362aeb1668ff43f35cb849f188eff20" + integrity sha512-JM7AhRcE+yW2GWYaKeHL5vt4xqee5N2WcezptmgyhNS+ScggqcT1OtXykhAb13Sn5Yas0j2uv9tHgrjwvzAP4A== + dependencies: + "@webassemblyjs/ast" "1.11.6" + "@xtuc/long" "4.2.2" + +"@webpack-cli/configtest@^2.1.1": + version "2.1.1" + resolved "https://registry.yarnpkg.com/@webpack-cli/configtest/-/configtest-2.1.1.tgz#3b2f852e91dac6e3b85fb2a314fb8bef46d94646" + integrity sha512-wy0mglZpDSiSS0XHrVR+BAdId2+yxPSoJW8fsna3ZpYSlufjvxnP4YbKTCBZnNIcGN4r6ZPXV55X4mYExOfLmw== + +"@webpack-cli/info@^2.0.2": + version "2.0.2" + resolved "https://registry.yarnpkg.com/@webpack-cli/info/-/info-2.0.2.tgz#cc3fbf22efeb88ff62310cf885c5b09f44ae0fdd" + integrity sha512-zLHQdI/Qs1UyT5UBdWNqsARasIA+AaF8t+4u2aS2nEpBQh2mWIVb8qAklq0eUENnC5mOItrIB4LiS9xMtph18A== + +"@webpack-cli/serve@^2.0.5": + version "2.0.5" + resolved "https://registry.yarnpkg.com/@webpack-cli/serve/-/serve-2.0.5.tgz#325db42395cd49fe6c14057f9a900e427df8810e" + integrity sha512-lqaoKnRYBdo1UgDX8uF24AfGMifWK19TxPmM5FHc2vAGxrJ/qtyUyFBWoY1tISZdelsQ5fBcOusifo5o5wSJxQ== + +"@xtuc/ieee754@^1.2.0": + version "1.2.0" + resolved "https://registry.yarnpkg.com/@xtuc/ieee754/-/ieee754-1.2.0.tgz#eef014a3145ae477a1cbc00cd1e552336dceb790" + integrity sha512-DX8nKgqcGwsc0eJSqYt5lwP4DH5FlHnmuWWBRy7X0NcaGR0ZtuyeESgMwTYVEtxmsNGY+qit4QYT/MIYTOTPeA== + +"@xtuc/long@4.2.2": + version "4.2.2" + resolved "https://registry.yarnpkg.com/@xtuc/long/-/long-4.2.2.tgz#d291c6a4e97989b5c61d9acf396ae4fe133a718d" + integrity sha512-NuHqBY1PB/D8xU6s/thBgOAiAP7HOYDQ32+BFZILJ8ivkUkAHQnWfn6WhL79Owj1qmUnoN/YPhktdIoucipkAQ== + +accepts@~1.3.4, accepts@~1.3.5, accepts@~1.3.8: + version "1.3.8" + resolved "https://registry.yarnpkg.com/accepts/-/accepts-1.3.8.tgz#0bf0be125b67014adcb0b0921e62db7bffe16b2e" + integrity sha512-PYAthTa2m2VKxuvSD3DPC/Gy+U+sOA1LAuT8mkmRuvw+NACSaeXEQ+NHcVF7rONl6qcaxV3Uuemwawk+7+SJLw== + dependencies: + mime-types "~2.1.34" + negotiator "0.6.3" + +acorn-import-assertions@^1.9.0: + version "1.9.0" + resolved "https://registry.yarnpkg.com/acorn-import-assertions/-/acorn-import-assertions-1.9.0.tgz#507276249d684797c84e0734ef84860334cfb1ac" + integrity sha512-cmMwop9x+8KFhxvKrKfPYmN6/pKTYYHBqLa0DfvVZcKMJWNyWLnaqND7dx/qn66R7ewM1UX5XMaDVP5wlVTaVA== + +acorn-jsx@^5.3.2: + version "5.3.2" + resolved "https://registry.yarnpkg.com/acorn-jsx/-/acorn-jsx-5.3.2.tgz#7ed5bb55908b3b2f1bc55c6af1653bada7f07937" + integrity sha512-rq9s+JNhf0IChjtDXxllJ7g41oZk5SlXtp0LHwyA5cejwn7vKmKp4pPri6YEePv2PU65sAsegbXtIinmDFDXgQ== + +acorn@^8.7.1, acorn@^8.8.0, acorn@^8.8.2: + version "8.9.0" + resolved "https://registry.yarnpkg.com/acorn/-/acorn-8.9.0.tgz#78a16e3b2bcc198c10822786fa6679e245db5b59" + integrity sha512-jaVNAFBHNLXspO543WnNNPZFRtavh3skAkITqD0/2aeMkKZTN+254PyhwxFYrk3vQ1xfY+2wbesJMs/JC8/PwQ== + +aggregate-error@^3.0.0: + version "3.1.0" + resolved "https://registry.yarnpkg.com/aggregate-error/-/aggregate-error-3.1.0.tgz#92670ff50f5359bdb7a3e0d40d0ec30c5737687a" + integrity sha512-4I7Td01quW/RpocfNayFdFVk1qSuoh0E7JrbRJ16nH01HhKFQ88INq9Sd+nd72zqRySlr9BmDA8xlEJ6vJMrYA== + dependencies: + clean-stack "^2.0.0" + indent-string "^4.0.0" + +ajv-formats@^2.1.1: + version "2.1.1" + resolved "https://registry.yarnpkg.com/ajv-formats/-/ajv-formats-2.1.1.tgz#6e669400659eb74973bbf2e33327180a0996b520" + integrity sha512-Wx0Kx52hxE7C18hkMEggYlEifqWZtYaRgouJor+WMdPnQyEK13vgEWyVNup7SoeeoLMsr4kf5h6dOW11I15MUA== + dependencies: + ajv "^8.0.0" + +ajv-keywords@^3.5.2: + version "3.5.2" + resolved "https://registry.yarnpkg.com/ajv-keywords/-/ajv-keywords-3.5.2.tgz#31f29da5ab6e00d1c2d329acf7b5929614d5014d" + integrity sha512-5p6WTN0DdTGVQk6VjcEju19IgaHudalcfabD7yhDGeA6bcQnmL+CpveLJq/3hvfwd1aof6L386Ougkx6RfyMIQ== + +ajv-keywords@^5.1.0: + version "5.1.0" + resolved "https://registry.yarnpkg.com/ajv-keywords/-/ajv-keywords-5.1.0.tgz#69d4d385a4733cdbeab44964a1170a88f87f0e16" + integrity sha512-YCS/JNFAUyr5vAuhk1DWm1CBxRHW9LbJ2ozWeemrIqpbsqKjHVxYPyi5GC0rjZIT5JxJ3virVTS8wk4i/Z+krw== + dependencies: + fast-deep-equal "^3.1.3" + +ajv@^6.10.0, ajv@^6.12.4, ajv@^6.12.5: + version "6.12.6" + resolved "https://registry.yarnpkg.com/ajv/-/ajv-6.12.6.tgz#baf5a62e802b07d977034586f8c3baf5adf26df4" + integrity sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g== + dependencies: + fast-deep-equal "^3.1.1" + fast-json-stable-stringify "^2.0.0" + json-schema-traverse "^0.4.1" + uri-js "^4.2.2" + +ajv@^8.0.0, ajv@^8.9.0: + version "8.12.0" + resolved "https://registry.yarnpkg.com/ajv/-/ajv-8.12.0.tgz#d1a0527323e22f53562c567c00991577dfbe19d1" + integrity sha512-sRu1kpcO9yLtYxBKvqfTeh9KzZEwO3STyX1HT+4CaDzC6HpTGYhIhPIzj9XuKU7KYDwnaeh5hcOwjy1QuJzBPA== + dependencies: + fast-deep-equal "^3.1.1" + json-schema-traverse "^1.0.0" + require-from-string "^2.0.2" + uri-js "^4.2.2" + +ansi-escapes@^4.3.0: + version "4.3.2" + resolved "https://registry.yarnpkg.com/ansi-escapes/-/ansi-escapes-4.3.2.tgz#6b2291d1db7d98b6521d5f1efa42d0f3a9feb65e" + integrity sha512-gKXj5ALrKWQLsYG9jlTRmR/xKluxHV+Z9QEwNIgCfM1/uwPMCuzVVnh5mwTd+OuBZcwSIMbqssNWRm1lE51QaQ== + dependencies: + type-fest "^0.21.3" + +ansi-html-community@^0.0.8: + version "0.0.8" + resolved "https://registry.yarnpkg.com/ansi-html-community/-/ansi-html-community-0.0.8.tgz#69fbc4d6ccbe383f9736934ae34c3f8290f1bf41" + integrity sha512-1APHAyr3+PCamwNw3bXCPp4HFLONZt/yIH0sZp0/469KWNTEy+qN5jQ3GVX6DMZ1UXAi34yVwtTeaG/HpBuuzw== + +ansi-regex@^5.0.1: + version "5.0.1" + resolved "https://registry.yarnpkg.com/ansi-regex/-/ansi-regex-5.0.1.tgz#082cb2c89c9fe8659a311a53bd6a4dc5301db304" + integrity sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ== + +ansi-regex@^6.0.1: + version "6.0.1" + resolved "https://registry.yarnpkg.com/ansi-regex/-/ansi-regex-6.0.1.tgz#3183e38fae9a65d7cb5e53945cd5897d0260a06a" + integrity sha512-n5M855fKb2SsfMIiFFoVrABHJC8QtHwVx+mHWP3QcEqBHYienj5dHSgjbxtC0WEZXYt4wcD6zrQElDPhFuZgfA== + +ansi-styles@^3.2.1: + version "3.2.1" + resolved "https://registry.yarnpkg.com/ansi-styles/-/ansi-styles-3.2.1.tgz#41fbb20243e50b12be0f04b8dedbf07520ce841d" + integrity sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA== + dependencies: + color-convert "^1.9.0" + +ansi-styles@^4.0.0, ansi-styles@^4.1.0: + version "4.3.0" + resolved "https://registry.yarnpkg.com/ansi-styles/-/ansi-styles-4.3.0.tgz#edd803628ae71c04c85ae7a0906edad34b648937" + integrity sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg== + dependencies: + color-convert "^2.0.1" + +ansi-styles@^6.0.0: + version "6.2.1" + resolved "https://registry.yarnpkg.com/ansi-styles/-/ansi-styles-6.2.1.tgz#0e62320cf99c21afff3b3012192546aacbfb05c5" + integrity sha512-bN798gFfQX+viw3R7yrGWRqnrN2oRkEkUjjl4JNn4E8GxxbjtG3FbrEIIY3l8/hrwUwIeCZvi4QuOTP4MErVug== + +anymatch@~3.1.2: + version "3.1.3" + resolved "https://registry.yarnpkg.com/anymatch/-/anymatch-3.1.3.tgz#790c58b19ba1720a84205b57c618d5ad8524973e" + integrity sha512-KMReFUr0B4t+D+OBkjR3KYqvocp2XaSzO55UcB6mgQMd3KbcE+mWTyvVV7D/zsdEbNnV6acZUutkiHQXvTr1Rw== + dependencies: + normalize-path "^3.0.0" + picomatch "^2.0.4" + +argparse@^2.0.1: + version "2.0.1" + resolved "https://registry.yarnpkg.com/argparse/-/argparse-2.0.1.tgz#246f50f3ca78a3240f6c997e8a9bd1eac49e4b38" + integrity sha512-8+9WqebbFzpX9OR+Wa6O29asIogeRMzcGtAINdpMHHyAg10f05aSFVBbcEqGf/PXw1EjAZ+q2/bEBg3DvurK3Q== + +array-buffer-byte-length@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/array-buffer-byte-length/-/array-buffer-byte-length-1.0.0.tgz#fabe8bc193fea865f317fe7807085ee0dee5aead" + integrity sha512-LPuwb2P+NrQw3XhxGc36+XSvuBPopovXYTR9Ew++Du9Yb/bx5AzBfrIsBoj0EZUifjQU+sHL21sseZ3jerWO/A== + dependencies: + call-bind "^1.0.2" + is-array-buffer "^3.0.1" + +array-flatten@1.1.1: + version "1.1.1" + resolved "https://registry.yarnpkg.com/array-flatten/-/array-flatten-1.1.1.tgz#9a5f699051b1e7073328f2a008968b64ea2955d2" + integrity sha512-PCVAQswWemu6UdxsDFFX/+gVeYqKAod3D3UVm91jHwynguOwAvYPhx8nNlM++NqRcK6CxxpUafjmhIdKiHibqg== + +array-flatten@^2.1.2: + version "2.1.2" + resolved "https://registry.yarnpkg.com/array-flatten/-/array-flatten-2.1.2.tgz#24ef80a28c1a893617e2149b0c6d0d788293b099" + integrity sha512-hNfzcOV8W4NdualtqBFPyVO+54DSJuZGY9qT4pRroB6S9e3iiido2ISIC5h9R2sPJ8H3FHCIiEnsv1lPXO3KtQ== + +array-includes@^3.1.6: + version "3.1.6" + resolved "https://registry.yarnpkg.com/array-includes/-/array-includes-3.1.6.tgz#9e9e720e194f198266ba9e18c29e6a9b0e4b225f" + integrity sha512-sgTbLvL6cNnw24FnbaDyjmvddQ2ML8arZsgaJhoABMoplz/4QRhtrYS+alr1BUM1Bwp6dhx8vVCBSLG+StwOFw== + dependencies: + call-bind "^1.0.2" + define-properties "^1.1.4" + es-abstract "^1.20.4" + get-intrinsic "^1.1.3" + is-string "^1.0.7" + +array.prototype.flat@^1.3.1: + version "1.3.1" + resolved "https://registry.yarnpkg.com/array.prototype.flat/-/array.prototype.flat-1.3.1.tgz#ffc6576a7ca3efc2f46a143b9d1dda9b4b3cf5e2" + integrity sha512-roTU0KWIOmJ4DRLmwKd19Otg0/mT3qPNt0Qb3GWW8iObuZXxrjB/pzn0R3hqpRSWg4HCwqx+0vwOnWnvlOyeIA== + dependencies: + call-bind "^1.0.2" + define-properties "^1.1.4" + es-abstract "^1.20.4" + es-shim-unscopables "^1.0.0" + +array.prototype.flatmap@^1.3.1: + version "1.3.1" + resolved "https://registry.yarnpkg.com/array.prototype.flatmap/-/array.prototype.flatmap-1.3.1.tgz#1aae7903c2100433cb8261cd4ed310aab5c4a183" + integrity sha512-8UGn9O1FDVvMNB0UlLv4voxRMze7+FpHyF5mSMRjWHUMlpoDViniy05870VlxhfgTnLbpuwTzvD76MTtWxB/mQ== + dependencies: + call-bind "^1.0.2" + define-properties "^1.1.4" + es-abstract "^1.20.4" + es-shim-unscopables "^1.0.0" + +astral-regex@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/astral-regex/-/astral-regex-2.0.0.tgz#483143c567aeed4785759c0865786dc77d7d2e31" + integrity sha512-Z7tMw1ytTXt5jqMcOP+OQteU1VuNK9Y02uuJtKQ1Sv69jXQKKg5cibLwGJow8yzZP+eAc18EmLGPal0bp36rvQ== + +available-typed-arrays@^1.0.5: + version "1.0.5" + resolved "https://registry.yarnpkg.com/available-typed-arrays/-/available-typed-arrays-1.0.5.tgz#92f95616501069d07d10edb2fc37d3e1c65123b7" + integrity sha512-DMD0KiN46eipeziST1LPP/STfDU0sufISXmjSgvVsoU2tqxctQeASejWcfNtxYKqETM1UxQ8sp2OrSBWpHY6sw== + +babel-loader@^9.1.2: + version "9.1.2" + resolved "https://registry.yarnpkg.com/babel-loader/-/babel-loader-9.1.2.tgz#a16a080de52d08854ee14570469905a5fc00d39c" + integrity sha512-mN14niXW43tddohGl8HPu5yfQq70iUThvFL/4QzESA7GcZoC0eVOhvWdQ8+3UlSjaDE9MVtsW9mxDY07W7VpVA== + dependencies: + find-cache-dir "^3.3.2" + schema-utils "^4.0.0" + +babel-plugin-module-resolver@^5.0.0: + version "5.0.0" + resolved "https://registry.yarnpkg.com/babel-plugin-module-resolver/-/babel-plugin-module-resolver-5.0.0.tgz#2b7fc176bd55da25f516abf96015617b4f70fc73" + integrity sha512-g0u+/ChLSJ5+PzYwLwP8Rp8Rcfowz58TJNCe+L/ui4rpzE/mg//JVX0EWBUYoxaextqnwuGHzfGp2hh0PPV25Q== + dependencies: + find-babel-config "^2.0.0" + glob "^8.0.3" + pkg-up "^3.1.0" + reselect "^4.1.7" + resolve "^1.22.1" + +babel-plugin-polyfill-corejs2@^0.4.3: + version "0.4.3" + resolved "https://registry.yarnpkg.com/babel-plugin-polyfill-corejs2/-/babel-plugin-polyfill-corejs2-0.4.3.tgz#75044d90ba5043a5fb559ac98496f62f3eb668fd" + integrity sha512-bM3gHc337Dta490gg+/AseNB9L4YLHxq1nGKZZSHbhXv4aTYU2MD2cjza1Ru4S6975YLTaL1K8uJf6ukJhhmtw== + dependencies: + "@babel/compat-data" "^7.17.7" + "@babel/helper-define-polyfill-provider" "^0.4.0" + semver "^6.1.1" + +babel-plugin-polyfill-corejs3@^0.8.1: + version "0.8.1" + resolved "https://registry.yarnpkg.com/babel-plugin-polyfill-corejs3/-/babel-plugin-polyfill-corejs3-0.8.1.tgz#39248263c38191f0d226f928d666e6db1b4b3a8a" + integrity sha512-ikFrZITKg1xH6pLND8zT14UPgjKHiGLqex7rGEZCH2EvhsneJaJPemmpQaIZV5AL03II+lXylw3UmddDK8RU5Q== + dependencies: + "@babel/helper-define-polyfill-provider" "^0.4.0" + core-js-compat "^3.30.1" + +babel-plugin-polyfill-regenerator@^0.5.0: + version "0.5.0" + resolved "https://registry.yarnpkg.com/babel-plugin-polyfill-regenerator/-/babel-plugin-polyfill-regenerator-0.5.0.tgz#e7344d88d9ef18a3c47ded99362ae4a757609380" + integrity sha512-hDJtKjMLVa7Z+LwnTCxoDLQj6wdc+B8dun7ayF2fYieI6OzfuvcLMB32ihJZ4UhCBwNYGl5bg/x/P9cMdnkc2g== + dependencies: + "@babel/helper-define-polyfill-provider" "^0.4.0" + +balanced-match@^1.0.0: + version "1.0.2" + resolved "https://registry.yarnpkg.com/balanced-match/-/balanced-match-1.0.2.tgz#e83e3a7e3f300b34cb9d87f615fa0cbf357690ee" + integrity sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw== + +batch@0.6.1: + version "0.6.1" + resolved "https://registry.yarnpkg.com/batch/-/batch-0.6.1.tgz#dc34314f4e679318093fc760272525f94bf25c16" + integrity sha512-x+VAiMRL6UPkx+kudNvxTl6hB2XNNCG2r+7wixVfIYwu/2HKRXimwQyaumLjMveWvT2Hkd/cAJw+QBMfJ/EKVw== + +binary-extensions@^2.0.0: + version "2.2.0" + resolved "https://registry.yarnpkg.com/binary-extensions/-/binary-extensions-2.2.0.tgz#75f502eeaf9ffde42fc98829645be4ea76bd9e2d" + integrity sha512-jDctJ/IVQbZoJykoeHbhXpOlNBqGNcwXJKJog42E5HDPUwQTSdjCHdihjj0DlnheQ7blbT6dHOafNAiS8ooQKA== + +body-parser@1.20.1: + version "1.20.1" + resolved "https://registry.yarnpkg.com/body-parser/-/body-parser-1.20.1.tgz#b1812a8912c195cd371a3ee5e66faa2338a5c668" + integrity sha512-jWi7abTbYwajOytWCQc37VulmWiRae5RyTpaCyDcS5/lMdtwSz5lOpDE67srw/HYe35f1z3fDQw+3txg7gNtWw== + dependencies: + bytes "3.1.2" + content-type "~1.0.4" + debug "2.6.9" + depd "2.0.0" + destroy "1.2.0" + http-errors "2.0.0" + iconv-lite "0.4.24" + on-finished "2.4.1" + qs "6.11.0" + raw-body "2.5.1" + type-is "~1.6.18" + unpipe "1.0.0" + +bonjour-service@^1.0.11: + version "1.1.1" + resolved "https://registry.yarnpkg.com/bonjour-service/-/bonjour-service-1.1.1.tgz#960948fa0e0153f5d26743ab15baf8e33752c135" + integrity sha512-Z/5lQRMOG9k7W+FkeGTNjh7htqn/2LMnfOvBZ8pynNZCM9MwkQkI3zeI4oz09uWdcgmgHugVvBqxGg4VQJ5PCg== + dependencies: + array-flatten "^2.1.2" + dns-equal "^1.0.0" + fast-deep-equal "^3.1.3" + multicast-dns "^7.2.5" + +boolbase@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/boolbase/-/boolbase-1.0.0.tgz#68dff5fbe60c51eb37725ea9e3ed310dcc1e776e" + integrity sha512-JZOSA7Mo9sNGB8+UjSgzdLtokWAky1zbztM3WRLCbZ70/3cTANmQmOdR7y2g+J0e2WXywy1yS468tY+IruqEww== + +brace-expansion@^1.1.7: + version "1.1.11" + resolved "https://registry.yarnpkg.com/brace-expansion/-/brace-expansion-1.1.11.tgz#3c7fcbf529d87226f3d2f52b966ff5271eb441dd" + integrity sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA== + dependencies: + balanced-match "^1.0.0" + concat-map "0.0.1" + +brace-expansion@^2.0.1: + version "2.0.1" + resolved "https://registry.yarnpkg.com/brace-expansion/-/brace-expansion-2.0.1.tgz#1edc459e0f0c548486ecf9fc99f2221364b9a0ae" + integrity sha512-XnAIvQ8eM+kC6aULx6wuQiwVsnzsi9d3WxzV3FpWTGA19F621kwdbsAcFKXgKUHZWsy+mY6iL1sHTxWEFCytDA== + dependencies: + balanced-match "^1.0.0" + +braces@^3.0.2, braces@~3.0.2: + version "3.0.2" + resolved "https://registry.yarnpkg.com/braces/-/braces-3.0.2.tgz#3454e1a462ee8d599e236df336cd9ea4f8afe107" + integrity sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A== + dependencies: + fill-range "^7.0.1" + +browserslist@^4.14.5, browserslist@^4.21.3, browserslist@^4.21.5: + version "4.21.9" + resolved "https://registry.yarnpkg.com/browserslist/-/browserslist-4.21.9.tgz#e11bdd3c313d7e2a9e87e8b4b0c7872b13897635" + integrity sha512-M0MFoZzbUrRU4KNfCrDLnvyE7gub+peetoTid3TBIqtunaDJyXlwhakT+/VkvSXcfIzFfK/nkCs4nmyTmxdNSg== + dependencies: + caniuse-lite "^1.0.30001503" + electron-to-chromium "^1.4.431" + node-releases "^2.0.12" + update-browserslist-db "^1.0.11" + +buffer-from@^1.0.0: + version "1.1.2" + resolved "https://registry.yarnpkg.com/buffer-from/-/buffer-from-1.1.2.tgz#2b146a6fd72e80b4f55d255f35ed59a3a9a41bd5" + integrity sha512-E+XQCRwSbaaiChtv6k6Dwgc+bx+Bs6vuKJHHl5kox/BaKbhiXzqQOwK4cO22yElGp2OCmjwVhT3HmxgyPGnJfQ== + +bytes@3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/bytes/-/bytes-3.0.0.tgz#d32815404d689699f85a4ea4fa8755dd13a96048" + integrity sha512-pMhOfFDPiv9t5jjIXkHosWmkSyQbvsgEVNkz0ERHbuLh2T/7j4Mqqpz523Fe8MVY89KC6Sh/QfS2sM+SjgFDcw== + +bytes@3.1.2: + version "3.1.2" + resolved "https://registry.yarnpkg.com/bytes/-/bytes-3.1.2.tgz#8b0beeb98605adf1b128fa4386403c009e0221a5" + integrity sha512-/Nf7TyzTx6S3yRJObOAV7956r8cr2+Oj8AC5dt8wSP3BQAoeX58NoHyCU8P8zGkNXStjTSi6fzO6F0pBdcYbEg== + +call-bind@^1.0.0, call-bind@^1.0.2: + version "1.0.2" + resolved "https://registry.yarnpkg.com/call-bind/-/call-bind-1.0.2.tgz#b1d4e89e688119c3c9a903ad30abb2f6a919be3c" + integrity sha512-7O+FbCihrB5WGbFYesctwmTKae6rOiIzmz1icreWJ+0aA7LJfuqhEso2T9ncpcFtzMQtzXf2QGGueWJGTYsqrA== + dependencies: + function-bind "^1.1.1" + get-intrinsic "^1.0.2" + +callsites@^3.0.0: + version "3.1.0" + resolved "https://registry.yarnpkg.com/callsites/-/callsites-3.1.0.tgz#b3630abd8943432f54b3f0519238e33cd7df2f73" + integrity sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ== + +camel-case@^4.1.2: + version "4.1.2" + resolved "https://registry.yarnpkg.com/camel-case/-/camel-case-4.1.2.tgz#9728072a954f805228225a6deea6b38461e1bd5a" + integrity sha512-gxGWBrTT1JuMx6R+o5PTXMmUnhnVzLQ9SNutD4YqKtI6ap897t3tKECYla6gCWEkplXnlNybEkZg9GEGxKFCgw== + dependencies: + pascal-case "^3.1.2" + tslib "^2.0.3" + +caniuse-lite@^1.0.30001503: + version "1.0.30001509" + resolved "https://registry.yarnpkg.com/caniuse-lite/-/caniuse-lite-1.0.30001509.tgz#2b7ad5265392d6d2de25cd8776d1ab3899570d14" + integrity sha512-2uDDk+TRiTX5hMcUYT/7CSyzMZxjfGu0vAUjS2g0LSD8UoXOv0LtpH4LxGMemsiPq6LCVIUjNwVM0erkOkGCDA== + +chalk@5.2.0: + version "5.2.0" + resolved "https://registry.yarnpkg.com/chalk/-/chalk-5.2.0.tgz#249623b7d66869c673699fb66d65723e54dfcfb3" + integrity sha512-ree3Gqw/nazQAPuJJEy+avdl7QfZMcUvmHIKgEZkGL+xOBzRvup5Hxo6LHuMceSxOabuJLJm5Yp/92R9eMmMvA== + +chalk@^2.0.0: + version "2.4.2" + resolved "https://registry.yarnpkg.com/chalk/-/chalk-2.4.2.tgz#cd42541677a54333cf541a49108c1432b44c9424" + integrity sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ== + dependencies: + ansi-styles "^3.2.1" + escape-string-regexp "^1.0.5" + supports-color "^5.3.0" + +chalk@^4.0.0: + version "4.1.2" + resolved "https://registry.yarnpkg.com/chalk/-/chalk-4.1.2.tgz#aac4e2b7734a740867aeb16bf02aad556a1e7a01" + integrity sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA== + dependencies: + ansi-styles "^4.1.0" + supports-color "^7.1.0" + +chokidar@^3.5.3: + version "3.5.3" + resolved "https://registry.yarnpkg.com/chokidar/-/chokidar-3.5.3.tgz#1cf37c8707b932bd1af1ae22c0432e2acd1903bd" + integrity sha512-Dr3sfKRP6oTcjf2JmUmFJfeVMvXBdegxB0iVQ5eb2V10uFJUCAS8OByZdVAyVb8xXNz3GjjTgj9kLWsZTqE6kw== + dependencies: + anymatch "~3.1.2" + braces "~3.0.2" + glob-parent "~5.1.2" + is-binary-path "~2.1.0" + is-glob "~4.0.1" + normalize-path "~3.0.0" + readdirp "~3.6.0" + optionalDependencies: + fsevents "~2.3.2" + +chrome-trace-event@^1.0.2: + version "1.0.3" + resolved "https://registry.yarnpkg.com/chrome-trace-event/-/chrome-trace-event-1.0.3.tgz#1015eced4741e15d06664a957dbbf50d041e26ac" + integrity sha512-p3KULyQg4S7NIHixdwbGX+nFHkoBiA4YQmyWtjb8XngSKV124nJmRysgAeujbUVb15vh+RvFUfCPqU7rXk+hZg== + +clean-css@^5.2.2: + version "5.3.2" + resolved "https://registry.yarnpkg.com/clean-css/-/clean-css-5.3.2.tgz#70ecc7d4d4114921f5d298349ff86a31a9975224" + integrity sha512-JVJbM+f3d3Q704rF4bqQ5UUyTtuJ0JRKNbTKVEeujCCBoMdkEi+V+e8oktO9qGQNSvHrFTM6JZRXrUvGR1czww== + dependencies: + source-map "~0.6.0" + +clean-stack@^2.0.0: + version "2.2.0" + resolved "https://registry.yarnpkg.com/clean-stack/-/clean-stack-2.2.0.tgz#ee8472dbb129e727b31e8a10a427dee9dfe4008b" + integrity sha512-4diC9HaTE+KRAMWhDhrGOECgWZxoevMc5TlkObMqNSsVU62PYzXZ/SMTjzyGAFF1YusgxGcSWTEXBhp0CPwQ1A== + +cli-cursor@^3.1.0: + version "3.1.0" + resolved "https://registry.yarnpkg.com/cli-cursor/-/cli-cursor-3.1.0.tgz#264305a7ae490d1d03bf0c9ba7c925d1753af307" + integrity sha512-I/zHAwsKf9FqGoXM4WWRACob9+SNukZTd94DWF57E4toouRulbCxcUh6RKUEOQlYTHJnzkPMySvPNaaSLNfLZw== + dependencies: + restore-cursor "^3.1.0" + +cli-truncate@^2.1.0: + version "2.1.0" + resolved "https://registry.yarnpkg.com/cli-truncate/-/cli-truncate-2.1.0.tgz#c39e28bf05edcde5be3b98992a22deed5a2b93c7" + integrity sha512-n8fOixwDD6b/ObinzTrp1ZKFzbgvKZvuz/TvejnLn1aQfC6r52XEx85FmuC+3HI+JM7coBRXUvNqEU2PHVrHpg== + dependencies: + slice-ansi "^3.0.0" + string-width "^4.2.0" + +cli-truncate@^3.1.0: + version "3.1.0" + resolved "https://registry.yarnpkg.com/cli-truncate/-/cli-truncate-3.1.0.tgz#3f23ab12535e3d73e839bb43e73c9de487db1389" + integrity sha512-wfOBkjXteqSnI59oPcJkcPl/ZmwvMMOj340qUIY1SKZCv0B9Cf4D4fAucRkIKQmsIuYK3x1rrgU7MeGRruiuiA== + dependencies: + slice-ansi "^5.0.0" + string-width "^5.0.0" + +clone-deep@^4.0.1: + version "4.0.1" + resolved "https://registry.yarnpkg.com/clone-deep/-/clone-deep-4.0.1.tgz#c19fd9bdbbf85942b4fd979c84dcf7d5f07c2387" + integrity sha512-neHB9xuzh/wk0dIHweyAXv2aPGZIVk3pLMe+/RNzINf17fe0OG96QroktYAUm7SM1PBnzTabaLboqqxDyMU+SQ== + dependencies: + is-plain-object "^2.0.4" + kind-of "^6.0.2" + shallow-clone "^3.0.0" + +color-convert@^1.9.0: + version "1.9.3" + resolved "https://registry.yarnpkg.com/color-convert/-/color-convert-1.9.3.tgz#bb71850690e1f136567de629d2d5471deda4c1e8" + integrity sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg== + dependencies: + color-name "1.1.3" + +color-convert@^2.0.1: + version "2.0.1" + resolved "https://registry.yarnpkg.com/color-convert/-/color-convert-2.0.1.tgz#72d3a68d598c9bdb3af2ad1e84f21d896abd4de3" + integrity sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ== + dependencies: + color-name "~1.1.4" + +color-name@1.1.3: + version "1.1.3" + resolved "https://registry.yarnpkg.com/color-name/-/color-name-1.1.3.tgz#a7d0558bd89c42f795dd42328f740831ca53bc25" + integrity sha512-72fSenhMw2HZMTVHeCA9KCmpEIbzWiQsjN+BHcBbS9vr1mtt+vJjPdksIBNUmKAW8TFUDPJK5SUU3QhE9NEXDw== + +color-name@~1.1.4: + version "1.1.4" + resolved "https://registry.yarnpkg.com/color-name/-/color-name-1.1.4.tgz#c2a09a87acbde69543de6f63fa3995c826c536a2" + integrity sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA== + +colorette@^2.0.10, colorette@^2.0.14, colorette@^2.0.19: + version "2.0.20" + resolved "https://registry.yarnpkg.com/colorette/-/colorette-2.0.20.tgz#9eb793e6833067f7235902fcd3b09917a000a95a" + integrity sha512-IfEDxwoWIjkeXL1eXcDiow4UbKjhLdq6/EuSVR9GMN7KVH3r9gQ83e73hsz1Nd1T3ijd5xv1wcWRYO+D6kCI2w== + +commander@^10.0.0, commander@^10.0.1: + version "10.0.1" + resolved "https://registry.yarnpkg.com/commander/-/commander-10.0.1.tgz#881ee46b4f77d1c1dccc5823433aa39b022cbe06" + integrity sha512-y4Mg2tXshplEbSGzx7amzPwKKOCGuoSRP/CjEdwwk0FOGlUbq6lKuoyDZTNZkmxHdJtp54hdfY/JUrdL7Xfdug== + +commander@^2.20.0: + version "2.20.3" + resolved "https://registry.yarnpkg.com/commander/-/commander-2.20.3.tgz#fd485e84c03eb4881c20722ba48035e8531aeb33" + integrity sha512-GpVkmM8vF2vQUkj2LvZmD35JxeJOLCwJ9cUkugyk2nuhbv3+mJvpLYYt+0+USMxE+oj+ey/lJEnhZw75x/OMcQ== + +commander@^8.3.0: + version "8.3.0" + resolved "https://registry.yarnpkg.com/commander/-/commander-8.3.0.tgz#4837ea1b2da67b9c616a67afbb0fafee567bca66" + integrity sha512-OkTL9umf+He2DZkUq8f8J9of7yL6RJKI24dVITBmNfZBmri9zYZQrKkuXiKhyfPSu8tUhnVBB1iKXevvnlR4Ww== + +commondir@^1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/commondir/-/commondir-1.0.1.tgz#ddd800da0c66127393cca5950ea968a3aaf1253b" + integrity sha512-W9pAhw0ja1Edb5GVdIF1mjZw/ASI0AlShXM83UUGe2DVr5TdAPEA1OA8m/g8zWp9x6On7gqufY+FatDbC3MDQg== + +compressible@~2.0.16: + version "2.0.18" + resolved "https://registry.yarnpkg.com/compressible/-/compressible-2.0.18.tgz#af53cca6b070d4c3c0750fbd77286a6d7cc46fba" + integrity sha512-AF3r7P5dWxL8MxyITRMlORQNaOA2IkAFaTr4k7BUumjPtRpGDTZpl0Pb1XCO6JeDCBdp126Cgs9sMxqSjgYyRg== + dependencies: + mime-db ">= 1.43.0 < 2" + +compression@^1.7.4: + version "1.7.4" + resolved "https://registry.yarnpkg.com/compression/-/compression-1.7.4.tgz#95523eff170ca57c29a0ca41e6fe131f41e5bb8f" + integrity sha512-jaSIDzP9pZVS4ZfQ+TzvtiWhdpFhE2RDHz8QJkpX9SIpLq88VueF5jJw6t+6CUQcAoA6t+x89MLrWAqpfDE8iQ== + dependencies: + accepts "~1.3.5" + bytes "3.0.0" + compressible "~2.0.16" + debug "2.6.9" + on-headers "~1.0.2" + safe-buffer "5.1.2" + vary "~1.1.2" + +concat-map@0.0.1: + version "0.0.1" + resolved "https://registry.yarnpkg.com/concat-map/-/concat-map-0.0.1.tgz#d8a96bd77fd68df7793a73036a3ba0d5405d477b" + integrity sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg== + +confusing-browser-globals@^1.0.10: + version "1.0.11" + resolved "https://registry.yarnpkg.com/confusing-browser-globals/-/confusing-browser-globals-1.0.11.tgz#ae40e9b57cdd3915408a2805ebd3a5585608dc81" + integrity sha512-JsPKdmh8ZkmnHxDk55FZ1TqVLvEQTvoByJZRN9jzI0UjxK/QgAmsphz7PGtqgPieQZ/CQcHWXCR7ATDNhGe+YA== + +connect-history-api-fallback@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/connect-history-api-fallback/-/connect-history-api-fallback-2.0.0.tgz#647264845251a0daf25b97ce87834cace0f5f1c8" + integrity sha512-U73+6lQFmfiNPrYbXqr6kZ1i1wiRqXnp2nhMsINseWXO8lDau0LGEffJ8kQi4EjLZympVgRdvqjAgiZ1tgzDDA== + +content-disposition@0.5.4: + version "0.5.4" + resolved "https://registry.yarnpkg.com/content-disposition/-/content-disposition-0.5.4.tgz#8b82b4efac82512a02bb0b1dcec9d2c5e8eb5bfe" + integrity sha512-FveZTNuGw04cxlAiWbzi6zTAL/lhehaWbTtgluJh4/E95DqMwTmha3KZN1aAWA8cFIhHzMZUvLevkw5Rqk+tSQ== + dependencies: + safe-buffer "5.2.1" + +content-type@~1.0.4: + version "1.0.5" + resolved "https://registry.yarnpkg.com/content-type/-/content-type-1.0.5.tgz#8b773162656d1d1086784c8f23a54ce6d73d7918" + integrity sha512-nTjqfcBFEipKdXCv4YDQWCfmcLZKm81ldF0pAopTvyrFGVbcR6P/VAAd5G7N+0tTr8QqiU0tFadD6FK4NtJwOA== + +convert-source-map@^1.7.0: + version "1.9.0" + resolved "https://registry.yarnpkg.com/convert-source-map/-/convert-source-map-1.9.0.tgz#7faae62353fb4213366d0ca98358d22e8368b05f" + integrity sha512-ASFBup0Mz1uyiIjANan1jzLQami9z1PoYSZCiiYW2FczPbenXc45FZdBZLzOT+r6+iciuEModtmCti+hjaAk0A== + +cookie-signature@1.0.6: + version "1.0.6" + resolved "https://registry.yarnpkg.com/cookie-signature/-/cookie-signature-1.0.6.tgz#e303a882b342cc3ee8ca513a79999734dab3ae2c" + integrity sha512-QADzlaHc8icV8I7vbaJXJwod9HWYp8uCqf1xa4OfNu1T7JVxQIrUgOWtHdNDtPiywmFbiS12VjotIXLrKM3orQ== + +cookie@0.5.0: + version "0.5.0" + resolved "https://registry.yarnpkg.com/cookie/-/cookie-0.5.0.tgz#d1f5d71adec6558c58f389987c366aa47e994f8b" + integrity sha512-YZ3GUyn/o8gfKJlnlX7g7xq4gyO6OSuhGPKaaGssGB2qgDUS0gPgtTvoyZLTt9Ab6dC4hfc9dV5arkvc/OCmrw== + +core-js-compat@^3.30.1, core-js-compat@^3.30.2: + version "3.31.0" + resolved "https://registry.yarnpkg.com/core-js-compat/-/core-js-compat-3.31.0.tgz#4030847c0766cc0e803dcdfb30055d7ef2064bf1" + integrity sha512-hM7YCu1cU6Opx7MXNu0NuumM0ezNeAeRKadixyiQELWY3vT3De9S4J5ZBMraWV2vZnrE1Cirl0GtFtDtMUXzPw== + dependencies: + browserslist "^4.21.5" + +core-util-is@~1.0.0: + version "1.0.3" + resolved "https://registry.yarnpkg.com/core-util-is/-/core-util-is-1.0.3.tgz#a6042d3634c2b27e9328f837b965fac83808db85" + integrity sha512-ZQBvi1DcpJ4GDqanjucZ2Hj3wEO5pZDS89BWbkcrvdxksJorwUDDZamX9ldFkp9aw2lmBDLgkObEA4DWNJ9FYQ== + +cross-spawn@^7.0.2, cross-spawn@^7.0.3: + version "7.0.3" + resolved "https://registry.yarnpkg.com/cross-spawn/-/cross-spawn-7.0.3.tgz#f73a85b9d5d41d045551c177e2882d4ac85728a6" + integrity sha512-iRDPJKUPVEND7dHPO8rkbOnPpyDygcDFtWjpeWNCgy8WP2rXcxXL8TskReQl6OrB2G7+UJrags1q15Fudc7G6w== + dependencies: + path-key "^3.1.0" + shebang-command "^2.0.0" + which "^2.0.1" + +css-loader@^6.8.1: + version "6.8.1" + resolved "https://registry.yarnpkg.com/css-loader/-/css-loader-6.8.1.tgz#0f8f52699f60f5e679eab4ec0fcd68b8e8a50a88" + integrity sha512-xDAXtEVGlD0gJ07iclwWVkLoZOpEvAWaSyf6W18S2pOC//K8+qUDIx8IIT3D+HjnmkJPQeesOPv5aiUaJsCM2g== + dependencies: + icss-utils "^5.1.0" + postcss "^8.4.21" + postcss-modules-extract-imports "^3.0.0" + postcss-modules-local-by-default "^4.0.3" + postcss-modules-scope "^3.0.0" + postcss-modules-values "^4.0.0" + postcss-value-parser "^4.2.0" + semver "^7.3.8" + +css-select@^4.1.3: + version "4.3.0" + resolved "https://registry.yarnpkg.com/css-select/-/css-select-4.3.0.tgz#db7129b2846662fd8628cfc496abb2b59e41529b" + integrity sha512-wPpOYtnsVontu2mODhA19JrqWxNsfdatRKd64kmpRbQgh1KtItko5sTnEpPdpSaJszTOhEMlF/RPz28qj4HqhQ== + dependencies: + boolbase "^1.0.0" + css-what "^6.0.1" + domhandler "^4.3.1" + domutils "^2.8.0" + nth-check "^2.0.1" + +css-what@^6.0.1: + version "6.1.0" + resolved "https://registry.yarnpkg.com/css-what/-/css-what-6.1.0.tgz#fb5effcf76f1ddea2c81bdfaa4de44e79bac70f4" + integrity sha512-HTUrgRJ7r4dsZKU6GjmpfRK1O76h97Z8MfS1G0FozR+oF2kG6Vfe8JE6zwrkbxigziPHinCJ+gCPjA9EaBDtRw== + +cssesc@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/cssesc/-/cssesc-3.0.0.tgz#37741919903b868565e1c09ea747445cd18983ee" + integrity sha512-/Tb/JcjK111nNScGob5MNtsntNM1aCNUDipB/TkwZFhyDrrE47SOx/18wF2bbjgc3ZzCSKW1T5nt5EbFoAz/Vg== + +debug@2.6.9: + version "2.6.9" + resolved "https://registry.yarnpkg.com/debug/-/debug-2.6.9.tgz#5d128515df134ff327e90a4c93f4e077a536341f" + integrity sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA== + dependencies: + ms "2.0.0" + +debug@^3.2.7: + version "3.2.7" + resolved "https://registry.yarnpkg.com/debug/-/debug-3.2.7.tgz#72580b7e9145fb39b6676f9c5e5fb100b934179a" + integrity sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ== + dependencies: + ms "^2.1.1" + +debug@^4.1.0, debug@^4.1.1, debug@^4.3.2, debug@^4.3.4: + version "4.3.4" + resolved "https://registry.yarnpkg.com/debug/-/debug-4.3.4.tgz#1319f6579357f2338d3337d2cdd4914bb5dcc865" + integrity sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ== + dependencies: + ms "2.1.2" + +deep-is@^0.1.3: + version "0.1.4" + resolved "https://registry.yarnpkg.com/deep-is/-/deep-is-0.1.4.tgz#a6f2dce612fadd2ef1f519b73551f17e85199831" + integrity sha512-oIPzksmTg4/MriiaYGO+okXDT7ztn/w3Eptv/+gSIdMdKsJo0u4CfYNFJPy+4SKMuCqGw2wxnA+URMg3t8a/bQ== + +default-gateway@^6.0.3: + version "6.0.3" + resolved "https://registry.yarnpkg.com/default-gateway/-/default-gateway-6.0.3.tgz#819494c888053bdb743edbf343d6cdf7f2943a71" + integrity sha512-fwSOJsbbNzZ/CUFpqFBqYfYNLj1NbMPm8MMCIzHjC83iSJRBEGmDUxU+WP661BaBQImeC2yHwXtz+P/O9o+XEg== + dependencies: + execa "^5.0.0" + +define-lazy-prop@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/define-lazy-prop/-/define-lazy-prop-2.0.0.tgz#3f7ae421129bcaaac9bc74905c98a0009ec9ee7f" + integrity sha512-Ds09qNh8yw3khSjiJjiUInaGX9xlqZDY7JVryGxdxV7NPeuqQfplOpQ66yJFZut3jLa5zOwkXw1g9EI2uKh4Og== + +define-properties@^1.1.3, define-properties@^1.1.4, define-properties@^1.2.0: + version "1.2.0" + resolved "https://registry.yarnpkg.com/define-properties/-/define-properties-1.2.0.tgz#52988570670c9eacedd8064f4a990f2405849bd5" + integrity sha512-xvqAVKGfT1+UAvPwKTVw/njhdQ8ZhXK4lI0bCIuCMrp2up9nPnaDftrLtmpTazqd1o+UY4zgzU+avtMbDP+ldA== + dependencies: + has-property-descriptors "^1.0.0" + object-keys "^1.1.1" + +depd@2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/depd/-/depd-2.0.0.tgz#b696163cc757560d09cf22cc8fad1571b79e76df" + integrity sha512-g7nH6P6dyDioJogAAGprGpCtVImJhpPk/roCzdb3fIh61/s/nPsfR6onyMwkCAR/OlC3yBC0lESvUoQEAssIrw== + +depd@~1.1.2: + version "1.1.2" + resolved "https://registry.yarnpkg.com/depd/-/depd-1.1.2.tgz#9bcd52e14c097763e749b274c4346ed2e560b5a9" + integrity sha512-7emPTl6Dpo6JRXOXjLRxck+FlLRX5847cLKEn00PLAgc3g2hTZZgr+e4c2v6QpSmLeFP3n5yUo7ft6avBK/5jQ== + +destroy@1.2.0: + version "1.2.0" + resolved "https://registry.yarnpkg.com/destroy/-/destroy-1.2.0.tgz#4803735509ad8be552934c67df614f94e66fa015" + integrity sha512-2sJGJTaXIIaR1w4iJSNoN0hnMY7Gpc/n8D4qSCJw8QqFWXf7cuAgnEHxBpweaVcPevC2l3KpjYCx3NypQQgaJg== + +detect-node@^2.0.4: + version "2.1.0" + resolved "https://registry.yarnpkg.com/detect-node/-/detect-node-2.1.0.tgz#c9c70775a49c3d03bc2c06d9a73be550f978f8b1" + integrity sha512-T0NIuQpnTvFDATNuHN5roPwSBG83rFsuO+MXXH9/3N1eFbn4wcPjttvjMLEPWJ0RGUYgQE7cGgS3tNxbqCGM7g== + +dns-equal@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/dns-equal/-/dns-equal-1.0.0.tgz#b39e7f1da6eb0a75ba9c17324b34753c47e0654d" + integrity sha512-z+paD6YUQsk+AbGCEM4PrOXSss5gd66QfcVBFTKR/HpFL9jCqikS94HYwKww6fQyO7IxrIIyUu+g0Ka9tUS2Cg== + +dns-packet@^5.2.2: + version "5.6.0" + resolved "https://registry.yarnpkg.com/dns-packet/-/dns-packet-5.6.0.tgz#2202c947845c7a63c23ece58f2f70ff6ab4c2f7d" + integrity sha512-rza3UH1LwdHh9qyPXp8lkwpjSNk/AMD3dPytUoRoqnypDUhY0xvbdmVhWOfxO68frEfV9BU8V12Ez7ZsHGZpCQ== + dependencies: + "@leichtgewicht/ip-codec" "^2.0.1" + +doctrine@^2.1.0: + version "2.1.0" + resolved "https://registry.yarnpkg.com/doctrine/-/doctrine-2.1.0.tgz#5cd01fc101621b42c4cd7f5d1a66243716d3f39d" + integrity sha512-35mSku4ZXK0vfCuHEDAwt55dg2jNajHZ1odvF+8SSr82EsZY4QmXfuWso8oEd8zRhVObSN18aM0CjSdoBX7zIw== + dependencies: + esutils "^2.0.2" + +doctrine@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/doctrine/-/doctrine-3.0.0.tgz#addebead72a6574db783639dc87a121773973961" + integrity sha512-yS+Q5i3hBf7GBkd4KG8a7eBNNWNGLTaEwwYWUijIYM7zrlYDM0BFXHjjPWlWZ1Rg7UaddZeIDmi9jF3HmqiQ2w== + dependencies: + esutils "^2.0.2" + +dom-converter@^0.2.0: + version "0.2.0" + resolved "https://registry.yarnpkg.com/dom-converter/-/dom-converter-0.2.0.tgz#6721a9daee2e293682955b6afe416771627bb768" + integrity sha512-gd3ypIPfOMr9h5jIKq8E3sHOTCjeirnl0WK5ZdS1AW0Odt0b1PaWaHdJ4Qk4klv+YB9aJBS7mESXjFoDQPu6DA== + dependencies: + utila "~0.4" + +dom-serializer@^1.0.1: + version "1.4.1" + resolved "https://registry.yarnpkg.com/dom-serializer/-/dom-serializer-1.4.1.tgz#de5d41b1aea290215dc45a6dae8adcf1d32e2d30" + integrity sha512-VHwB3KfrcOOkelEG2ZOfxqLZdfkil8PtJi4P8N2MMXucZq2yLp75ClViUlOVwyoHEDjYU433Aq+5zWP61+RGag== + dependencies: + domelementtype "^2.0.1" + domhandler "^4.2.0" + entities "^2.0.0" + +domelementtype@^2.0.1, domelementtype@^2.2.0: + version "2.3.0" + resolved "https://registry.yarnpkg.com/domelementtype/-/domelementtype-2.3.0.tgz#5c45e8e869952626331d7aab326d01daf65d589d" + integrity sha512-OLETBj6w0OsagBwdXnPdN0cnMfF9opN69co+7ZrbfPGrdpPVNBUj02spi6B1N7wChLQiPn4CSH/zJvXw56gmHw== + +domhandler@^4.0.0, domhandler@^4.2.0, domhandler@^4.3.1: + version "4.3.1" + resolved "https://registry.yarnpkg.com/domhandler/-/domhandler-4.3.1.tgz#8d792033416f59d68bc03a5aa7b018c1ca89279c" + integrity sha512-GrwoxYN+uWlzO8uhUXRl0P+kHE4GtVPfYzVLcUxPL7KNdHKj66vvlhiweIHqYYXWlw+T8iLMp42Lm67ghw4WMQ== + dependencies: + domelementtype "^2.2.0" + +domutils@^2.5.2, domutils@^2.8.0: + version "2.8.0" + resolved "https://registry.yarnpkg.com/domutils/-/domutils-2.8.0.tgz#4437def5db6e2d1f5d6ee859bd95ca7d02048135" + integrity sha512-w96Cjofp72M5IIhpjgobBimYEfoPjx1Vx0BSX9P30WBdZW2WIKU0T1Bd0kz2eNZ9ikjKgHbEyKx8BB6H1L3h3A== + dependencies: + dom-serializer "^1.0.1" + domelementtype "^2.2.0" + domhandler "^4.2.0" + +dot-case@^3.0.4: + version "3.0.4" + resolved "https://registry.yarnpkg.com/dot-case/-/dot-case-3.0.4.tgz#9b2b670d00a431667a8a75ba29cd1b98809ce751" + integrity sha512-Kv5nKlh6yRrdrGvxeJ2e5y2eRUpkUosIW4A2AS38zwSz27zu7ufDwQPi5Jhs3XAlGNetl3bmnGhQsMtkKJnj3w== + dependencies: + no-case "^3.0.4" + tslib "^2.0.3" + +eastasianwidth@^0.2.0: + version "0.2.0" + resolved "https://registry.yarnpkg.com/eastasianwidth/-/eastasianwidth-0.2.0.tgz#696ce2ec0aa0e6ea93a397ffcf24aa7840c827cb" + integrity sha512-I88TYZWc9XiYHRQ4/3c5rjjfgkjhLyW2luGIheGERbNQ6OY7yTybanSpDXZa8y7VUP9YmDcYa+eyq4ca7iLqWA== + +ee-first@1.1.1: + version "1.1.1" + resolved "https://registry.yarnpkg.com/ee-first/-/ee-first-1.1.1.tgz#590c61156b0ae2f4f0255732a158b266bc56b21d" + integrity sha512-WMwm9LhRUo+WUaRN+vRuETqG89IgZphVSNkdFgeb6sS/E4OrDIN7t48CAewSHXc6C8lefD8KKfr5vY61brQlow== + +electron-to-chromium@^1.4.431: + version "1.4.446" + resolved "https://registry.yarnpkg.com/electron-to-chromium/-/electron-to-chromium-1.4.446.tgz#c23fbc7884bfe43088d5eb908a858dbb35ab190b" + integrity sha512-4Gnw7ztEQ/E0eOt5JWfPn9jjeupfUlKoeW5ETKP9nLdWj+4spFoS3Stj19fqlKIaX28UQs0fNX+uKEyoLCBnkw== + +emoji-regex@^8.0.0: + version "8.0.0" + resolved "https://registry.yarnpkg.com/emoji-regex/-/emoji-regex-8.0.0.tgz#e818fd69ce5ccfcb404594f842963bf53164cc37" + integrity sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A== + +emoji-regex@^9.2.2: + version "9.2.2" + resolved "https://registry.yarnpkg.com/emoji-regex/-/emoji-regex-9.2.2.tgz#840c8803b0d8047f4ff0cf963176b32d4ef3ed72" + integrity sha512-L18DaJsXSUk2+42pv8mLs5jJT2hqFkFE4j21wOmgbUqsZ2hL72NsUU785g9RXgo3s0ZNgVl42TiHp3ZtOv/Vyg== + +encodeurl@~1.0.2: + version "1.0.2" + resolved "https://registry.yarnpkg.com/encodeurl/-/encodeurl-1.0.2.tgz#ad3ff4c86ec2d029322f5a02c3a9a606c95b3f59" + integrity sha512-TPJXq8JqFaVYm2CWmPvnP2Iyo4ZSM7/QKcSmuMLDObfpH5fi7RUGmd/rTDf+rut/saiDiQEeVTNgAmJEdAOx0w== + +enhanced-resolve@^5.15.0: + version "5.15.0" + resolved "https://registry.yarnpkg.com/enhanced-resolve/-/enhanced-resolve-5.15.0.tgz#1af946c7d93603eb88e9896cee4904dc012e9c35" + integrity sha512-LXYT42KJ7lpIKECr2mAXIaMldcNCh/7E0KBKOu4KSfkHmP+mZmSs+8V5gBAqisWBy0OO4W5Oyys0GO1Y8KtdKg== + dependencies: + graceful-fs "^4.2.4" + tapable "^2.2.0" + +entities@^2.0.0: + version "2.2.0" + resolved "https://registry.yarnpkg.com/entities/-/entities-2.2.0.tgz#098dc90ebb83d8dffa089d55256b351d34c4da55" + integrity sha512-p92if5Nz619I0w+akJrLZH0MX0Pb5DX39XOwQTtXSdQQOaYH03S1uIQp4mhOZtAXrxq4ViO67YTiLBo2638o9A== + +envinfo@^7.7.3: + version "7.10.0" + resolved "https://registry.yarnpkg.com/envinfo/-/envinfo-7.10.0.tgz#55146e3909cc5fe63c22da63fb15b05aeac35b13" + integrity sha512-ZtUjZO6l5mwTHvc1L9+1q5p/R3wTopcfqMW8r5t8SJSKqeVI/LtajORwRFEKpEFuekjD0VBjwu1HMxL4UalIRw== + +es-abstract@^1.19.0, es-abstract@^1.20.4: + version "1.21.2" + resolved "https://registry.yarnpkg.com/es-abstract/-/es-abstract-1.21.2.tgz#a56b9695322c8a185dc25975aa3b8ec31d0e7eff" + integrity sha512-y/B5POM2iBnIxCiernH1G7rC9qQoM77lLIMQLuob0zhp8C56Po81+2Nj0WFKnd0pNReDTnkYryc+zhOzpEIROg== + dependencies: + array-buffer-byte-length "^1.0.0" + available-typed-arrays "^1.0.5" + call-bind "^1.0.2" + es-set-tostringtag "^2.0.1" + es-to-primitive "^1.2.1" + function.prototype.name "^1.1.5" + get-intrinsic "^1.2.0" + get-symbol-description "^1.0.0" + globalthis "^1.0.3" + gopd "^1.0.1" + has "^1.0.3" + has-property-descriptors "^1.0.0" + has-proto "^1.0.1" + has-symbols "^1.0.3" + internal-slot "^1.0.5" + is-array-buffer "^3.0.2" + is-callable "^1.2.7" + is-negative-zero "^2.0.2" + is-regex "^1.1.4" + is-shared-array-buffer "^1.0.2" + is-string "^1.0.7" + is-typed-array "^1.1.10" + is-weakref "^1.0.2" + object-inspect "^1.12.3" + object-keys "^1.1.1" + object.assign "^4.1.4" + regexp.prototype.flags "^1.4.3" + safe-regex-test "^1.0.0" + string.prototype.trim "^1.2.7" + string.prototype.trimend "^1.0.6" + string.prototype.trimstart "^1.0.6" + typed-array-length "^1.0.4" + unbox-primitive "^1.0.2" + which-typed-array "^1.1.9" + +es-module-lexer@^1.2.1: + version "1.3.0" + resolved "https://registry.yarnpkg.com/es-module-lexer/-/es-module-lexer-1.3.0.tgz#6be9c9e0b4543a60cd166ff6f8b4e9dae0b0c16f" + integrity sha512-vZK7T0N2CBmBOixhmjdqx2gWVbFZ4DXZ/NyRMZVlJXPa7CyFS+/a4QQsDGDQy9ZfEzxFuNEsMLeQJnKP2p5/JA== + +es-set-tostringtag@^2.0.1: + version "2.0.1" + resolved "https://registry.yarnpkg.com/es-set-tostringtag/-/es-set-tostringtag-2.0.1.tgz#338d502f6f674301d710b80c8592de8a15f09cd8" + integrity sha512-g3OMbtlwY3QewlqAiMLI47KywjWZoEytKr8pf6iTC8uJq5bIAH52Z9pnQ8pVL6whrCto53JZDuUIsifGeLorTg== + dependencies: + get-intrinsic "^1.1.3" + has "^1.0.3" + has-tostringtag "^1.0.0" + +es-shim-unscopables@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/es-shim-unscopables/-/es-shim-unscopables-1.0.0.tgz#702e632193201e3edf8713635d083d378e510241" + integrity sha512-Jm6GPcCdC30eMLbZ2x8z2WuRwAws3zTBBKuusffYVUrNj/GVSUAZ+xKMaUpfNDR5IbyNA5LJbaecoUVbmUcB1w== + dependencies: + has "^1.0.3" + +es-to-primitive@^1.2.1: + version "1.2.1" + resolved "https://registry.yarnpkg.com/es-to-primitive/-/es-to-primitive-1.2.1.tgz#e55cd4c9cdc188bcefb03b366c736323fc5c898a" + integrity sha512-QCOllgZJtaUo9miYBcLChTUaHNjJF3PYs1VidD7AwiEj1kYxKeQTctLAezAOH5ZKRH0g2IgPn6KwB4IT8iRpvA== + dependencies: + is-callable "^1.1.4" + is-date-object "^1.0.1" + is-symbol "^1.0.2" + +escalade@^3.1.1: + version "3.1.1" + resolved "https://registry.yarnpkg.com/escalade/-/escalade-3.1.1.tgz#d8cfdc7000965c5a0174b4a82eaa5c0552742e40" + integrity sha512-k0er2gUkLf8O0zKJiAhmkTnJlTvINGv7ygDNPbeIsX/TJjGJZHuh9B2UxbsaEkmlEo9MfhrSzmhIlhRlI2GXnw== + +escape-html@~1.0.3: + version "1.0.3" + resolved "https://registry.yarnpkg.com/escape-html/-/escape-html-1.0.3.tgz#0258eae4d3d0c0974de1c169188ef0051d1d1988" + integrity sha512-NiSupZ4OeuGwr68lGIeym/ksIZMJodUGOSCZ/FSnTxcrekbvqrgdUxlJOMpijaKZVjAJrWrGs/6Jy8OMuyj9ow== + +escape-string-regexp@^1.0.5: + version "1.0.5" + resolved "https://registry.yarnpkg.com/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz#1b61c0562190a8dff6ae3bb2cf0200ca130b86d4" + integrity sha512-vbRorB5FUQWvla16U8R/qgaFIya2qGzwDrNmCZuYKrbdSUMG6I1ZCGQRefkRVhuOkIGVne7BQ35DSfo1qvJqFg== + +escape-string-regexp@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/escape-string-regexp/-/escape-string-regexp-4.0.0.tgz#14ba83a5d373e3d311e5afca29cf5bfad965bf34" + integrity sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA== + +eslint-config-airbnb-base@^15.0.0: + version "15.0.0" + resolved "https://registry.yarnpkg.com/eslint-config-airbnb-base/-/eslint-config-airbnb-base-15.0.0.tgz#6b09add90ac79c2f8d723a2580e07f3925afd236" + integrity sha512-xaX3z4ZZIcFLvh2oUNvcX5oEofXda7giYmuplVxoOg5A7EXJMrUyqRgR+mhDhPK8LZ4PttFOBvCYDbX3sUoUig== + dependencies: + confusing-browser-globals "^1.0.10" + object.assign "^4.1.2" + object.entries "^1.1.5" + semver "^6.3.0" + +eslint-config-prettier@^8.8.0: + version "8.8.0" + resolved "https://registry.yarnpkg.com/eslint-config-prettier/-/eslint-config-prettier-8.8.0.tgz#bfda738d412adc917fd7b038857110efe98c9348" + integrity sha512-wLbQiFre3tdGgpDv67NQKnJuTlcUVYHas3k+DZCc2U2BadthoEY4B7hLPvAxaqdyOGCzuLfii2fqGph10va7oA== + +eslint-import-resolver-node@^0.3.7: + version "0.3.7" + resolved "https://registry.yarnpkg.com/eslint-import-resolver-node/-/eslint-import-resolver-node-0.3.7.tgz#83b375187d412324a1963d84fa664377a23eb4d7" + integrity sha512-gozW2blMLJCeFpBwugLTGyvVjNoeo1knonXAcatC6bjPBZitotxdWf7Gimr25N4c0AAOo4eOUfaG82IJPDpqCA== + dependencies: + debug "^3.2.7" + is-core-module "^2.11.0" + resolve "^1.22.1" + +eslint-module-utils@^2.7.4: + version "2.8.0" + resolved "https://registry.yarnpkg.com/eslint-module-utils/-/eslint-module-utils-2.8.0.tgz#e439fee65fc33f6bba630ff621efc38ec0375c49" + integrity sha512-aWajIYfsqCKRDgUfjEXNN/JlrzauMuSEy5sbd7WXbtW3EH6A6MpwEh42c7qD+MqQo9QMJ6fWLAeIJynx0g6OAw== + dependencies: + debug "^3.2.7" + +eslint-plugin-import@^2.27.5: + version "2.27.5" + resolved "https://registry.yarnpkg.com/eslint-plugin-import/-/eslint-plugin-import-2.27.5.tgz#876a6d03f52608a3e5bb439c2550588e51dd6c65" + integrity sha512-LmEt3GVofgiGuiE+ORpnvP+kAm3h6MLZJ4Q5HCyHADofsb4VzXFsRiWj3c0OFiV+3DWFh0qg3v9gcPlfc3zRow== + dependencies: + array-includes "^3.1.6" + array.prototype.flat "^1.3.1" + array.prototype.flatmap "^1.3.1" + debug "^3.2.7" + doctrine "^2.1.0" + eslint-import-resolver-node "^0.3.7" + eslint-module-utils "^2.7.4" + has "^1.0.3" + is-core-module "^2.11.0" + is-glob "^4.0.3" + minimatch "^3.1.2" + object.values "^1.1.6" + resolve "^1.22.1" + semver "^6.3.0" + tsconfig-paths "^3.14.1" + +eslint-scope@5.1.1: + version "5.1.1" + resolved "https://registry.yarnpkg.com/eslint-scope/-/eslint-scope-5.1.1.tgz#e786e59a66cb92b3f6c1fb0d508aab174848f48c" + integrity sha512-2NxwbF/hZ0KpepYN0cNbo+FN6XoK7GaHlQhgx/hIZl6Va0bF45RQOOwhLIy8lQDbuCiadSLCBnH2CFYquit5bw== + dependencies: + esrecurse "^4.3.0" + estraverse "^4.1.1" + +eslint-scope@^7.2.0: + version "7.2.0" + resolved "https://registry.yarnpkg.com/eslint-scope/-/eslint-scope-7.2.0.tgz#f21ebdafda02352f103634b96dd47d9f81ca117b" + integrity sha512-DYj5deGlHBfMt15J7rdtyKNq/Nqlv5KfU4iodrQ019XESsRnwXH9KAE0y3cwtUHDo2ob7CypAnCqefh6vioWRw== + dependencies: + esrecurse "^4.3.0" + estraverse "^5.2.0" + +eslint-visitor-keys@^3.3.0, eslint-visitor-keys@^3.4.1: + version "3.4.1" + resolved "https://registry.yarnpkg.com/eslint-visitor-keys/-/eslint-visitor-keys-3.4.1.tgz#c22c48f48942d08ca824cc526211ae400478a994" + integrity sha512-pZnmmLwYzf+kWaM/Qgrvpen51upAktaaiI01nsJD/Yr3lMOdNtq0cxkrrg16w64VtisN6okbs7Q8AfGqj4c9fA== + +eslint@^8.43.0: + version "8.43.0" + resolved "https://registry.yarnpkg.com/eslint/-/eslint-8.43.0.tgz#3e8c6066a57097adfd9d390b8fc93075f257a094" + integrity sha512-aaCpf2JqqKesMFGgmRPessmVKjcGXqdlAYLLC3THM8t5nBRZRQ+st5WM/hoJXkdioEXLLbXgclUpM0TXo5HX5Q== + dependencies: + "@eslint-community/eslint-utils" "^4.2.0" + "@eslint-community/regexpp" "^4.4.0" + "@eslint/eslintrc" "^2.0.3" + "@eslint/js" "8.43.0" + "@humanwhocodes/config-array" "^0.11.10" + "@humanwhocodes/module-importer" "^1.0.1" + "@nodelib/fs.walk" "^1.2.8" + ajv "^6.10.0" + chalk "^4.0.0" + cross-spawn "^7.0.2" + debug "^4.3.2" + doctrine "^3.0.0" + escape-string-regexp "^4.0.0" + eslint-scope "^7.2.0" + eslint-visitor-keys "^3.4.1" + espree "^9.5.2" + esquery "^1.4.2" + esutils "^2.0.2" + fast-deep-equal "^3.1.3" + file-entry-cache "^6.0.1" + find-up "^5.0.0" + glob-parent "^6.0.2" + globals "^13.19.0" + graphemer "^1.4.0" + ignore "^5.2.0" + import-fresh "^3.0.0" + imurmurhash "^0.1.4" + is-glob "^4.0.0" + is-path-inside "^3.0.3" + js-yaml "^4.1.0" + json-stable-stringify-without-jsonify "^1.0.1" + levn "^0.4.1" + lodash.merge "^4.6.2" + minimatch "^3.1.2" + natural-compare "^1.4.0" + optionator "^0.9.1" + strip-ansi "^6.0.1" + strip-json-comments "^3.1.0" + text-table "^0.2.0" + +espree@^9.5.2: + version "9.5.2" + resolved "https://registry.yarnpkg.com/espree/-/espree-9.5.2.tgz#e994e7dc33a082a7a82dceaf12883a829353215b" + integrity sha512-7OASN1Wma5fum5SrNhFMAMJxOUAbhyfQ8dQ//PJaJbNw0URTPWqIghHWt1MmAANKhHZIYOHruW4Kw4ruUWOdGw== + dependencies: + acorn "^8.8.0" + acorn-jsx "^5.3.2" + eslint-visitor-keys "^3.4.1" + +esquery@^1.4.2: + version "1.5.0" + resolved "https://registry.yarnpkg.com/esquery/-/esquery-1.5.0.tgz#6ce17738de8577694edd7361c57182ac8cb0db0b" + integrity sha512-YQLXUplAwJgCydQ78IMJywZCceoqk1oH01OERdSAJc/7U2AylwjhSCLDEtqwg811idIS/9fIU5GjG73IgjKMVg== + dependencies: + estraverse "^5.1.0" + +esrecurse@^4.3.0: + version "4.3.0" + resolved "https://registry.yarnpkg.com/esrecurse/-/esrecurse-4.3.0.tgz#7ad7964d679abb28bee72cec63758b1c5d2c9921" + integrity sha512-KmfKL3b6G+RXvP8N1vr3Tq1kL/oCFgn2NYXEtqP8/L3pKapUA4G8cFVaoF3SU323CD4XypR/ffioHmkti6/Tag== + dependencies: + estraverse "^5.2.0" + +estraverse@^4.1.1: + version "4.3.0" + resolved "https://registry.yarnpkg.com/estraverse/-/estraverse-4.3.0.tgz#398ad3f3c5a24948be7725e83d11a7de28cdbd1d" + integrity sha512-39nnKffWz8xN1BU/2c79n9nB9HDzo0niYUqx6xyqUnyoAnQyyWpOTdZEeiCch8BBu515t4wp9ZmgVfVhn9EBpw== + +estraverse@^5.1.0, estraverse@^5.2.0: + version "5.3.0" + resolved "https://registry.yarnpkg.com/estraverse/-/estraverse-5.3.0.tgz#2eea5290702f26ab8fe5370370ff86c965d21123" + integrity sha512-MMdARuVEQziNTeJD8DgMqmhwR11BRQ/cBP+pLtYdSTnf3MIO8fFeiINEbX36ZdNlfU/7A9f3gUw49B3oQsvwBA== + +esutils@^2.0.2: + version "2.0.3" + resolved "https://registry.yarnpkg.com/esutils/-/esutils-2.0.3.tgz#74d2eb4de0b8da1293711910d50775b9b710ef64" + integrity sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g== + +etag@~1.8.1: + version "1.8.1" + resolved "https://registry.yarnpkg.com/etag/-/etag-1.8.1.tgz#41ae2eeb65efa62268aebfea83ac7d79299b0887" + integrity sha512-aIL5Fx7mawVa300al2BnEE4iNvo1qETxLrPI/o05L7z6go7fCw1J6EQmbK4FmJ2AS7kgVF/KEZWufBfdClMcPg== + +eventemitter3@^4.0.0: + version "4.0.7" + resolved "https://registry.yarnpkg.com/eventemitter3/-/eventemitter3-4.0.7.tgz#2de9b68f6528d5644ef5c59526a1b4a07306169f" + integrity sha512-8guHBZCwKnFhYdHr2ysuRWErTwhoN2X8XELRlrRwpmfeY2jjuUN4taQMsULKUVo1K4DvZl+0pgfyoysHxvmvEw== + +events@^3.2.0: + version "3.3.0" + resolved "https://registry.yarnpkg.com/events/-/events-3.3.0.tgz#31a95ad0a924e2d2c419a813aeb2c4e878ea7400" + integrity sha512-mQw+2fkQbALzQ7V0MY0IqdnXNOeTtP4r0lN9z7AAawCXgqea7bDii20AYrIBrFd/Hx0M2Ocz6S111CaFkUcb0Q== + +execa@^5.0.0: + version "5.1.1" + resolved "https://registry.yarnpkg.com/execa/-/execa-5.1.1.tgz#f80ad9cbf4298f7bd1d4c9555c21e93741c411dd" + integrity sha512-8uSpZZocAZRBAPIEINJj3Lo9HyGitllczc27Eh5YYojjMFMn8yHMDMaUHE2Jqfq05D/wucwI4JGURyXt1vchyg== + dependencies: + cross-spawn "^7.0.3" + get-stream "^6.0.0" + human-signals "^2.1.0" + is-stream "^2.0.0" + merge-stream "^2.0.0" + npm-run-path "^4.0.1" + onetime "^5.1.2" + signal-exit "^3.0.3" + strip-final-newline "^2.0.0" + +execa@^7.0.0: + version "7.1.1" + resolved "https://registry.yarnpkg.com/execa/-/execa-7.1.1.tgz#3eb3c83d239488e7b409d48e8813b76bb55c9c43" + integrity sha512-wH0eMf/UXckdUYnO21+HDztteVv05rq2GXksxT4fCGeHkBhw1DROXh40wcjMcRqDOWE7iPJ4n3M7e2+YFP+76Q== + dependencies: + cross-spawn "^7.0.3" + get-stream "^6.0.1" + human-signals "^4.3.0" + is-stream "^3.0.0" + merge-stream "^2.0.0" + npm-run-path "^5.1.0" + onetime "^6.0.0" + signal-exit "^3.0.7" + strip-final-newline "^3.0.0" + +express@^4.17.3: + version "4.18.2" + resolved "https://registry.yarnpkg.com/express/-/express-4.18.2.tgz#3fabe08296e930c796c19e3c516979386ba9fd59" + integrity sha512-5/PsL6iGPdfQ/lKM1UuielYgv3BUoJfz1aUwU9vHZ+J7gyvwdQXFEBIEIaxeGf0GIcreATNyBExtalisDbuMqQ== + dependencies: + accepts "~1.3.8" + array-flatten "1.1.1" + body-parser "1.20.1" + content-disposition "0.5.4" + content-type "~1.0.4" + cookie "0.5.0" + cookie-signature "1.0.6" + debug "2.6.9" + depd "2.0.0" + encodeurl "~1.0.2" + escape-html "~1.0.3" + etag "~1.8.1" + finalhandler "1.2.0" + fresh "0.5.2" + http-errors "2.0.0" + merge-descriptors "1.0.1" + methods "~1.1.2" + on-finished "2.4.1" + parseurl "~1.3.3" + path-to-regexp "0.1.7" + proxy-addr "~2.0.7" + qs "6.11.0" + range-parser "~1.2.1" + safe-buffer "5.2.1" + send "0.18.0" + serve-static "1.15.0" + setprototypeof "1.2.0" + statuses "2.0.1" + type-is "~1.6.18" + utils-merge "1.0.1" + vary "~1.1.2" + +fast-deep-equal@^3.1.1, fast-deep-equal@^3.1.3: + version "3.1.3" + resolved "https://registry.yarnpkg.com/fast-deep-equal/-/fast-deep-equal-3.1.3.tgz#3a7d56b559d6cbc3eb512325244e619a65c6c525" + integrity sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q== + +fast-json-stable-stringify@^2.0.0: + version "2.1.0" + resolved "https://registry.yarnpkg.com/fast-json-stable-stringify/-/fast-json-stable-stringify-2.1.0.tgz#874bf69c6f404c2b5d99c481341399fd55892633" + integrity sha512-lhd/wF+Lk98HZoTCtlVraHtfh5XYijIjalXck7saUtuanSDyLMxnHhSXEDJqHxD7msR8D0uCmqlkwjCV8xvwHw== + +fast-levenshtein@^2.0.6: + version "2.0.6" + resolved "https://registry.yarnpkg.com/fast-levenshtein/-/fast-levenshtein-2.0.6.tgz#3d8a5c66883a16a30ca8643e851f19baa7797917" + integrity sha512-DCXu6Ifhqcks7TZKY3Hxp3y6qphY5SJZmrWMDrKcERSOXWQdMhU9Ig/PYrzyw/ul9jOIyh0N4M0tbC5hodg8dw== + +fastest-levenshtein@^1.0.12: + version "1.0.16" + resolved "https://registry.yarnpkg.com/fastest-levenshtein/-/fastest-levenshtein-1.0.16.tgz#210e61b6ff181de91ea9b3d1b84fdedd47e034e5" + integrity sha512-eRnCtTTtGZFpQCwhJiUOuxPQWRXVKYDn0b2PeHfXL6/Zi53SLAzAHfVhVWK2AryC/WH05kGfxhFIPvTF0SXQzg== + +fastq@^1.6.0: + version "1.15.0" + resolved "https://registry.yarnpkg.com/fastq/-/fastq-1.15.0.tgz#d04d07c6a2a68fe4599fea8d2e103a937fae6b3a" + integrity sha512-wBrocU2LCXXa+lWBt8RoIRD89Fi8OdABODa/kEnyeyjS5aZO5/GNvI5sEINADqP/h8M29UHTHUb53sUu5Ihqdw== + dependencies: + reusify "^1.0.4" + +faye-websocket@^0.11.3: + version "0.11.4" + resolved "https://registry.yarnpkg.com/faye-websocket/-/faye-websocket-0.11.4.tgz#7f0d9275cfdd86a1c963dc8b65fcc451edcbb1da" + integrity sha512-CzbClwlXAuiRQAlUyfqPgvPoNKTckTPGfwZV4ZdAhVcP2lh9KUxJg2b5GkE7XbjKQ3YJnQ9z6D9ntLAlB+tP8g== + dependencies: + websocket-driver ">=0.5.1" + +file-entry-cache@^6.0.1: + version "6.0.1" + resolved "https://registry.yarnpkg.com/file-entry-cache/-/file-entry-cache-6.0.1.tgz#211b2dd9659cb0394b073e7323ac3c933d522027" + integrity sha512-7Gps/XWymbLk2QLYK4NzpMOrYjMhdIxXuIvy2QBsLE6ljuodKvdkWs/cpyJJ3CVIVpH0Oi1Hvg1ovbMzLdFBBg== + dependencies: + flat-cache "^3.0.4" + +fill-range@^7.0.1: + version "7.0.1" + resolved "https://registry.yarnpkg.com/fill-range/-/fill-range-7.0.1.tgz#1919a6a7c75fe38b2c7c77e5198535da9acdda40" + integrity sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ== + dependencies: + to-regex-range "^5.0.1" + +finalhandler@1.2.0: + version "1.2.0" + resolved "https://registry.yarnpkg.com/finalhandler/-/finalhandler-1.2.0.tgz#7d23fe5731b207b4640e4fcd00aec1f9207a7b32" + integrity sha512-5uXcUVftlQMFnWC9qu/svkWv3GTd2PfUhK/3PLkYNAe7FbqJMt3515HaxE6eRL74GdsriiwujiawdaB1BpEISg== + dependencies: + debug "2.6.9" + encodeurl "~1.0.2" + escape-html "~1.0.3" + on-finished "2.4.1" + parseurl "~1.3.3" + statuses "2.0.1" + unpipe "~1.0.0" + +find-babel-config@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/find-babel-config/-/find-babel-config-2.0.0.tgz#a8216f825415a839d0f23f4d18338a1cc966f701" + integrity sha512-dOKT7jvF3hGzlW60Gc3ONox/0rRZ/tz7WCil0bqA1In/3I8f1BctpXahRnEKDySZqci7u+dqq93sZST9fOJpFw== + dependencies: + json5 "^2.1.1" + path-exists "^4.0.0" + +find-cache-dir@^3.3.2: + version "3.3.2" + resolved "https://registry.yarnpkg.com/find-cache-dir/-/find-cache-dir-3.3.2.tgz#b30c5b6eff0730731aea9bbd9dbecbd80256d64b" + integrity sha512-wXZV5emFEjrridIgED11OoUKLxiYjAcqot/NJdAkOhlJ+vGzwhOAfcG5OX1jP+S0PcjEn8bdMJv+g2jwQ3Onig== + dependencies: + commondir "^1.0.1" + make-dir "^3.0.2" + pkg-dir "^4.1.0" + +find-up@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/find-up/-/find-up-3.0.0.tgz#49169f1d7993430646da61ecc5ae355c21c97b73" + integrity sha512-1yD6RmLI1XBfxugvORwlck6f75tYL+iR0jqwsOrOxMZyGYqUuDhJ0l4AXdO1iX/FTs9cBAMEk1gWSEx1kSbylg== + dependencies: + locate-path "^3.0.0" + +find-up@^4.0.0: + version "4.1.0" + resolved "https://registry.yarnpkg.com/find-up/-/find-up-4.1.0.tgz#97afe7d6cdc0bc5928584b7c8d7b16e8a9aa5d19" + integrity sha512-PpOwAdQ/YlXQ2vj8a3h8IipDuYRi3wceVQQGYWxNINccq40Anw7BlsEXCMbt1Zt+OLA6Fq9suIpIWD0OsnISlw== + dependencies: + locate-path "^5.0.0" + path-exists "^4.0.0" + +find-up@^5.0.0: + version "5.0.0" + resolved "https://registry.yarnpkg.com/find-up/-/find-up-5.0.0.tgz#4c92819ecb7083561e4f4a240a86be5198f536fc" + integrity sha512-78/PXT1wlLLDgTzDs7sjq9hzz0vXD+zn+7wypEe4fXQxCmdmqfGsEPQxmiCSQI3ajFV91bVSsvNtrJRiW6nGng== + dependencies: + locate-path "^6.0.0" + path-exists "^4.0.0" + +flat-cache@^3.0.4: + version "3.0.4" + resolved "https://registry.yarnpkg.com/flat-cache/-/flat-cache-3.0.4.tgz#61b0338302b2fe9f957dcc32fc2a87f1c3048b11" + integrity sha512-dm9s5Pw7Jc0GvMYbshN6zchCA9RgQlzzEZX3vylR9IqFfS8XciblUXOKfW6SiuJ0e13eDYZoZV5wdrev7P3Nwg== + dependencies: + flatted "^3.1.0" + rimraf "^3.0.2" + +flatted@^3.1.0: + version "3.2.7" + resolved "https://registry.yarnpkg.com/flatted/-/flatted-3.2.7.tgz#609f39207cb614b89d0765b477cb2d437fbf9787" + integrity sha512-5nqDSxl8nn5BSNxyR3n4I6eDmbolI6WT+QqR547RwxQapgjQBmtktdP+HTBb/a/zLsbzERTONyUB5pefh5TtjQ== + +follow-redirects@^1.0.0: + version "1.15.2" + resolved "https://registry.yarnpkg.com/follow-redirects/-/follow-redirects-1.15.2.tgz#b460864144ba63f2681096f274c4e57026da2c13" + integrity sha512-VQLG33o04KaQ8uYi2tVNbdrWp1QWxNNea+nmIB4EVM28v0hmP17z7aG1+wAkNzVq4KeXTq3221ye5qTJP91JwA== + +for-each@^0.3.3: + version "0.3.3" + resolved "https://registry.yarnpkg.com/for-each/-/for-each-0.3.3.tgz#69b447e88a0a5d32c3e7084f3f1710034b21376e" + integrity sha512-jqYfLp7mo9vIyQf8ykW2v7A+2N4QjeCeI5+Dz9XraiO1ign81wjiH7Fb9vSOWvQfNtmSa4H2RoQTrrXivdUZmw== + dependencies: + is-callable "^1.1.3" + +forwarded@0.2.0: + version "0.2.0" + resolved "https://registry.yarnpkg.com/forwarded/-/forwarded-0.2.0.tgz#2269936428aad4c15c7ebe9779a84bf0b2a81811" + integrity sha512-buRG0fpBtRHSTCOASe6hD258tEubFoRLb4ZNA6NxMVHNw2gOcwHo9wyablzMzOA5z9xA9L1KNjk/Nt6MT9aYow== + +fresh@0.5.2: + version "0.5.2" + resolved "https://registry.yarnpkg.com/fresh/-/fresh-0.5.2.tgz#3d8cadd90d976569fa835ab1f8e4b23a105605a7" + integrity sha512-zJ2mQYM18rEFOudeV4GShTGIQ7RbzA7ozbU9I/XBpm7kqgMywgmylMwXHxZJmkVoYkna9d2pVXVXPdYTP9ej8Q== + +fs-monkey@^1.0.4: + version "1.0.4" + resolved "https://registry.yarnpkg.com/fs-monkey/-/fs-monkey-1.0.4.tgz#ee8c1b53d3fe8bb7e5d2c5c5dfc0168afdd2f747" + integrity sha512-INM/fWAxMICjttnD0DX1rBvinKskj5G1w+oy/pnm9u/tSlnBrzFonJMcalKJ30P8RRsPzKcCG7Q8l0jx5Fh9YQ== + +fs.realpath@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/fs.realpath/-/fs.realpath-1.0.0.tgz#1504ad2523158caa40db4a2787cb01411994ea4f" + integrity sha512-OO0pH2lK6a0hZnAdau5ItzHPI6pUlvI7jMVnxUQRtw4owF2wk8lOSabtGDCTP4Ggrg2MbGnWO9X8K1t4+fGMDw== + +fsevents@~2.3.2: + version "2.3.2" + resolved "https://registry.yarnpkg.com/fsevents/-/fsevents-2.3.2.tgz#8a526f78b8fdf4623b709e0b975c52c24c02fd1a" + integrity sha512-xiqMQR4xAeHTuB9uWm+fFRcIOgKBMiOBP+eXiyT7jsgVCq1bkVygt00oASowB7EdtpOHaaPgKt812P9ab+DDKA== + +function-bind@^1.1.1: + version "1.1.1" + resolved "https://registry.yarnpkg.com/function-bind/-/function-bind-1.1.1.tgz#a56899d3ea3c9bab874bb9773b7c5ede92f4895d" + integrity sha512-yIovAzMX49sF8Yl58fSCWJ5svSLuaibPxXQJFLmBObTuCr0Mf1KiPopGM9NiFjiYBCbfaa2Fh6breQ6ANVTI0A== + +function.prototype.name@^1.1.5: + version "1.1.5" + resolved "https://registry.yarnpkg.com/function.prototype.name/-/function.prototype.name-1.1.5.tgz#cce0505fe1ffb80503e6f9e46cc64e46a12a9621" + integrity sha512-uN7m/BzVKQnCUF/iW8jYea67v++2u7m5UgENbHRtdDVclOUP+FMPlCNdmk0h/ysGyo2tavMJEDqJAkJdRa1vMA== + dependencies: + call-bind "^1.0.2" + define-properties "^1.1.3" + es-abstract "^1.19.0" + functions-have-names "^1.2.2" + +functions-have-names@^1.2.2, functions-have-names@^1.2.3: + version "1.2.3" + resolved "https://registry.yarnpkg.com/functions-have-names/-/functions-have-names-1.2.3.tgz#0404fe4ee2ba2f607f0e0ec3c80bae994133b834" + integrity sha512-xckBUXyTIqT97tq2x2AMb+g163b5JFysYk0x4qxNFwbfQkmNZoiRHb6sPzI9/QV33WeuvVYBUIiD4NzNIyqaRQ== + +gensync@^1.0.0-beta.2: + version "1.0.0-beta.2" + resolved "https://registry.yarnpkg.com/gensync/-/gensync-1.0.0-beta.2.tgz#32a6ee76c3d7f52d46b2b1ae5d93fea8580a25e0" + integrity sha512-3hN7NaskYvMDLQY55gnW3NQ+mesEAepTqlg+VEbj7zzqEMBVNhzcGYYeqFo/TlYz6eQiFcp1HcsCZO+nGgS8zg== + +get-intrinsic@^1.0.2, get-intrinsic@^1.1.1, get-intrinsic@^1.1.3, get-intrinsic@^1.2.0: + version "1.2.1" + resolved "https://registry.yarnpkg.com/get-intrinsic/-/get-intrinsic-1.2.1.tgz#d295644fed4505fc9cde952c37ee12b477a83d82" + integrity sha512-2DcsyfABl+gVHEfCOaTrWgyt+tb6MSEGmKq+kI5HwLbIYgjgmMcV8KQ41uaKz1xxUcn9tJtgFbQUEVcEbd0FYw== + dependencies: + function-bind "^1.1.1" + has "^1.0.3" + has-proto "^1.0.1" + has-symbols "^1.0.3" + +get-stream@^6.0.0, get-stream@^6.0.1: + version "6.0.1" + resolved "https://registry.yarnpkg.com/get-stream/-/get-stream-6.0.1.tgz#a262d8eef67aced57c2852ad6167526a43cbf7b7" + integrity sha512-ts6Wi+2j3jQjqi70w5AlN8DFnkSwC+MqmxEzdEALB2qXZYV3X/b1CTfgPLGJNMeAWxdPfU8FO1ms3NUfaHCPYg== + +get-symbol-description@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/get-symbol-description/-/get-symbol-description-1.0.0.tgz#7fdb81c900101fbd564dd5f1a30af5aadc1e58d6" + integrity sha512-2EmdH1YvIQiZpltCNgkuiUnyukzxM/R6NDJX31Ke3BG1Nq5b0S2PhX59UKi9vZpPDQVdqn+1IcaAwnzTT5vCjw== + dependencies: + call-bind "^1.0.2" + get-intrinsic "^1.1.1" + +glob-parent@^6.0.2: + version "6.0.2" + resolved "https://registry.yarnpkg.com/glob-parent/-/glob-parent-6.0.2.tgz#6d237d99083950c79290f24c7642a3de9a28f9e3" + integrity sha512-XxwI8EOhVQgWp6iDL+3b0r86f4d6AX6zSU55HfB4ydCEuXLXc5FcYeOu+nnGftS4TEju/11rt4KJPTMgbfmv4A== + dependencies: + is-glob "^4.0.3" + +glob-parent@~5.1.2: + version "5.1.2" + resolved "https://registry.yarnpkg.com/glob-parent/-/glob-parent-5.1.2.tgz#869832c58034fe68a4093c17dc15e8340d8401c4" + integrity sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow== + dependencies: + is-glob "^4.0.1" + +glob-to-regexp@^0.4.1: + version "0.4.1" + resolved "https://registry.yarnpkg.com/glob-to-regexp/-/glob-to-regexp-0.4.1.tgz#c75297087c851b9a578bd217dd59a92f59fe546e" + integrity sha512-lkX1HJXwyMcprw/5YUZc2s7DrpAiHB21/V+E1rHUrVNokkvB6bqMzT0VfV6/86ZNabt1k14YOIaT7nDvOX3Iiw== + +glob@^7.1.3: + version "7.2.3" + resolved "https://registry.yarnpkg.com/glob/-/glob-7.2.3.tgz#b8df0fb802bbfa8e89bd1d938b4e16578ed44f2b" + integrity sha512-nFR0zLpU2YCaRxwoCJvL6UvCH2JFyFVIvwTLsIf21AuHlMskA1hhTdk+LlYJtOlYt9v6dvszD2BGRqBL+iQK9Q== + dependencies: + fs.realpath "^1.0.0" + inflight "^1.0.4" + inherits "2" + minimatch "^3.1.1" + once "^1.3.0" + path-is-absolute "^1.0.0" + +glob@^8.0.3: + version "8.1.0" + resolved "https://registry.yarnpkg.com/glob/-/glob-8.1.0.tgz#d388f656593ef708ee3e34640fdfb99a9fd1c33e" + integrity sha512-r8hpEjiQEYlF2QU0df3dS+nxxSIreXQS1qRhMJM0Q5NDdR386C7jb7Hwwod8Fgiuex+k0GFjgft18yvxm5XoCQ== + dependencies: + fs.realpath "^1.0.0" + inflight "^1.0.4" + inherits "2" + minimatch "^5.0.1" + once "^1.3.0" + +globals@^11.1.0: + version "11.12.0" + resolved "https://registry.yarnpkg.com/globals/-/globals-11.12.0.tgz#ab8795338868a0babd8525758018c2a7eb95c42e" + integrity sha512-WOBp/EEGUiIsJSp7wcv/y6MO+lV9UoncWqxuFfm8eBwzWNgyfBd6Gz+IeKQ9jCmyhoH99g15M3T+QaVHFjizVA== + +globals@^13.19.0: + version "13.20.0" + resolved "https://registry.yarnpkg.com/globals/-/globals-13.20.0.tgz#ea276a1e508ffd4f1612888f9d1bad1e2717bf82" + integrity sha512-Qg5QtVkCy/kv3FUSlu4ukeZDVf9ee0iXLAUYX13gbR17bnejFTzr4iS9bY7kwCf1NztRNm1t91fjOiyx4CSwPQ== + dependencies: + type-fest "^0.20.2" + +globalthis@^1.0.3: + version "1.0.3" + resolved "https://registry.yarnpkg.com/globalthis/-/globalthis-1.0.3.tgz#5852882a52b80dc301b0660273e1ed082f0b6ccf" + integrity sha512-sFdI5LyBiNTHjRd7cGPWapiHWMOXKyuBNX/cWJ3NfzrZQVa8GI/8cofCl74AOVqq9W5kNmguTIzJ/1s2gyI9wA== + dependencies: + define-properties "^1.1.3" + +gopd@^1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/gopd/-/gopd-1.0.1.tgz#29ff76de69dac7489b7c0918a5788e56477c332c" + integrity sha512-d65bNlIadxvpb/A2abVdlqKqV563juRnZ1Wtk6s1sIR8uNsXR70xqIzVqxVf1eTqDunwT2MkczEeaezCKTZhwA== + dependencies: + get-intrinsic "^1.1.3" + +graceful-fs@^4.1.2, graceful-fs@^4.2.4, graceful-fs@^4.2.6, graceful-fs@^4.2.9: + version "4.2.11" + resolved "https://registry.yarnpkg.com/graceful-fs/-/graceful-fs-4.2.11.tgz#4183e4e8bf08bb6e05bbb2f7d2e0c8f712ca40e3" + integrity sha512-RbJ5/jmFcNNCcDV5o9eTnBLJ/HszWV0P73bc+Ff4nS/rJj+YaS6IGyiOL0VoBYX+l1Wrl3k63h/KrH+nhJ0XvQ== + +graphemer@^1.4.0: + version "1.4.0" + resolved "https://registry.yarnpkg.com/graphemer/-/graphemer-1.4.0.tgz#fb2f1d55e0e3a1849aeffc90c4fa0dd53a0e66c6" + integrity sha512-EtKwoO6kxCL9WO5xipiHTZlSzBm7WLT627TqC/uVRd0HKmq8NXyebnNYxDoBi7wt8eTWrUrKXCOVaFq9x1kgag== + +handle-thing@^2.0.0: + version "2.0.1" + resolved "https://registry.yarnpkg.com/handle-thing/-/handle-thing-2.0.1.tgz#857f79ce359580c340d43081cc648970d0bb234e" + integrity sha512-9Qn4yBxelxoh2Ow62nP+Ka/kMnOXRi8BXnRaUwezLNhqelnN49xKz4F/dPP8OYLxLxq6JDtZb2i9XznUQbNPTg== + +has-bigints@^1.0.1, has-bigints@^1.0.2: + version "1.0.2" + resolved "https://registry.yarnpkg.com/has-bigints/-/has-bigints-1.0.2.tgz#0871bd3e3d51626f6ca0966668ba35d5602d6eaa" + integrity sha512-tSvCKtBr9lkF0Ex0aQiP9N+OpV4zi2r/Nee5VkRDbaqv35RLYMzbwQfFSZZH0kR+Rd6302UJZ2p/bJCEoR3VoQ== + +has-flag@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/has-flag/-/has-flag-3.0.0.tgz#b5d454dc2199ae225699f3467e5a07f3b955bafd" + integrity sha512-sKJf1+ceQBr4SMkvQnBDNDtf4TXpVhVGateu0t918bl30FnbE2m4vNLX+VWe/dpjlb+HugGYzW7uQXH98HPEYw== + +has-flag@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/has-flag/-/has-flag-4.0.0.tgz#944771fd9c81c81265c4d6941860da06bb59479b" + integrity sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ== + +has-property-descriptors@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/has-property-descriptors/-/has-property-descriptors-1.0.0.tgz#610708600606d36961ed04c196193b6a607fa861" + integrity sha512-62DVLZGoiEBDHQyqG4w9xCuZ7eJEwNmJRWw2VY84Oedb7WFcA27fiEVe8oUQx9hAUJ4ekurquucTGwsyO1XGdQ== + dependencies: + get-intrinsic "^1.1.1" + +has-proto@^1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/has-proto/-/has-proto-1.0.1.tgz#1885c1305538958aff469fef37937c22795408e0" + integrity sha512-7qE+iP+O+bgF9clE5+UoBFzE65mlBiVj3tKCrlNQ0Ogwm0BjpT/gK4SlLYDMybDh5I3TCTKnPPa0oMG7JDYrhg== + +has-symbols@^1.0.2, has-symbols@^1.0.3: + version "1.0.3" + resolved "https://registry.yarnpkg.com/has-symbols/-/has-symbols-1.0.3.tgz#bb7b2c4349251dce87b125f7bdf874aa7c8b39f8" + integrity sha512-l3LCuF6MgDNwTDKkdYGEihYjt5pRPbEg46rtlmnSPlUbgmB8LOIrKJbYYFBSbnPaJexMKtiPO8hmeRjRz2Td+A== + +has-tostringtag@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/has-tostringtag/-/has-tostringtag-1.0.0.tgz#7e133818a7d394734f941e73c3d3f9291e658b25" + integrity sha512-kFjcSNhnlGV1kyoGk7OXKSawH5JOb/LzUc5w9B02hOTO0dfFRjbHQKvg1d6cf3HbeUmtU9VbbV3qzZ2Teh97WQ== + dependencies: + has-symbols "^1.0.2" + +has@^1.0.3: + version "1.0.3" + resolved "https://registry.yarnpkg.com/has/-/has-1.0.3.tgz#722d7cbfc1f6aa8241f16dd814e011e1f41e8796" + integrity sha512-f2dvO0VU6Oej7RkWJGrehjbzMAjFp5/VKPp5tTpWIV4JHHZK1/BxbFRtf/siA2SWTe09caDmVtYYzWEIbBS4zw== + dependencies: + function-bind "^1.1.1" + +he@^1.2.0: + version "1.2.0" + resolved "https://registry.yarnpkg.com/he/-/he-1.2.0.tgz#84ae65fa7eafb165fddb61566ae14baf05664f0f" + integrity sha512-F/1DnUGPopORZi0ni+CvrCgHQ5FyEAHRLSApuYWMmrbSwoN2Mn/7k+Gl38gJnR7yyDZk6WLXwiGod1JOWNDKGw== + +hpack.js@^2.1.6: + version "2.1.6" + resolved "https://registry.yarnpkg.com/hpack.js/-/hpack.js-2.1.6.tgz#87774c0949e513f42e84575b3c45681fade2a0b2" + integrity sha512-zJxVehUdMGIKsRaNt7apO2Gqp0BdqW5yaiGHXXmbpvxgBYVZnAql+BJb4RO5ad2MgpbZKn5G6nMnegrH1FcNYQ== + dependencies: + inherits "^2.0.1" + obuf "^1.0.0" + readable-stream "^2.0.1" + wbuf "^1.1.0" + +html-entities@^2.3.2: + version "2.4.0" + resolved "https://registry.yarnpkg.com/html-entities/-/html-entities-2.4.0.tgz#edd0cee70402584c8c76cc2c0556db09d1f45061" + integrity sha512-igBTJcNNNhvZFRtm8uA6xMY6xYleeDwn3PeBCkDz7tHttv4F2hsDI2aPgNERWzvRcNYHNT3ymRaQzllmXj4YsQ== + +html-minifier-terser@^6.0.2: + version "6.1.0" + resolved "https://registry.yarnpkg.com/html-minifier-terser/-/html-minifier-terser-6.1.0.tgz#bfc818934cc07918f6b3669f5774ecdfd48f32ab" + integrity sha512-YXxSlJBZTP7RS3tWnQw74ooKa6L9b9i9QYXY21eUEvhZ3u9XLfv6OnFsQq6RxkhHygsaUMvYsZRV5rU/OVNZxw== + dependencies: + camel-case "^4.1.2" + clean-css "^5.2.2" + commander "^8.3.0" + he "^1.2.0" + param-case "^3.0.4" + relateurl "^0.2.7" + terser "^5.10.0" + +html-webpack-plugin@^5.5.3: + version "5.5.3" + resolved "https://registry.yarnpkg.com/html-webpack-plugin/-/html-webpack-plugin-5.5.3.tgz#72270f4a78e222b5825b296e5e3e1328ad525a3e" + integrity sha512-6YrDKTuqaP/TquFH7h4srYWsZx+x6k6+FbsTm0ziCwGHDP78Unr1r9F/H4+sGmMbX08GQcJ+K64x55b+7VM/jg== + dependencies: + "@types/html-minifier-terser" "^6.0.0" + html-minifier-terser "^6.0.2" + lodash "^4.17.21" + pretty-error "^4.0.0" + tapable "^2.0.0" + +htmlparser2@^6.1.0: + version "6.1.0" + resolved "https://registry.yarnpkg.com/htmlparser2/-/htmlparser2-6.1.0.tgz#c4d762b6c3371a05dbe65e94ae43a9f845fb8fb7" + integrity sha512-gyyPk6rgonLFEDGoeRgQNaEUvdJ4ktTmmUh/h2t7s+M8oPpIPxgNACWa+6ESR57kXstwqPiCut0V8NRpcwgU7A== + dependencies: + domelementtype "^2.0.1" + domhandler "^4.0.0" + domutils "^2.5.2" + entities "^2.0.0" + +http-deceiver@^1.2.7: + version "1.2.7" + resolved "https://registry.yarnpkg.com/http-deceiver/-/http-deceiver-1.2.7.tgz#fa7168944ab9a519d337cb0bec7284dc3e723d87" + integrity sha512-LmpOGxTfbpgtGVxJrj5k7asXHCgNZp5nLfp+hWc8QQRqtb7fUy6kRY3BO1h9ddF6yIPYUARgxGOwB42DnxIaNw== + +http-errors@2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/http-errors/-/http-errors-2.0.0.tgz#b7774a1486ef73cf7667ac9ae0858c012c57b9d3" + integrity sha512-FtwrG/euBzaEjYeRqOgly7G0qviiXoJWnvEH2Z1plBdXgbyjv34pHTSb9zoeHMyDy33+DWy5Wt9Wo+TURtOYSQ== + dependencies: + depd "2.0.0" + inherits "2.0.4" + setprototypeof "1.2.0" + statuses "2.0.1" + toidentifier "1.0.1" + +http-errors@~1.6.2: + version "1.6.3" + resolved "https://registry.yarnpkg.com/http-errors/-/http-errors-1.6.3.tgz#8b55680bb4be283a0b5bf4ea2e38580be1d9320d" + integrity sha512-lks+lVC8dgGyh97jxvxeYTWQFvh4uw4yC12gVl63Cg30sjPX4wuGcdkICVXDAESr6OJGjqGA8Iz5mkeN6zlD7A== + dependencies: + depd "~1.1.2" + inherits "2.0.3" + setprototypeof "1.1.0" + statuses ">= 1.4.0 < 2" + +http-parser-js@>=0.5.1: + version "0.5.8" + resolved "https://registry.yarnpkg.com/http-parser-js/-/http-parser-js-0.5.8.tgz#af23090d9ac4e24573de6f6aecc9d84a48bf20e3" + integrity sha512-SGeBX54F94Wgu5RH3X5jsDtf4eHyRogWX1XGT3b4HuW3tQPM4AaBzoUji/4AAJNXCEOWZ5O0DgZmJw1947gD5Q== + +http-proxy-middleware@^2.0.3: + version "2.0.6" + resolved "https://registry.yarnpkg.com/http-proxy-middleware/-/http-proxy-middleware-2.0.6.tgz#e1a4dd6979572c7ab5a4e4b55095d1f32a74963f" + integrity sha512-ya/UeJ6HVBYxrgYotAZo1KvPWlgB48kUJLDePFeneHsVujFaW5WNj2NgWCAE//B1Dl02BIfYlpNgBy8Kf8Rjmw== + dependencies: + "@types/http-proxy" "^1.17.8" + http-proxy "^1.18.1" + is-glob "^4.0.1" + is-plain-obj "^3.0.0" + micromatch "^4.0.2" + +http-proxy@^1.18.1: + version "1.18.1" + resolved "https://registry.yarnpkg.com/http-proxy/-/http-proxy-1.18.1.tgz#401541f0534884bbf95260334e72f88ee3976549" + integrity sha512-7mz/721AbnJwIVbnaSv1Cz3Am0ZLT/UBwkC92VlxhXv/k/BBQfM2fXElQNC27BVGr0uwUpplYPQM9LnaBMR5NQ== + dependencies: + eventemitter3 "^4.0.0" + follow-redirects "^1.0.0" + requires-port "^1.0.0" + +human-signals@^2.1.0: + version "2.1.0" + resolved "https://registry.yarnpkg.com/human-signals/-/human-signals-2.1.0.tgz#dc91fcba42e4d06e4abaed33b3e7a3c02f514ea0" + integrity sha512-B4FFZ6q/T2jhhksgkbEW3HBvWIfDW85snkQgawt07S7J5QXTk6BkNV+0yAeZrM5QpMAdYlocGoljn0sJ/WQkFw== + +human-signals@^4.3.0: + version "4.3.1" + resolved "https://registry.yarnpkg.com/human-signals/-/human-signals-4.3.1.tgz#ab7f811e851fca97ffbd2c1fe9a958964de321b2" + integrity sha512-nZXjEF2nbo7lIw3mgYjItAfgQXog3OjJogSbKa2CQIIvSGWcKgeJnQlNXip6NglNzYH45nSRiEVimMvYL8DDqQ== + +husky@^8.0.0: + version "8.0.3" + resolved "https://registry.yarnpkg.com/husky/-/husky-8.0.3.tgz#4936d7212e46d1dea28fef29bb3a108872cd9184" + integrity sha512-+dQSyqPh4x1hlO1swXBiNb2HzTDN1I2IGLQx1GrBuiqFJfoMrnZWwVmatvSiO+Iz8fBUnf+lekwNo4c2LlXItg== + +iconv-lite@0.4.24: + version "0.4.24" + resolved "https://registry.yarnpkg.com/iconv-lite/-/iconv-lite-0.4.24.tgz#2022b4b25fbddc21d2f524974a474aafe733908b" + integrity sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA== + dependencies: + safer-buffer ">= 2.1.2 < 3" + +icss-utils@^5.0.0, icss-utils@^5.1.0: + version "5.1.0" + resolved "https://registry.yarnpkg.com/icss-utils/-/icss-utils-5.1.0.tgz#c6be6858abd013d768e98366ae47e25d5887b1ae" + integrity sha512-soFhflCVWLfRNOPU3iv5Z9VUdT44xFRbzjLsEzSr5AQmgqPMTHdU3PMT1Cf1ssx8fLNJDA1juftYl+PUcv3MqA== + +ignore@^5.2.0: + version "5.2.4" + resolved "https://registry.yarnpkg.com/ignore/-/ignore-5.2.4.tgz#a291c0c6178ff1b960befe47fcdec301674a6324" + integrity sha512-MAb38BcSbH0eHNBxn7ql2NH/kX33OkB3lZ1BNdh7ENeRChHTYsTvWrMubiIAMNS2llXEEgZ1MUOBtXChP3kaFQ== + +import-fresh@^3.0.0, import-fresh@^3.2.1: + version "3.3.0" + resolved "https://registry.yarnpkg.com/import-fresh/-/import-fresh-3.3.0.tgz#37162c25fcb9ebaa2e6e53d5b4d88ce17d9e0c2b" + integrity sha512-veYYhQa+D1QBKznvhUHxb8faxlrwUnxseDAbAp457E0wLNio2bOSKnjYDhMj+YiAq61xrMGhQk9iXVk5FzgQMw== + dependencies: + parent-module "^1.0.0" + resolve-from "^4.0.0" + +import-local@^3.0.2: + version "3.1.0" + resolved "https://registry.yarnpkg.com/import-local/-/import-local-3.1.0.tgz#b4479df8a5fd44f6cdce24070675676063c95cb4" + integrity sha512-ASB07uLtnDs1o6EHjKpX34BKYDSqnFerfTOJL2HvMqF70LnxpjkzDB8J44oT9pu4AMPkQwf8jl6szgvNd2tRIg== + dependencies: + pkg-dir "^4.2.0" + resolve-cwd "^3.0.0" + +imurmurhash@^0.1.4: + version "0.1.4" + resolved "https://registry.yarnpkg.com/imurmurhash/-/imurmurhash-0.1.4.tgz#9218b9b2b928a238b13dc4fb6b6d576f231453ea" + integrity sha512-JmXMZ6wuvDmLiHEml9ykzqO6lwFbof0GG4IkcGaENdCRDDmMVnny7s5HsIgHCbaq0w2MyPhDqkhTUgS2LU2PHA== + +indent-string@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/indent-string/-/indent-string-4.0.0.tgz#624f8f4497d619b2d9768531d58f4122854d7251" + integrity sha512-EdDDZu4A2OyIK7Lr/2zG+w5jmbuk1DVBnEwREQvBzspBJkCEbRa8GxU1lghYcaGJCnRWibjDXlq779X1/y5xwg== + +inflight@^1.0.4: + version "1.0.6" + resolved "https://registry.yarnpkg.com/inflight/-/inflight-1.0.6.tgz#49bd6331d7d02d0c09bc910a1075ba8165b56df9" + integrity sha512-k92I/b08q4wvFscXCLvqfsHCrjrF7yiXsQuIVvVE7N82W3+aqpzuUdBbfhWcy/FZR3/4IgflMgKLOsvPDrGCJA== + dependencies: + once "^1.3.0" + wrappy "1" + +inherits@2, inherits@2.0.4, inherits@^2.0.1, inherits@^2.0.3, inherits@~2.0.3: + version "2.0.4" + resolved "https://registry.yarnpkg.com/inherits/-/inherits-2.0.4.tgz#0fa2c64f932917c3433a0ded55363aae37416b7c" + integrity sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ== + +inherits@2.0.3: + version "2.0.3" + resolved "https://registry.yarnpkg.com/inherits/-/inherits-2.0.3.tgz#633c2c83e3da42a502f52466022480f4208261de" + integrity sha512-x00IRNXNy63jwGkJmzPigoySHbaqpNuzKbBOmzK+g2OdZpQ9w+sxCN+VSB3ja7IAge2OP2qpfxTjeNcyjmW1uw== + +internal-slot@^1.0.5: + version "1.0.5" + resolved "https://registry.yarnpkg.com/internal-slot/-/internal-slot-1.0.5.tgz#f2a2ee21f668f8627a4667f309dc0f4fb6674986" + integrity sha512-Y+R5hJrzs52QCG2laLn4udYVnxsfny9CpOhNhUvk/SSSVyF6T27FzRbF0sroPidSu3X8oEAkOn2K804mjpt6UQ== + dependencies: + get-intrinsic "^1.2.0" + has "^1.0.3" + side-channel "^1.0.4" + +interpret@^3.1.1: + version "3.1.1" + resolved "https://registry.yarnpkg.com/interpret/-/interpret-3.1.1.tgz#5be0ceed67ca79c6c4bc5cf0d7ee843dcea110c4" + integrity sha512-6xwYfHbajpoF0xLW+iwLkhwgvLoZDfjYfoFNu8ftMoXINzwuymNLd9u/KmwtdT2GbR+/Cz66otEGEVVUHX9QLQ== + +ipaddr.js@1.9.1: + version "1.9.1" + resolved "https://registry.yarnpkg.com/ipaddr.js/-/ipaddr.js-1.9.1.tgz#bff38543eeb8984825079ff3a2a8e6cbd46781b3" + integrity sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g== + +ipaddr.js@^2.0.1: + version "2.1.0" + resolved "https://registry.yarnpkg.com/ipaddr.js/-/ipaddr.js-2.1.0.tgz#2119bc447ff8c257753b196fc5f1ce08a4cdf39f" + integrity sha512-LlbxQ7xKzfBusov6UMi4MFpEg0m+mAm9xyNGEduwXMEDuf4WfzB/RZwMVYEd7IKGvh4IUkEXYxtAVu9T3OelJQ== + +is-array-buffer@^3.0.1, is-array-buffer@^3.0.2: + version "3.0.2" + resolved "https://registry.yarnpkg.com/is-array-buffer/-/is-array-buffer-3.0.2.tgz#f2653ced8412081638ecb0ebbd0c41c6e0aecbbe" + integrity sha512-y+FyyR/w8vfIRq4eQcM1EYgSTnmHXPqaF+IgzgraytCFq5Xh8lllDVmAZolPJiZttZLeFSINPYMaEJ7/vWUa1w== + dependencies: + call-bind "^1.0.2" + get-intrinsic "^1.2.0" + is-typed-array "^1.1.10" + +is-bigint@^1.0.1: + version "1.0.4" + resolved "https://registry.yarnpkg.com/is-bigint/-/is-bigint-1.0.4.tgz#08147a1875bc2b32005d41ccd8291dffc6691df3" + integrity sha512-zB9CruMamjym81i2JZ3UMn54PKGsQzsJeo6xvN3HJJ4CAsQNB6iRutp2To77OfCNuoxspsIhzaPoO1zyCEhFOg== + dependencies: + has-bigints "^1.0.1" + +is-binary-path@~2.1.0: + version "2.1.0" + resolved "https://registry.yarnpkg.com/is-binary-path/-/is-binary-path-2.1.0.tgz#ea1f7f3b80f064236e83470f86c09c254fb45b09" + integrity sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw== + dependencies: + binary-extensions "^2.0.0" + +is-boolean-object@^1.1.0: + version "1.1.2" + resolved "https://registry.yarnpkg.com/is-boolean-object/-/is-boolean-object-1.1.2.tgz#5c6dc200246dd9321ae4b885a114bb1f75f63719" + integrity sha512-gDYaKHJmnj4aWxyj6YHyXVpdQawtVLHU5cb+eztPGczf6cjuTdwve5ZIEfgXqH4e57An1D1AKf8CZ3kYrQRqYA== + dependencies: + call-bind "^1.0.2" + has-tostringtag "^1.0.0" + +is-callable@^1.1.3, is-callable@^1.1.4, is-callable@^1.2.7: + version "1.2.7" + resolved "https://registry.yarnpkg.com/is-callable/-/is-callable-1.2.7.tgz#3bc2a85ea742d9e36205dcacdd72ca1fdc51b055" + integrity sha512-1BC0BVFhS/p0qtw6enp8e+8OD0UrK0oFLztSjNzhcKA3WDuJxxAPXzPuPtKkjEY9UUoEWlX/8fgKeu2S8i9JTA== + +is-core-module@^2.11.0: + version "2.12.1" + resolved "https://registry.yarnpkg.com/is-core-module/-/is-core-module-2.12.1.tgz#0c0b6885b6f80011c71541ce15c8d66cf5a4f9fd" + integrity sha512-Q4ZuBAe2FUsKtyQJoQHlvP8OvBERxO3jEmy1I7hcRXcJBGGHFh/aJBswbXuS9sgrDH2QUO8ilkwNPHvHMd8clg== + dependencies: + has "^1.0.3" + +is-date-object@^1.0.1: + version "1.0.5" + resolved "https://registry.yarnpkg.com/is-date-object/-/is-date-object-1.0.5.tgz#0841d5536e724c25597bf6ea62e1bd38298df31f" + integrity sha512-9YQaSxsAiSwcvS33MBk3wTCVnWK+HhF8VZR2jRxehM16QcVOdHqPn4VPHmRK4lSr38n9JriurInLcP90xsYNfQ== + dependencies: + has-tostringtag "^1.0.0" + +is-docker@^2.0.0, is-docker@^2.1.1: + version "2.2.1" + resolved "https://registry.yarnpkg.com/is-docker/-/is-docker-2.2.1.tgz#33eeabe23cfe86f14bde4408a02c0cfb853acdaa" + integrity sha512-F+i2BKsFrH66iaUFc0woD8sLy8getkwTwtOBjvs56Cx4CgJDeKQeqfz8wAYiSb8JOprWhHH5p77PbmYCvvUuXQ== + +is-extglob@^2.1.1: + version "2.1.1" + resolved "https://registry.yarnpkg.com/is-extglob/-/is-extglob-2.1.1.tgz#a88c02535791f02ed37c76a1b9ea9773c833f8c2" + integrity sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ== + +is-fullwidth-code-point@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz#f116f8064fe90b3f7844a38997c0b75051269f1d" + integrity sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg== + +is-fullwidth-code-point@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/is-fullwidth-code-point/-/is-fullwidth-code-point-4.0.0.tgz#fae3167c729e7463f8461ce512b080a49268aa88" + integrity sha512-O4L094N2/dZ7xqVdrXhh9r1KODPJpFms8B5sGdJLPy664AgvXsreZUyCQQNItZRDlYug4xStLjNp/sz3HvBowQ== + +is-glob@^4.0.0, is-glob@^4.0.1, is-glob@^4.0.3, is-glob@~4.0.1: + version "4.0.3" + resolved "https://registry.yarnpkg.com/is-glob/-/is-glob-4.0.3.tgz#64f61e42cbbb2eec2071a9dac0b28ba1e65d5084" + integrity sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg== + dependencies: + is-extglob "^2.1.1" + +is-negative-zero@^2.0.2: + version "2.0.2" + resolved "https://registry.yarnpkg.com/is-negative-zero/-/is-negative-zero-2.0.2.tgz#7bf6f03a28003b8b3965de3ac26f664d765f3150" + integrity sha512-dqJvarLawXsFbNDeJW7zAz8ItJ9cd28YufuuFzh0G8pNHjJMnY08Dv7sYX2uF5UpQOwieAeOExEYAWWfu7ZZUA== + +is-number-object@^1.0.4: + version "1.0.7" + resolved "https://registry.yarnpkg.com/is-number-object/-/is-number-object-1.0.7.tgz#59d50ada4c45251784e9904f5246c742f07a42fc" + integrity sha512-k1U0IRzLMo7ZlYIfzRu23Oh6MiIFasgpb9X76eqfFZAqwH44UI4KTBvBYIZ1dSL9ZzChTB9ShHfLkR4pdW5krQ== + dependencies: + has-tostringtag "^1.0.0" + +is-number@^7.0.0: + version "7.0.0" + resolved "https://registry.yarnpkg.com/is-number/-/is-number-7.0.0.tgz#7535345b896734d5f80c4d06c50955527a14f12b" + integrity sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng== + +is-path-inside@^3.0.3: + version "3.0.3" + resolved "https://registry.yarnpkg.com/is-path-inside/-/is-path-inside-3.0.3.tgz#d231362e53a07ff2b0e0ea7fed049161ffd16283" + integrity sha512-Fd4gABb+ycGAmKou8eMftCupSir5lRxqf4aD/vd0cD2qc4HL07OjCeuHMr8Ro4CoMaeCKDB0/ECBOVWjTwUvPQ== + +is-plain-obj@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/is-plain-obj/-/is-plain-obj-3.0.0.tgz#af6f2ea14ac5a646183a5bbdb5baabbc156ad9d7" + integrity sha512-gwsOE28k+23GP1B6vFl1oVh/WOzmawBrKwo5Ev6wMKzPkaXaCDIQKzLnvsA42DRlbVTWorkgTKIviAKCWkfUwA== + +is-plain-object@^2.0.4: + version "2.0.4" + resolved "https://registry.yarnpkg.com/is-plain-object/-/is-plain-object-2.0.4.tgz#2c163b3fafb1b606d9d17928f05c2a1c38e07677" + integrity sha512-h5PpgXkWitc38BBMYawTYMWJHFZJVnBquFE57xFpjB8pJFiF6gZ+bU+WyI/yqXiFR5mdLsgYNaPe8uao6Uv9Og== + dependencies: + isobject "^3.0.1" + +is-regex@^1.1.4: + version "1.1.4" + resolved "https://registry.yarnpkg.com/is-regex/-/is-regex-1.1.4.tgz#eef5663cd59fa4c0ae339505323df6854bb15958" + integrity sha512-kvRdxDsxZjhzUX07ZnLydzS1TU/TJlTUHHY4YLL87e37oUA49DfkLqgy+VjFocowy29cKvcSiu+kIv728jTTVg== + dependencies: + call-bind "^1.0.2" + has-tostringtag "^1.0.0" + +is-shared-array-buffer@^1.0.2: + version "1.0.2" + resolved "https://registry.yarnpkg.com/is-shared-array-buffer/-/is-shared-array-buffer-1.0.2.tgz#8f259c573b60b6a32d4058a1a07430c0a7344c79" + integrity sha512-sqN2UDu1/0y6uvXyStCOzyhAjCSlHceFoMKJW8W9EU9cvic/QdsZ0kEU93HEy3IUEFZIiH/3w+AH/UQbPHNdhA== + dependencies: + call-bind "^1.0.2" + +is-stream@^2.0.0: + version "2.0.1" + resolved "https://registry.yarnpkg.com/is-stream/-/is-stream-2.0.1.tgz#fac1e3d53b97ad5a9d0ae9cef2389f5810a5c077" + integrity sha512-hFoiJiTl63nn+kstHGBtewWSKnQLpyb155KHheA1l39uvtO9nWIop1p3udqPcUd/xbF1VLMO4n7OI6p7RbngDg== + +is-stream@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/is-stream/-/is-stream-3.0.0.tgz#e6bfd7aa6bef69f4f472ce9bb681e3e57b4319ac" + integrity sha512-LnQR4bZ9IADDRSkvpqMGvt/tEJWclzklNgSw48V5EAaAeDd6qGvN8ei6k5p0tvxSR171VmGyHuTiAOfxAbr8kA== + +is-string@^1.0.5, is-string@^1.0.7: + version "1.0.7" + resolved "https://registry.yarnpkg.com/is-string/-/is-string-1.0.7.tgz#0dd12bf2006f255bb58f695110eff7491eebc0fd" + integrity sha512-tE2UXzivje6ofPW7l23cjDOMa09gb7xlAqG6jG5ej6uPV32TlWP3NKPigtaGeHNu9fohccRYvIiZMfOOnOYUtg== + dependencies: + has-tostringtag "^1.0.0" + +is-symbol@^1.0.2, is-symbol@^1.0.3: + version "1.0.4" + resolved "https://registry.yarnpkg.com/is-symbol/-/is-symbol-1.0.4.tgz#a6dac93b635b063ca6872236de88910a57af139c" + integrity sha512-C/CPBqKWnvdcxqIARxyOh4v1UUEOCHpgDa0WYgpKDFMszcrPcffg5uhwSgPCLD2WWxmq6isisz87tzT01tuGhg== + dependencies: + has-symbols "^1.0.2" + +is-typed-array@^1.1.10, is-typed-array@^1.1.9: + version "1.1.10" + resolved "https://registry.yarnpkg.com/is-typed-array/-/is-typed-array-1.1.10.tgz#36a5b5cb4189b575d1a3e4b08536bfb485801e3f" + integrity sha512-PJqgEHiWZvMpaFZ3uTc8kHPM4+4ADTlDniuQL7cU/UDA0Ql7F70yGfHph3cLNe+c9toaigv+DFzTJKhc2CtO6A== + dependencies: + available-typed-arrays "^1.0.5" + call-bind "^1.0.2" + for-each "^0.3.3" + gopd "^1.0.1" + has-tostringtag "^1.0.0" + +is-weakref@^1.0.2: + version "1.0.2" + resolved "https://registry.yarnpkg.com/is-weakref/-/is-weakref-1.0.2.tgz#9529f383a9338205e89765e0392efc2f100f06f2" + integrity sha512-qctsuLZmIQ0+vSSMfoVvyFe2+GSEvnmZ2ezTup1SBse9+twCCeial6EEi3Nc2KFcf6+qz2FBPnjXsk8xhKSaPQ== + dependencies: + call-bind "^1.0.2" + +is-wsl@^2.2.0: + version "2.2.0" + resolved "https://registry.yarnpkg.com/is-wsl/-/is-wsl-2.2.0.tgz#74a4c76e77ca9fd3f932f290c17ea326cd157271" + integrity sha512-fKzAra0rGJUUBwGBgNkHZuToZcn+TtXHpeCgmkMJMMYx1sQDYaCSyjJBSCa2nH1DGm7s3n1oBnohoVTBaN7Lww== + dependencies: + is-docker "^2.0.0" + +isarray@~1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/isarray/-/isarray-1.0.0.tgz#bb935d48582cba168c06834957a54a3e07124f11" + integrity sha512-VLghIWNM6ELQzo7zwmcg0NmTVyWKYjvIeM83yjp0wRDTmUnrM678fQbcKBo6n2CJEF0szoG//ytg+TKla89ALQ== + +isexe@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/isexe/-/isexe-2.0.0.tgz#e8fbf374dc556ff8947a10dcb0572d633f2cfa10" + integrity sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw== + +isobject@^3.0.1: + version "3.0.1" + resolved "https://registry.yarnpkg.com/isobject/-/isobject-3.0.1.tgz#4e431e92b11a9731636aa1f9c8d1ccbcfdab78df" + integrity sha512-WhB9zCku7EGTj/HQQRz5aUQEUeoQZH2bWcltRErOpymJ4boYE6wL9Tbr23krRPSZ+C5zqNSrSw+Cc7sZZ4b7vg== + +jest-worker@^27.4.5: + version "27.5.1" + resolved "https://registry.yarnpkg.com/jest-worker/-/jest-worker-27.5.1.tgz#8d146f0900e8973b106b6f73cc1e9a8cb86f8db0" + integrity sha512-7vuh85V5cdDofPyxn58nrPjBktZo0u9x1g8WtjQol+jZDaE+fhN+cIvTj11GndBnMnyfrUOG1sZQxCdjKh+DKg== + dependencies: + "@types/node" "*" + merge-stream "^2.0.0" + supports-color "^8.0.0" + +js-tokens@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/js-tokens/-/js-tokens-4.0.0.tgz#19203fb59991df98e3a287050d4647cdeaf32499" + integrity sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ== + +js-yaml@^4.1.0: + version "4.1.0" + resolved "https://registry.yarnpkg.com/js-yaml/-/js-yaml-4.1.0.tgz#c1fb65f8f5017901cdd2c951864ba18458a10602" + integrity sha512-wpxZs9NoxZaJESJGIZTyDEaYpl0FKSA+FB9aJiyemKhMwkxQg63h4T1KJgUGHpTqPDNRcmmYLugrRjJlBtWvRA== + dependencies: + argparse "^2.0.1" + +jsesc@^2.5.1: + version "2.5.2" + resolved "https://registry.yarnpkg.com/jsesc/-/jsesc-2.5.2.tgz#80564d2e483dacf6e8ef209650a67df3f0c283a4" + integrity sha512-OYu7XEzjkCQ3C5Ps3QIZsQfNpqoJyZZA99wd9aWd05NCtC5pWOkShK2mkL6HXQR6/Cy2lbNdPlZBpuQHXE63gA== + +jsesc@~0.5.0: + version "0.5.0" + resolved "https://registry.yarnpkg.com/jsesc/-/jsesc-0.5.0.tgz#e7dee66e35d6fc16f710fe91d5cf69f70f08911d" + integrity sha512-uZz5UnB7u4T9LvwmFqXii7pZSouaRPorGs5who1Ip7VO0wxanFvBL7GkM6dTHlgX+jhBApRetaWpnDabOeTcnA== + +json-parse-even-better-errors@^2.3.1: + version "2.3.1" + resolved "https://registry.yarnpkg.com/json-parse-even-better-errors/-/json-parse-even-better-errors-2.3.1.tgz#7c47805a94319928e05777405dc12e1f7a4ee02d" + integrity sha512-xyFwyhro/JEof6Ghe2iz2NcXoj2sloNsWr/XsERDK/oiPCfaNhl5ONfp+jQdAZRQQ0IJWNzH9zIZF7li91kh2w== + +json-schema-traverse@^0.4.1: + version "0.4.1" + resolved "https://registry.yarnpkg.com/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz#69f6a87d9513ab8bb8fe63bdb0979c448e684660" + integrity sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg== + +json-schema-traverse@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/json-schema-traverse/-/json-schema-traverse-1.0.0.tgz#ae7bcb3656ab77a73ba5c49bf654f38e6b6860e2" + integrity sha512-NM8/P9n3XjXhIZn1lLhkFaACTOURQXjWhV4BA/RnOv8xvgqtqpAX9IO4mRQxSx1Rlo4tqzeqb0sOlruaOy3dug== + +json-stable-stringify-without-jsonify@^1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/json-stable-stringify-without-jsonify/-/json-stable-stringify-without-jsonify-1.0.1.tgz#9db7b59496ad3f3cfef30a75142d2d930ad72651" + integrity sha512-Bdboy+l7tA3OGW6FjyFHWkP5LuByj1Tk33Ljyq0axyzdk9//JSi2u3fP1QSmd1KNwq6VOKYGlAu87CisVir6Pw== + +json5@^1.0.2: + version "1.0.2" + resolved "https://registry.yarnpkg.com/json5/-/json5-1.0.2.tgz#63d98d60f21b313b77c4d6da18bfa69d80e1d593" + integrity sha512-g1MWMLBiz8FKi1e4w0UyVL3w+iJceWAFBAaBnnGKOpNa5f8TLktkbre1+s6oICydWAm+HRUGTmI+//xv2hvXYA== + dependencies: + minimist "^1.2.0" + +json5@^2.1.1, json5@^2.2.2: + version "2.2.3" + resolved "https://registry.yarnpkg.com/json5/-/json5-2.2.3.tgz#78cd6f1a19bdc12b73db5ad0c61efd66c1e29283" + integrity sha512-XmOWe7eyHYH14cLdVPoyg+GOH3rYX++KpzrylJwSW98t3Nk+U8XOl8FWKOgwtzdb8lXGf6zYwDUzeHMWfxasyg== + +kind-of@^6.0.2: + version "6.0.3" + resolved "https://registry.yarnpkg.com/kind-of/-/kind-of-6.0.3.tgz#07c05034a6c349fa06e24fa35aa76db4580ce4dd" + integrity sha512-dcS1ul+9tmeD95T+x28/ehLgd9mENa3LsvDTtzm3vyBEO7RPptvAD+t44WVXaUjTBRcrpFeFlC8WCruUR456hw== + +launch-editor@^2.6.0: + version "2.6.0" + resolved "https://registry.yarnpkg.com/launch-editor/-/launch-editor-2.6.0.tgz#4c0c1a6ac126c572bd9ff9a30da1d2cae66defd7" + integrity sha512-JpDCcQnyAAzZZaZ7vEiSqL690w7dAEyLao+KC96zBplnYbJS7TYNjvM3M7y3dGz+v7aIsJk3hllWuc0kWAjyRQ== + dependencies: + picocolors "^1.0.0" + shell-quote "^1.7.3" + +levn@^0.4.1: + version "0.4.1" + resolved "https://registry.yarnpkg.com/levn/-/levn-0.4.1.tgz#ae4562c007473b932a6200d403268dd2fffc6ade" + integrity sha512-+bT2uH4E5LGE7h/n3evcS/sQlJXCpIp6ym8OWJ5eV6+67Dsql/LaaT7qJBAt2rzfoa/5QBGBhxDix1dMt2kQKQ== + dependencies: + prelude-ls "^1.2.1" + type-check "~0.4.0" + +lilconfig@2.1.0: + version "2.1.0" + resolved "https://registry.yarnpkg.com/lilconfig/-/lilconfig-2.1.0.tgz#78e23ac89ebb7e1bfbf25b18043de756548e7f52" + integrity sha512-utWOt/GHzuUxnLKxB6dk81RoOeoNeHgbrXiuGk4yyF5qlRz+iIVWu56E2fqGHFrXz0QNUhLB/8nKqvRH66JKGQ== + +lint-staged@^13.2.3: + version "13.2.3" + resolved "https://registry.yarnpkg.com/lint-staged/-/lint-staged-13.2.3.tgz#f899aad6c093473467e9c9e316e3c2d8a28f87a7" + integrity sha512-zVVEXLuQIhr1Y7R7YAWx4TZLdvuzk7DnmrsTNL0fax6Z3jrpFcas+vKbzxhhvp6TA55m1SQuWkpzI1qbfDZbAg== + dependencies: + chalk "5.2.0" + cli-truncate "^3.1.0" + commander "^10.0.0" + debug "^4.3.4" + execa "^7.0.0" + lilconfig "2.1.0" + listr2 "^5.0.7" + micromatch "^4.0.5" + normalize-path "^3.0.0" + object-inspect "^1.12.3" + pidtree "^0.6.0" + string-argv "^0.3.1" + yaml "^2.2.2" + +listr2@^5.0.7: + version "5.0.8" + resolved "https://registry.yarnpkg.com/listr2/-/listr2-5.0.8.tgz#a9379ffeb4bd83a68931a65fb223a11510d6ba23" + integrity sha512-mC73LitKHj9w6v30nLNGPetZIlfpUniNSsxxrbaPcWOjDb92SHPzJPi/t+v1YC/lxKz/AJ9egOjww0qUuFxBpA== + dependencies: + cli-truncate "^2.1.0" + colorette "^2.0.19" + log-update "^4.0.0" + p-map "^4.0.0" + rfdc "^1.3.0" + rxjs "^7.8.0" + through "^2.3.8" + wrap-ansi "^7.0.0" + +loader-runner@^4.2.0: + version "4.3.0" + resolved "https://registry.yarnpkg.com/loader-runner/-/loader-runner-4.3.0.tgz#c1b4a163b99f614830353b16755e7149ac2314e1" + integrity sha512-3R/1M+yS3j5ou80Me59j7F9IMs4PXs3VqRrm0TU3AbKPxlmpoY1TNscJV/oGJXo8qCatFGTfDbY6W6ipGOYXfg== + +locate-path@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/locate-path/-/locate-path-3.0.0.tgz#dbec3b3ab759758071b58fe59fc41871af21400e" + integrity sha512-7AO748wWnIhNqAuaty2ZWHkQHRSNfPVIsPIfwEOWO22AmaoVrWavlOcMR5nzTLNYvp36X220/maaRsrec1G65A== + dependencies: + p-locate "^3.0.0" + path-exists "^3.0.0" + +locate-path@^5.0.0: + version "5.0.0" + resolved "https://registry.yarnpkg.com/locate-path/-/locate-path-5.0.0.tgz#1afba396afd676a6d42504d0a67a3a7eb9f62aa0" + integrity sha512-t7hw9pI+WvuwNJXwk5zVHpyhIqzg2qTlklJOf0mVxGSbe3Fp2VieZcduNYjaLDoy6p9uGpQEGWG87WpMKlNq8g== + dependencies: + p-locate "^4.1.0" + +locate-path@^6.0.0: + version "6.0.0" + resolved "https://registry.yarnpkg.com/locate-path/-/locate-path-6.0.0.tgz#55321eb309febbc59c4801d931a72452a681d286" + integrity sha512-iPZK6eYjbxRu3uB4/WZ3EsEIMJFMqAoopl3R+zuq0UjcAm/MO6KCweDgPfP3elTztoKP3KtnVHxTn2NHBSDVUw== + dependencies: + p-locate "^5.0.0" + +lodash.debounce@^4.0.8: + version "4.0.8" + resolved "https://registry.yarnpkg.com/lodash.debounce/-/lodash.debounce-4.0.8.tgz#82d79bff30a67c4005ffd5e2515300ad9ca4d7af" + integrity sha512-FT1yDzDYEoYWhnSGnpE/4Kj1fLZkDFyqRb7fNt6FdYOSxlUWAtp42Eh6Wb0rGIv/m9Bgo7x4GhQbm5Ys4SG5ow== + +lodash.merge@^4.6.2: + version "4.6.2" + resolved "https://registry.yarnpkg.com/lodash.merge/-/lodash.merge-4.6.2.tgz#558aa53b43b661e1925a0afdfa36a9a1085fe57a" + integrity sha512-0KpjqXRVvrYyCsX1swR/XTK0va6VQkQM6MNo7PqW77ByjAhoARA8EfrP1N4+KlKj8YS0ZUCtRT/YUuhyYDujIQ== + +lodash@^4.17.20, lodash@^4.17.21: + version "4.17.21" + resolved "https://registry.yarnpkg.com/lodash/-/lodash-4.17.21.tgz#679591c564c3bffaae8454cf0b3df370c3d6911c" + integrity sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg== + +log-update@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/log-update/-/log-update-4.0.0.tgz#589ecd352471f2a1c0c570287543a64dfd20e0a1" + integrity sha512-9fkkDevMefjg0mmzWFBW8YkFP91OrizzkW3diF7CpG+S2EYdy4+TVfGwz1zeF8x7hCx1ovSPTOE9Ngib74qqUg== + dependencies: + ansi-escapes "^4.3.0" + cli-cursor "^3.1.0" + slice-ansi "^4.0.0" + wrap-ansi "^6.2.0" + +lower-case@^2.0.2: + version "2.0.2" + resolved "https://registry.yarnpkg.com/lower-case/-/lower-case-2.0.2.tgz#6fa237c63dbdc4a82ca0fd882e4722dc5e634e28" + integrity sha512-7fm3l3NAF9WfN6W3JOmf5drwpVqX78JtoGJ3A6W0a6ZnldM41w2fV5D490psKFTpMds8TJse/eHLFFsNHHjHgg== + dependencies: + tslib "^2.0.3" + +lru-cache@^5.1.1: + version "5.1.1" + resolved "https://registry.yarnpkg.com/lru-cache/-/lru-cache-5.1.1.tgz#1da27e6710271947695daf6848e847f01d84b920" + integrity sha512-KpNARQA3Iwv+jTA0utUVVbrh+Jlrr1Fv0e56GGzAFOXN7dk/FviaDW8LHmK52DlcH4WP2n6gI8vN1aesBFgo9w== + dependencies: + yallist "^3.0.2" + +lru-cache@^6.0.0: + version "6.0.0" + resolved "https://registry.yarnpkg.com/lru-cache/-/lru-cache-6.0.0.tgz#6d6fe6570ebd96aaf90fcad1dafa3b2566db3a94" + integrity sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA== + dependencies: + yallist "^4.0.0" + +make-dir@^3.0.2: + version "3.1.0" + resolved "https://registry.yarnpkg.com/make-dir/-/make-dir-3.1.0.tgz#415e967046b3a7f1d185277d84aa58203726a13f" + integrity sha512-g3FeP20LNwhALb/6Cz6Dd4F2ngze0jz7tbzrD2wAV+o9FeNHe4rL+yK2md0J/fiSf1sa1ADhXqi5+oVwOM/eGw== + dependencies: + semver "^6.0.0" + +media-typer@0.3.0: + version "0.3.0" + resolved "https://registry.yarnpkg.com/media-typer/-/media-typer-0.3.0.tgz#8710d7af0aa626f8fffa1ce00168545263255748" + integrity sha512-dq+qelQ9akHpcOl/gUVRTxVIOkAJ1wR3QAvb4RsVjS8oVoFjDGTc679wJYmUmknUF5HwMLOgb5O+a3KxfWapPQ== + +memfs@^3.4.3: + version "3.6.0" + resolved "https://registry.yarnpkg.com/memfs/-/memfs-3.6.0.tgz#d7a2110f86f79dd950a8b6df6d57bc984aa185f6" + integrity sha512-EGowvkkgbMcIChjMTMkESFDbZeSh8xZ7kNSF0hAiAN4Jh6jgHCRS0Ga/+C8y6Au+oqpezRHCfPsmJ2+DwAgiwQ== + dependencies: + fs-monkey "^1.0.4" + +merge-descriptors@1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/merge-descriptors/-/merge-descriptors-1.0.1.tgz#b00aaa556dd8b44568150ec9d1b953f3f90cbb61" + integrity sha512-cCi6g3/Zr1iqQi6ySbseM1Xvooa98N0w31jzUYrXPX2xqObmFGHJ0tQ5u74H3mVh7wLouTseZyYIq39g8cNp1w== + +merge-stream@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/merge-stream/-/merge-stream-2.0.0.tgz#52823629a14dd00c9770fb6ad47dc6310f2c1f60" + integrity sha512-abv/qOcuPfk3URPfDzmZU1LKmuw8kT+0nIHvKrKgFrwifol/doWcdA4ZqsWQ8ENrFKkd67Mfpo/LovbIUsbt3w== + +methods@~1.1.2: + version "1.1.2" + resolved "https://registry.yarnpkg.com/methods/-/methods-1.1.2.tgz#5529a4d67654134edcc5266656835b0f851afcee" + integrity sha512-iclAHeNqNm68zFtnZ0e+1L2yUIdvzNoauKU4WBA3VvH/vPFieF7qfRlwUZU+DA9P9bPXIS90ulxoUoCH23sV2w== + +micromatch@^4.0.2, micromatch@^4.0.5: + version "4.0.5" + resolved "https://registry.yarnpkg.com/micromatch/-/micromatch-4.0.5.tgz#bc8999a7cbbf77cdc89f132f6e467051b49090c6" + integrity sha512-DMy+ERcEW2q8Z2Po+WNXuw3c5YaUSFjAO5GsJqfEl7UjvtIuFKO6ZrKvcItdy98dwFI2N1tg3zNIdKaQT+aNdA== + dependencies: + braces "^3.0.2" + picomatch "^2.3.1" + +mime-db@1.52.0, "mime-db@>= 1.43.0 < 2": + version "1.52.0" + resolved "https://registry.yarnpkg.com/mime-db/-/mime-db-1.52.0.tgz#bbabcdc02859f4987301c856e3387ce5ec43bf70" + integrity sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg== + +mime-types@^2.1.27, mime-types@^2.1.31, mime-types@~2.1.17, mime-types@~2.1.24, mime-types@~2.1.34: + version "2.1.35" + resolved "https://registry.yarnpkg.com/mime-types/-/mime-types-2.1.35.tgz#381a871b62a734450660ae3deee44813f70d959a" + integrity sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw== + dependencies: + mime-db "1.52.0" + +mime@1.6.0: + version "1.6.0" + resolved "https://registry.yarnpkg.com/mime/-/mime-1.6.0.tgz#32cd9e5c64553bd58d19a568af452acff04981b1" + integrity sha512-x0Vn8spI+wuJ1O6S7gnbaQg8Pxh4NNHb7KSINmEWKiPE4RKOplvijn+NkmYmmRgP68mc70j2EbeTFRsrswaQeg== + +mimic-fn@^2.1.0: + version "2.1.0" + resolved "https://registry.yarnpkg.com/mimic-fn/-/mimic-fn-2.1.0.tgz#7ed2c2ccccaf84d3ffcb7a69b57711fc2083401b" + integrity sha512-OqbOk5oEQeAZ8WXWydlu9HJjz9WVdEIvamMCcXmuqUYjTknH/sqsWvhQ3vgwKFRR1HpjvNBKQ37nbJgYzGqGcg== + +mimic-fn@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/mimic-fn/-/mimic-fn-4.0.0.tgz#60a90550d5cb0b239cca65d893b1a53b29871ecc" + integrity sha512-vqiC06CuhBTUdZH+RYl8sFrL096vA45Ok5ISO6sE/Mr1jRbGH4Csnhi8f3wKVl7x8mO4Au7Ir9D3Oyv1VYMFJw== + +minimalistic-assert@^1.0.0: + version "1.0.1" + resolved "https://registry.yarnpkg.com/minimalistic-assert/-/minimalistic-assert-1.0.1.tgz#2e194de044626d4a10e7f7fbc00ce73e83e4d5c7" + integrity sha512-UtJcAD4yEaGtjPezWuO9wC4nwUnVH/8/Im3yEHQP4b67cXlD/Qr9hdITCU1xDbSEXg2XKNaP8jsReV7vQd00/A== + +minimatch@^3.0.5, minimatch@^3.1.1, minimatch@^3.1.2: + version "3.1.2" + resolved "https://registry.yarnpkg.com/minimatch/-/minimatch-3.1.2.tgz#19cd194bfd3e428f049a70817c038d89ab4be35b" + integrity sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw== + dependencies: + brace-expansion "^1.1.7" + +minimatch@^5.0.1: + version "5.1.6" + resolved "https://registry.yarnpkg.com/minimatch/-/minimatch-5.1.6.tgz#1cfcb8cf5522ea69952cd2af95ae09477f122a96" + integrity sha512-lKwV/1brpG6mBUFHtb7NUmtABCb2WZZmm2wNiOA5hAb8VdCS4B3dtMWyvcoViccwAW/COERjXLt0zP1zXUN26g== + dependencies: + brace-expansion "^2.0.1" + +minimist@^1.2.0, minimist@^1.2.6: + version "1.2.8" + resolved "https://registry.yarnpkg.com/minimist/-/minimist-1.2.8.tgz#c1a464e7693302e082a075cee0c057741ac4772c" + integrity sha512-2yyAR8qBkN3YuheJanUpWC5U3bb5osDywNB8RzDVlDwDHbocAJveqqj1u8+SVD7jkWT4yvsHCpWqqWqAxb0zCA== + +ms@2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/ms/-/ms-2.0.0.tgz#5608aeadfc00be6c2901df5f9861788de0d597c8" + integrity sha512-Tpp60P6IUJDTuOq/5Z8cdskzJujfwqfOTkrwIwj7IRISpnkJnT6SyJ4PCPnGMoFjC9ddhal5KVIYtAt97ix05A== + +ms@2.1.2: + version "2.1.2" + resolved "https://registry.yarnpkg.com/ms/-/ms-2.1.2.tgz#d09d1f357b443f493382a8eb3ccd183872ae6009" + integrity sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w== + +ms@2.1.3, ms@^2.1.1: + version "2.1.3" + resolved "https://registry.yarnpkg.com/ms/-/ms-2.1.3.tgz#574c8138ce1d2b5861f0b44579dbadd60c6615b2" + integrity sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA== + +multicast-dns@^7.2.5: + version "7.2.5" + resolved "https://registry.yarnpkg.com/multicast-dns/-/multicast-dns-7.2.5.tgz#77eb46057f4d7adbd16d9290fa7299f6fa64cced" + integrity sha512-2eznPJP8z2BFLX50tf0LuODrpINqP1RVIm/CObbTcBRITQgmC/TjcREF1NeTBzIcR5XO/ukWo+YHOjBbFwIupg== + dependencies: + dns-packet "^5.2.2" + thunky "^1.0.2" + +nanoid@^3.3.6: + version "3.3.6" + resolved "https://registry.yarnpkg.com/nanoid/-/nanoid-3.3.6.tgz#443380c856d6e9f9824267d960b4236ad583ea4c" + integrity sha512-BGcqMMJuToF7i1rt+2PWSNVnWIkGCU78jBG3RxO/bZlnZPK2Cmi2QaffxGO/2RvWi9sL+FAiRiXMgsyxQ1DIDA== + +natural-compare@^1.4.0: + version "1.4.0" + resolved "https://registry.yarnpkg.com/natural-compare/-/natural-compare-1.4.0.tgz#4abebfeed7541f2c27acfb29bdbbd15c8d5ba4f7" + integrity sha512-OWND8ei3VtNC9h7V60qff3SVobHr996CTwgxubgyQYEpg290h9J0buyECNNJexkFm5sOajh5G116RYA1c8ZMSw== + +negotiator@0.6.3: + version "0.6.3" + resolved "https://registry.yarnpkg.com/negotiator/-/negotiator-0.6.3.tgz#58e323a72fedc0d6f9cd4d31fe49f51479590ccd" + integrity sha512-+EUsqGPLsM+j/zdChZjsnX51g4XrHFOIXwfnCVPGlQk/k5giakcKsuxCObBRu6DSm9opw/O6slWbJdghQM4bBg== + +neo-async@^2.6.2: + version "2.6.2" + resolved "https://registry.yarnpkg.com/neo-async/-/neo-async-2.6.2.tgz#b4aafb93e3aeb2d8174ca53cf163ab7d7308305f" + integrity sha512-Yd3UES5mWCSqR+qNT93S3UoYUkqAZ9lLg8a7g9rimsWmYGK8cVToA4/sF3RrshdyV3sAGMXVUmpMYOw+dLpOuw== + +no-case@^3.0.4: + version "3.0.4" + resolved "https://registry.yarnpkg.com/no-case/-/no-case-3.0.4.tgz#d361fd5c9800f558551a8369fc0dcd4662b6124d" + integrity sha512-fgAN3jGAh+RoxUGZHTSOLJIqUc2wmoBwGR4tbpNAKmmovFoWq0OdRkb0VkldReO2a2iBT/OEulG9XSUc10r3zg== + dependencies: + lower-case "^2.0.2" + tslib "^2.0.3" + +node-forge@^1: + version "1.3.1" + resolved "https://registry.yarnpkg.com/node-forge/-/node-forge-1.3.1.tgz#be8da2af243b2417d5f646a770663a92b7e9ded3" + integrity sha512-dPEtOeMvF9VMcYV/1Wb8CPoVAXtp6MKMlcbAt4ddqmGqUJ6fQZFXkNZNkNlfevtNkGtaSoXf/vNNNSvgrdXwtA== + +node-releases@^2.0.12: + version "2.0.12" + resolved "https://registry.yarnpkg.com/node-releases/-/node-releases-2.0.12.tgz#35627cc224a23bfb06fb3380f2b3afaaa7eb1039" + integrity sha512-QzsYKWhXTWx8h1kIvqfnC++o0pEmpRQA/aenALsL2F4pqNVr7YzcdMlDij5WBnwftRbJCNJL/O7zdKaxKPHqgQ== + +normalize-path@^3.0.0, normalize-path@~3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/normalize-path/-/normalize-path-3.0.0.tgz#0dcd69ff23a1c9b11fd0978316644a0388216a65" + integrity sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA== + +npm-run-path@^4.0.1: + version "4.0.1" + resolved "https://registry.yarnpkg.com/npm-run-path/-/npm-run-path-4.0.1.tgz#b7ecd1e5ed53da8e37a55e1c2269e0b97ed748ea" + integrity sha512-S48WzZW777zhNIrn7gxOlISNAqi9ZC/uQFnRdbeIHhZhCA6UqpkOT8T1G7BvfdgP4Er8gF4sUbaS0i7QvIfCWw== + dependencies: + path-key "^3.0.0" + +npm-run-path@^5.1.0: + version "5.1.0" + resolved "https://registry.yarnpkg.com/npm-run-path/-/npm-run-path-5.1.0.tgz#bc62f7f3f6952d9894bd08944ba011a6ee7b7e00" + integrity sha512-sJOdmRGrY2sjNTRMbSvluQqg+8X7ZK61yvzBEIDhz4f8z1TZFYABsqjjCBd/0PUNE9M6QDgHJXQkGUEm7Q+l9Q== + dependencies: + path-key "^4.0.0" + +nth-check@^2.0.1: + version "2.1.1" + resolved "https://registry.yarnpkg.com/nth-check/-/nth-check-2.1.1.tgz#c9eab428effce36cd6b92c924bdb000ef1f1ed1d" + integrity sha512-lqjrjmaOoAnWfMmBPL+XNnynZh2+swxiX3WUE0s4yEHI6m+AwrK2UZOimIRl3X/4QctVqS8AiZjFqyOGrMXb/w== + dependencies: + boolbase "^1.0.0" + +object-inspect@^1.12.3, object-inspect@^1.9.0: + version "1.12.3" + resolved "https://registry.yarnpkg.com/object-inspect/-/object-inspect-1.12.3.tgz#ba62dffd67ee256c8c086dfae69e016cd1f198b9" + integrity sha512-geUvdk7c+eizMNUDkRpW1wJwgfOiOeHbxBR/hLXK1aT6zmVSO0jsQcs7fj6MGw89jC/cjGfLcNOrtMYtGqm81g== + +object-keys@^1.1.1: + version "1.1.1" + resolved "https://registry.yarnpkg.com/object-keys/-/object-keys-1.1.1.tgz#1c47f272df277f3b1daf061677d9c82e2322c60e" + integrity sha512-NuAESUOUMrlIXOfHKzD6bpPu3tYt3xvjNdRIQ+FeT0lNb4K8WR70CaDxhuNguS2XG+GjkyMwOzsN5ZktImfhLA== + +object.assign@^4.1.2, object.assign@^4.1.4: + version "4.1.4" + resolved "https://registry.yarnpkg.com/object.assign/-/object.assign-4.1.4.tgz#9673c7c7c351ab8c4d0b516f4343ebf4dfb7799f" + integrity sha512-1mxKf0e58bvyjSCtKYY4sRe9itRk3PJpquJOjeIkz885CczcI4IvJJDLPS72oowuSh+pBxUFROpX+TU++hxhZQ== + dependencies: + call-bind "^1.0.2" + define-properties "^1.1.4" + has-symbols "^1.0.3" + object-keys "^1.1.1" + +object.entries@^1.1.5: + version "1.1.6" + resolved "https://registry.yarnpkg.com/object.entries/-/object.entries-1.1.6.tgz#9737d0e5b8291edd340a3e3264bb8a3b00d5fa23" + integrity sha512-leTPzo4Zvg3pmbQ3rDK69Rl8GQvIqMWubrkxONG9/ojtFE2rD9fjMKfSI5BxW3osRH1m6VdzmqK8oAY9aT4x5w== + dependencies: + call-bind "^1.0.2" + define-properties "^1.1.4" + es-abstract "^1.20.4" + +object.values@^1.1.6: + version "1.1.6" + resolved "https://registry.yarnpkg.com/object.values/-/object.values-1.1.6.tgz#4abbaa71eba47d63589d402856f908243eea9b1d" + integrity sha512-FVVTkD1vENCsAcwNs9k6jea2uHC/X0+JcjG8YA60FN5CMaJmG95wT9jek/xX9nornqGRrBkKtzuAu2wuHpKqvw== + dependencies: + call-bind "^1.0.2" + define-properties "^1.1.4" + es-abstract "^1.20.4" + +obuf@^1.0.0, obuf@^1.1.2: + version "1.1.2" + resolved "https://registry.yarnpkg.com/obuf/-/obuf-1.1.2.tgz#09bea3343d41859ebd446292d11c9d4db619084e" + integrity sha512-PX1wu0AmAdPqOL1mWhqmlOd8kOIZQwGZw6rh7uby9fTc5lhaOWFLX3I6R1hrF9k3zUY40e6igsLGkDXK92LJNg== + +on-finished@2.4.1: + version "2.4.1" + resolved "https://registry.yarnpkg.com/on-finished/-/on-finished-2.4.1.tgz#58c8c44116e54845ad57f14ab10b03533184ac3f" + integrity sha512-oVlzkg3ENAhCk2zdv7IJwd/QUD4z2RxRwpkcGY8psCVcCYZNq4wYnVWALHM+brtuJjePWiYF/ClmuDr8Ch5+kg== + dependencies: + ee-first "1.1.1" + +on-headers@~1.0.2: + version "1.0.2" + resolved "https://registry.yarnpkg.com/on-headers/-/on-headers-1.0.2.tgz#772b0ae6aaa525c399e489adfad90c403eb3c28f" + integrity sha512-pZAE+FJLoyITytdqK0U5s+FIpjN0JP3OzFi/u8Rx+EV5/W+JTWGXG8xFzevE7AjBfDqHv/8vL8qQsIhHnqRkrA== + +once@^1.3.0: + version "1.4.0" + resolved "https://registry.yarnpkg.com/once/-/once-1.4.0.tgz#583b1aa775961d4b113ac17d9c50baef9dd76bd1" + integrity sha512-lNaJgI+2Q5URQBkccEKHTQOPaXdUxnZZElQTZY0MFUAuaEqe1E+Nyvgdz/aIyNi6Z9MzO5dv1H8n58/GELp3+w== + dependencies: + wrappy "1" + +onetime@^5.1.0, onetime@^5.1.2: + version "5.1.2" + resolved "https://registry.yarnpkg.com/onetime/-/onetime-5.1.2.tgz#d0e96ebb56b07476df1dd9c4806e5237985ca45e" + integrity sha512-kbpaSSGJTWdAY5KPVeMOKXSrPtr8C8C7wodJbcsd51jRnmD+GZu8Y0VoU6Dm5Z4vWr0Ig/1NKuWRKf7j5aaYSg== + dependencies: + mimic-fn "^2.1.0" + +onetime@^6.0.0: + version "6.0.0" + resolved "https://registry.yarnpkg.com/onetime/-/onetime-6.0.0.tgz#7c24c18ed1fd2e9bca4bd26806a33613c77d34b4" + integrity sha512-1FlR+gjXK7X+AsAHso35MnyN5KqGwJRi/31ft6x0M194ht7S+rWAvd7PHss9xSKMzE0asv1pyIHaJYq+BbacAQ== + dependencies: + mimic-fn "^4.0.0" + +open@^8.0.9: + version "8.4.2" + resolved "https://registry.yarnpkg.com/open/-/open-8.4.2.tgz#5b5ffe2a8f793dcd2aad73e550cb87b59cb084f9" + integrity sha512-7x81NCL719oNbsq/3mh+hVrAWmFuEYUqrq/Iw3kUzH8ReypT9QQ0BLoJS7/G9k6N81XjW4qHWtjWwe/9eLy1EQ== + dependencies: + define-lazy-prop "^2.0.0" + is-docker "^2.1.1" + is-wsl "^2.2.0" + +optionator@^0.9.1: + version "0.9.3" + resolved "https://registry.yarnpkg.com/optionator/-/optionator-0.9.3.tgz#007397d44ed1872fdc6ed31360190f81814e2c64" + integrity sha512-JjCoypp+jKn1ttEFExxhetCKeJt9zhAgAve5FXHixTvFDW/5aEktX9bufBKLRRMdU7bNtpLfcGu94B3cdEJgjg== + dependencies: + "@aashutoshrathi/word-wrap" "^1.2.3" + deep-is "^0.1.3" + fast-levenshtein "^2.0.6" + levn "^0.4.1" + prelude-ls "^1.2.1" + type-check "^0.4.0" + +p-limit@^2.0.0, p-limit@^2.2.0: + version "2.3.0" + resolved "https://registry.yarnpkg.com/p-limit/-/p-limit-2.3.0.tgz#3dd33c647a214fdfffd835933eb086da0dc21db1" + integrity sha512-//88mFWSJx8lxCzwdAABTJL2MyWB12+eIY7MDL2SqLmAkeKU9qxRvWuSyTjm3FUmpBEMuFfckAIqEaVGUDxb6w== + dependencies: + p-try "^2.0.0" + +p-limit@^3.0.2: + version "3.1.0" + resolved "https://registry.yarnpkg.com/p-limit/-/p-limit-3.1.0.tgz#e1daccbe78d0d1388ca18c64fea38e3e57e3706b" + integrity sha512-TYOanM3wGwNGsZN2cVTYPArw454xnXj5qmWF1bEoAc4+cU/ol7GVh7odevjp1FNHduHc3KZMcFduxU5Xc6uJRQ== + dependencies: + yocto-queue "^0.1.0" + +p-locate@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/p-locate/-/p-locate-3.0.0.tgz#322d69a05c0264b25997d9f40cd8a891ab0064a4" + integrity sha512-x+12w/To+4GFfgJhBEpiDcLozRJGegY+Ei7/z0tSLkMmxGZNybVMSfWj9aJn8Z5Fc7dBUNJOOVgPv2H7IwulSQ== + dependencies: + p-limit "^2.0.0" + +p-locate@^4.1.0: + version "4.1.0" + resolved "https://registry.yarnpkg.com/p-locate/-/p-locate-4.1.0.tgz#a3428bb7088b3a60292f66919278b7c297ad4f07" + integrity sha512-R79ZZ/0wAxKGu3oYMlz8jy/kbhsNrS7SKZ7PxEHBgJ5+F2mtFW2fK2cOtBh1cHYkQsbzFV7I+EoRKe6Yt0oK7A== + dependencies: + p-limit "^2.2.0" + +p-locate@^5.0.0: + version "5.0.0" + resolved "https://registry.yarnpkg.com/p-locate/-/p-locate-5.0.0.tgz#83c8315c6785005e3bd021839411c9e110e6d834" + integrity sha512-LaNjtRWUBY++zB5nE/NwcaoMylSPk+S+ZHNB1TzdbMJMny6dynpAGt7X/tl/QYq3TIeE6nxHppbo2LGymrG5Pw== + dependencies: + p-limit "^3.0.2" + +p-map@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/p-map/-/p-map-4.0.0.tgz#bb2f95a5eda2ec168ec9274e06a747c3e2904d2b" + integrity sha512-/bjOqmgETBYB5BoEeGVea8dmvHb2m9GLy1E9W43yeyfP6QQCZGFNa+XRceJEuDB6zqr+gKpIAmlLebMpykw/MQ== + dependencies: + aggregate-error "^3.0.0" + +p-retry@^4.5.0: + version "4.6.2" + resolved "https://registry.yarnpkg.com/p-retry/-/p-retry-4.6.2.tgz#9baae7184057edd4e17231cee04264106e092a16" + integrity sha512-312Id396EbJdvRONlngUx0NydfrIQ5lsYu0znKVUzVvArzEIt08V1qhtyESbGVd1FGX7UKtiFp5uwKZdM8wIuQ== + dependencies: + "@types/retry" "0.12.0" + retry "^0.13.1" + +p-try@^2.0.0: + version "2.2.0" + resolved "https://registry.yarnpkg.com/p-try/-/p-try-2.2.0.tgz#cb2868540e313d61de58fafbe35ce9004d5540e6" + integrity sha512-R4nPAVTAU0B9D35/Gk3uJf/7XYbQcyohSKdvAxIRSNghFl4e71hVoGnBNQz9cWaXxO2I10KTC+3jMdvvoKw6dQ== + +param-case@^3.0.4: + version "3.0.4" + resolved "https://registry.yarnpkg.com/param-case/-/param-case-3.0.4.tgz#7d17fe4aa12bde34d4a77d91acfb6219caad01c5" + integrity sha512-RXlj7zCYokReqWpOPH9oYivUzLYZ5vAPIfEmCTNViosC78F8F0H9y7T7gG2M39ymgutxF5gcFEsyZQSph9Bp3A== + dependencies: + dot-case "^3.0.4" + tslib "^2.0.3" + +parent-module@^1.0.0: + version "1.0.1" + resolved "https://registry.yarnpkg.com/parent-module/-/parent-module-1.0.1.tgz#691d2709e78c79fae3a156622452d00762caaaa2" + integrity sha512-GQ2EWRpQV8/o+Aw8YqtfZZPfNRWZYkbidE9k5rpl/hC3vtHHBfGm2Ifi6qWV+coDGkrUKZAxE3Lot5kcsRlh+g== + dependencies: + callsites "^3.0.0" + +parseurl@~1.3.2, parseurl@~1.3.3: + version "1.3.3" + resolved "https://registry.yarnpkg.com/parseurl/-/parseurl-1.3.3.tgz#9da19e7bee8d12dff0513ed5b76957793bc2e8d4" + integrity sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ== + +pascal-case@^3.1.2: + version "3.1.2" + resolved "https://registry.yarnpkg.com/pascal-case/-/pascal-case-3.1.2.tgz#b48e0ef2b98e205e7c1dae747d0b1508237660eb" + integrity sha512-uWlGT3YSnK9x3BQJaOdcZwrnV6hPpd8jFH1/ucpiLRPh/2zCVJKS19E4GvYHvaCcACn3foXZ0cLB9Wrx1KGe5g== + dependencies: + no-case "^3.0.4" + tslib "^2.0.3" + +path-exists@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/path-exists/-/path-exists-3.0.0.tgz#ce0ebeaa5f78cb18925ea7d810d7b59b010fd515" + integrity sha512-bpC7GYwiDYQ4wYLe+FA8lhRjhQCMcQGuSgGGqDkg/QerRWw9CmGRT0iSOVRSZJ29NMLZgIzqaljJ63oaL4NIJQ== + +path-exists@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/path-exists/-/path-exists-4.0.0.tgz#513bdbe2d3b95d7762e8c1137efa195c6c61b5b3" + integrity sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w== + +path-is-absolute@^1.0.0: + version "1.0.1" + resolved "https://registry.yarnpkg.com/path-is-absolute/-/path-is-absolute-1.0.1.tgz#174b9268735534ffbc7ace6bf53a5a9e1b5c5f5f" + integrity sha512-AVbw3UJ2e9bq64vSaS9Am0fje1Pa8pbGqTTsmXfaIiMpnr5DlDhfJOuLj9Sf95ZPVDAUerDfEk88MPmPe7UCQg== + +path-key@^3.0.0, path-key@^3.1.0: + version "3.1.1" + resolved "https://registry.yarnpkg.com/path-key/-/path-key-3.1.1.tgz#581f6ade658cbba65a0d3380de7753295054f375" + integrity sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q== + +path-key@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/path-key/-/path-key-4.0.0.tgz#295588dc3aee64154f877adb9d780b81c554bf18" + integrity sha512-haREypq7xkM7ErfgIyA0z+Bj4AGKlMSdlQE2jvJo6huWD1EdkKYV+G/T4nq0YEF2vgTT8kqMFKo1uHn950r4SQ== + +path-parse@^1.0.7: + version "1.0.7" + resolved "https://registry.yarnpkg.com/path-parse/-/path-parse-1.0.7.tgz#fbc114b60ca42b30d9daf5858e4bd68bbedb6735" + integrity sha512-LDJzPVEEEPR+y48z93A0Ed0yXb8pAByGWo/k5YYdYgpY2/2EsOsksJrq7lOHxryrVOn1ejG6oAp8ahvOIQD8sw== + +path-to-regexp@0.1.7: + version "0.1.7" + resolved "https://registry.yarnpkg.com/path-to-regexp/-/path-to-regexp-0.1.7.tgz#df604178005f522f15eb4490e7247a1bfaa67f8c" + integrity sha512-5DFkuoqlv1uYQKxy8omFBeJPQcdoE07Kv2sferDCrAq1ohOU+MSDswDIbnx3YAM60qIOnYa53wBhXW0EbMonrQ== + +picocolors@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/picocolors/-/picocolors-1.0.0.tgz#cb5bdc74ff3f51892236eaf79d68bc44564ab81c" + integrity sha512-1fygroTLlHu66zi26VoTDv8yRgm0Fccecssto+MhsZ0D/DGW2sm8E8AjW7NU5VVTRt5GxbeZ5qBuJr+HyLYkjQ== + +picomatch@^2.0.4, picomatch@^2.2.1, picomatch@^2.3.1: + version "2.3.1" + resolved "https://registry.yarnpkg.com/picomatch/-/picomatch-2.3.1.tgz#3ba3833733646d9d3e4995946c1365a67fb07a42" + integrity sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA== + +pidtree@^0.6.0: + version "0.6.0" + resolved "https://registry.yarnpkg.com/pidtree/-/pidtree-0.6.0.tgz#90ad7b6d42d5841e69e0a2419ef38f8883aa057c" + integrity sha512-eG2dWTVw5bzqGRztnHExczNxt5VGsE6OwTeCG3fdUf9KBsZzO3R5OIIIzWR+iZA0NtZ+RDVdaoE2dK1cn6jH4g== + +pkg-dir@^4.1.0, pkg-dir@^4.2.0: + version "4.2.0" + resolved "https://registry.yarnpkg.com/pkg-dir/-/pkg-dir-4.2.0.tgz#f099133df7ede422e81d1d8448270eeb3e4261f3" + integrity sha512-HRDzbaKjC+AOWVXxAU/x54COGeIv9eb+6CkDSQoNTt4XyWoIJvuPsXizxu/Fr23EiekbtZwmh1IcIG/l/a10GQ== + dependencies: + find-up "^4.0.0" + +pkg-up@^3.1.0: + version "3.1.0" + resolved "https://registry.yarnpkg.com/pkg-up/-/pkg-up-3.1.0.tgz#100ec235cc150e4fd42519412596a28512a0def5" + integrity sha512-nDywThFk1i4BQK4twPQ6TA4RT8bDY96yeuCVBWL3ePARCiEKDRSrNGbFIgUJpLp+XeIR65v8ra7WuJOFUBtkMA== + dependencies: + find-up "^3.0.0" + +postcss-modules-extract-imports@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/postcss-modules-extract-imports/-/postcss-modules-extract-imports-3.0.0.tgz#cda1f047c0ae80c97dbe28c3e76a43b88025741d" + integrity sha512-bdHleFnP3kZ4NYDhuGlVK+CMrQ/pqUm8bx/oGL93K6gVwiclvX5x0n76fYMKuIGKzlABOy13zsvqjb0f92TEXw== + +postcss-modules-local-by-default@^4.0.3: + version "4.0.3" + resolved "https://registry.yarnpkg.com/postcss-modules-local-by-default/-/postcss-modules-local-by-default-4.0.3.tgz#b08eb4f083050708998ba2c6061b50c2870ca524" + integrity sha512-2/u2zraspoACtrbFRnTijMiQtb4GW4BvatjaG/bCjYQo8kLTdevCUlwuBHx2sCnSyrI3x3qj4ZK1j5LQBgzmwA== + dependencies: + icss-utils "^5.0.0" + postcss-selector-parser "^6.0.2" + postcss-value-parser "^4.1.0" + +postcss-modules-scope@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/postcss-modules-scope/-/postcss-modules-scope-3.0.0.tgz#9ef3151456d3bbfa120ca44898dfca6f2fa01f06" + integrity sha512-hncihwFA2yPath8oZ15PZqvWGkWf+XUfQgUGamS4LqoP1anQLOsOJw0vr7J7IwLpoY9fatA2qiGUGmuZL0Iqlg== + dependencies: + postcss-selector-parser "^6.0.4" + +postcss-modules-values@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/postcss-modules-values/-/postcss-modules-values-4.0.0.tgz#d7c5e7e68c3bb3c9b27cbf48ca0bb3ffb4602c9c" + integrity sha512-RDxHkAiEGI78gS2ofyvCsu7iycRv7oqw5xMWn9iMoR0N/7mf9D50ecQqUo5BZ9Zh2vH4bCUR/ktCqbB9m8vJjQ== + dependencies: + icss-utils "^5.0.0" + +postcss-selector-parser@^6.0.2, postcss-selector-parser@^6.0.4: + version "6.0.13" + resolved "https://registry.yarnpkg.com/postcss-selector-parser/-/postcss-selector-parser-6.0.13.tgz#d05d8d76b1e8e173257ef9d60b706a8e5e99bf1b" + integrity sha512-EaV1Gl4mUEV4ddhDnv/xtj7sxwrwxdetHdWUGnT4VJQf+4d05v6lHYZr8N573k5Z0BViss7BDhfWtKS3+sfAqQ== + dependencies: + cssesc "^3.0.0" + util-deprecate "^1.0.2" + +postcss-value-parser@^4.1.0, postcss-value-parser@^4.2.0: + version "4.2.0" + resolved "https://registry.yarnpkg.com/postcss-value-parser/-/postcss-value-parser-4.2.0.tgz#723c09920836ba6d3e5af019f92bc0971c02e514" + integrity sha512-1NNCs6uurfkVbeXG4S8JFT9t19m45ICnif8zWLd5oPSZ50QnwMfK+H3jv408d4jw/7Bttv5axS5IiHoLaVNHeQ== + +postcss@^8.4.21: + version "8.4.24" + resolved "https://registry.yarnpkg.com/postcss/-/postcss-8.4.24.tgz#f714dba9b2284be3cc07dbd2fc57ee4dc972d2df" + integrity sha512-M0RzbcI0sO/XJNucsGjvWU9ERWxb/ytp1w6dKtxTKgixdtQDq4rmx/g8W1hnaheq9jgwL/oyEdH5Bc4WwJKMqg== + dependencies: + nanoid "^3.3.6" + picocolors "^1.0.0" + source-map-js "^1.0.2" + +prelude-ls@^1.2.1: + version "1.2.1" + resolved "https://registry.yarnpkg.com/prelude-ls/-/prelude-ls-1.2.1.tgz#debc6489d7a6e6b0e7611888cec880337d316396" + integrity sha512-vkcDPrRZo1QZLbn5RLGPpg/WmIQ65qoWWhcGKf/b5eplkkarX0m9z8ppCat4mlOqUsWpyNuYgO3VRyrYHSzX5g== + +prettier@^2.8.8: + version "2.8.8" + resolved "https://registry.yarnpkg.com/prettier/-/prettier-2.8.8.tgz#e8c5d7e98a4305ffe3de2e1fc4aca1a71c28b1da" + integrity sha512-tdN8qQGvNjw4CHbY+XXk0JgCXn9QiF21a55rBe5LJAU+kDyC4WQn4+awm2Xfk2lQMk5fKup9XgzTZtGkjBdP9Q== + +pretty-error@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/pretty-error/-/pretty-error-4.0.0.tgz#90a703f46dd7234adb46d0f84823e9d1cb8f10d6" + integrity sha512-AoJ5YMAcXKYxKhuJGdcvse+Voc6v1RgnsR3nWcYU7q4t6z0Q6T86sv5Zq8VIRbOWWFpvdGE83LtdSMNd+6Y0xw== + dependencies: + lodash "^4.17.20" + renderkid "^3.0.0" + +process-nextick-args@~2.0.0: + version "2.0.1" + resolved "https://registry.yarnpkg.com/process-nextick-args/-/process-nextick-args-2.0.1.tgz#7820d9b16120cc55ca9ae7792680ae7dba6d7fe2" + integrity sha512-3ouUOpQhtgrbOa17J7+uxOTpITYWaGP7/AhoR3+A+/1e9skrzelGi/dXzEYyvbxubEF6Wn2ypscTKiKJFFn1ag== + +proxy-addr@~2.0.7: + version "2.0.7" + resolved "https://registry.yarnpkg.com/proxy-addr/-/proxy-addr-2.0.7.tgz#f19fe69ceab311eeb94b42e70e8c2070f9ba1025" + integrity sha512-llQsMLSUDUPT44jdrU/O37qlnifitDP+ZwrmmZcoSKyLKvtZxpyV0n2/bD/N4tBAAZ/gJEdZU7KMraoK1+XYAg== + dependencies: + forwarded "0.2.0" + ipaddr.js "1.9.1" + +punycode@^2.1.0: + version "2.3.0" + resolved "https://registry.yarnpkg.com/punycode/-/punycode-2.3.0.tgz#f67fa67c94da8f4d0cfff981aee4118064199b8f" + integrity sha512-rRV+zQD8tVFys26lAGR9WUuS4iUAngJScM+ZRSKtvl5tKeZ2t5bvdNFdNHBW9FWR4guGHlgmsZ1G7BSm2wTbuA== + +qs@6.11.0: + version "6.11.0" + resolved "https://registry.yarnpkg.com/qs/-/qs-6.11.0.tgz#fd0d963446f7a65e1367e01abd85429453f0c37a" + integrity sha512-MvjoMCJwEarSbUYk5O+nmoSzSutSsTwF85zcHPQ9OrlFoZOYIjaqBAJIqIXjptyD5vThxGq52Xu/MaJzRkIk4Q== + dependencies: + side-channel "^1.0.4" + +queue-microtask@^1.2.2: + version "1.2.3" + resolved "https://registry.yarnpkg.com/queue-microtask/-/queue-microtask-1.2.3.tgz#4929228bbc724dfac43e0efb058caf7b6cfb6243" + integrity sha512-NuaNSa6flKT5JaSYQzJok04JzTL1CA6aGhv5rfLW3PgqA+M2ChpZQnAC8h8i4ZFkBS8X5RqkDBHA7r4hej3K9A== + +randombytes@^2.1.0: + version "2.1.0" + resolved "https://registry.yarnpkg.com/randombytes/-/randombytes-2.1.0.tgz#df6f84372f0270dc65cdf6291349ab7a473d4f2a" + integrity sha512-vYl3iOX+4CKUWuxGi9Ukhie6fsqXqS9FE2Zaic4tNFD2N2QQaXOMFbuKK4QmDHC0JO6B1Zp41J0LpT0oR68amQ== + dependencies: + safe-buffer "^5.1.0" + +range-parser@^1.2.1, range-parser@~1.2.1: + version "1.2.1" + resolved "https://registry.yarnpkg.com/range-parser/-/range-parser-1.2.1.tgz#3cf37023d199e1c24d1a55b84800c2f3e6468031" + integrity sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg== + +raw-body@2.5.1: + version "2.5.1" + resolved "https://registry.yarnpkg.com/raw-body/-/raw-body-2.5.1.tgz#fe1b1628b181b700215e5fd42389f98b71392857" + integrity sha512-qqJBtEyVgS0ZmPGdCFPWJ3FreoqvG4MVQln/kCgF7Olq95IbOp0/BWyMwbdtn4VTvkM8Y7khCQ2Xgk/tcrCXig== + dependencies: + bytes "3.1.2" + http-errors "2.0.0" + iconv-lite "0.4.24" + unpipe "1.0.0" + +readable-stream@^2.0.1: + version "2.3.8" + resolved "https://registry.yarnpkg.com/readable-stream/-/readable-stream-2.3.8.tgz#91125e8042bba1b9887f49345f6277027ce8be9b" + integrity sha512-8p0AUk4XODgIewSi0l8Epjs+EVnWiK7NoDIEGU0HhE7+ZyY8D1IMY7odu5lRrFXGg71L15KG8QrPmum45RTtdA== + dependencies: + core-util-is "~1.0.0" + inherits "~2.0.3" + isarray "~1.0.0" + process-nextick-args "~2.0.0" + safe-buffer "~5.1.1" + string_decoder "~1.1.1" + util-deprecate "~1.0.1" + +readable-stream@^3.0.6: + version "3.6.2" + resolved "https://registry.yarnpkg.com/readable-stream/-/readable-stream-3.6.2.tgz#56a9b36ea965c00c5a93ef31eb111a0f11056967" + integrity sha512-9u/sniCrY3D5WdsERHzHE4G2YCXqoG5FTHUiCC4SIbr6XcLZBY05ya9EKjYek9O5xOAwjGq+1JdGBAS7Q9ScoA== + dependencies: + inherits "^2.0.3" + string_decoder "^1.1.1" + util-deprecate "^1.0.1" + +readdirp@~3.6.0: + version "3.6.0" + resolved "https://registry.yarnpkg.com/readdirp/-/readdirp-3.6.0.tgz#74a370bd857116e245b29cc97340cd431a02a6c7" + integrity sha512-hOS089on8RduqdbhvQ5Z37A0ESjsqz6qnRcffsMU3495FuTdqSm+7bhJ29JvIOsBDEEnan5DPu9t3To9VRlMzA== + dependencies: + picomatch "^2.2.1" + +rechoir@^0.8.0: + version "0.8.0" + resolved "https://registry.yarnpkg.com/rechoir/-/rechoir-0.8.0.tgz#49f866e0d32146142da3ad8f0eff352b3215ff22" + integrity sha512-/vxpCXddiX8NGfGO/mTafwjq4aFa/71pvamip0++IQk3zG8cbCj0fifNPrjjF1XMXUne91jL9OoxmdykoEtifQ== + dependencies: + resolve "^1.20.0" + +regenerate-unicode-properties@^10.1.0: + version "10.1.0" + resolved "https://registry.yarnpkg.com/regenerate-unicode-properties/-/regenerate-unicode-properties-10.1.0.tgz#7c3192cab6dd24e21cb4461e5ddd7dd24fa8374c" + integrity sha512-d1VudCLoIGitcU/hEg2QqvyGZQmdC0Lf8BqdOMXGFSvJP4bNV1+XqbPQeHHLD51Jh4QJJ225dlIFvY4Ly6MXmQ== + dependencies: + regenerate "^1.4.2" + +regenerate@^1.4.2: + version "1.4.2" + resolved "https://registry.yarnpkg.com/regenerate/-/regenerate-1.4.2.tgz#b9346d8827e8f5a32f7ba29637d398b69014848a" + integrity sha512-zrceR/XhGYU/d/opr2EKO7aRHUeiBI8qjtfHqADTwZd6Szfy16la6kqD0MIUs5z5hx6AaKa+PixpPrR289+I0A== + +regenerator-runtime@^0.13.11: + version "0.13.11" + resolved "https://registry.yarnpkg.com/regenerator-runtime/-/regenerator-runtime-0.13.11.tgz#f6dca3e7ceec20590d07ada785636a90cdca17f9" + integrity sha512-kY1AZVr2Ra+t+piVaJ4gxaFaReZVH40AKNo7UCX6W+dEwBo/2oZJzqfuN1qLq1oL45o56cPaTXELwrTh8Fpggg== + +regenerator-transform@^0.15.1: + version "0.15.1" + resolved "https://registry.yarnpkg.com/regenerator-transform/-/regenerator-transform-0.15.1.tgz#f6c4e99fc1b4591f780db2586328e4d9a9d8dc56" + integrity sha512-knzmNAcuyxV+gQCufkYcvOqX/qIIfHLv0u5x79kRxuGojfYVky1f15TzZEu2Avte8QGepvUNTnLskf8E6X6Vyg== + dependencies: + "@babel/runtime" "^7.8.4" + +regexp.prototype.flags@^1.4.3: + version "1.5.0" + resolved "https://registry.yarnpkg.com/regexp.prototype.flags/-/regexp.prototype.flags-1.5.0.tgz#fe7ce25e7e4cca8db37b6634c8a2c7009199b9cb" + integrity sha512-0SutC3pNudRKgquxGoRGIz946MZVHqbNfPjBdxeOhBrdgDKlRoXmYLQN9xRbrR09ZXWeGAdPuif7egofn6v5LA== + dependencies: + call-bind "^1.0.2" + define-properties "^1.2.0" + functions-have-names "^1.2.3" + +regexpu-core@^5.3.1: + version "5.3.2" + resolved "https://registry.yarnpkg.com/regexpu-core/-/regexpu-core-5.3.2.tgz#11a2b06884f3527aec3e93dbbf4a3b958a95546b" + integrity sha512-RAM5FlZz+Lhmo7db9L298p2vHP5ZywrVXmVXpmAD9GuL5MPH6t9ROw1iA/wfHkQ76Qe7AaPF0nGuim96/IrQMQ== + dependencies: + "@babel/regjsgen" "^0.8.0" + regenerate "^1.4.2" + regenerate-unicode-properties "^10.1.0" + regjsparser "^0.9.1" + unicode-match-property-ecmascript "^2.0.0" + unicode-match-property-value-ecmascript "^2.1.0" + +regjsparser@^0.9.1: + version "0.9.1" + resolved "https://registry.yarnpkg.com/regjsparser/-/regjsparser-0.9.1.tgz#272d05aa10c7c1f67095b1ff0addae8442fc5709" + integrity sha512-dQUtn90WanSNl+7mQKcXAgZxvUe7Z0SqXlgzv0za4LwiUhyzBC58yQO3liFoUgu8GiJVInAhJjkj1N0EtQ5nkQ== + dependencies: + jsesc "~0.5.0" + +relateurl@^0.2.7: + version "0.2.7" + resolved "https://registry.yarnpkg.com/relateurl/-/relateurl-0.2.7.tgz#54dbf377e51440aca90a4cd274600d3ff2d888a9" + integrity sha512-G08Dxvm4iDN3MLM0EsP62EDV9IuhXPR6blNz6Utcp7zyV3tr4HVNINt6MpaRWbxoOHT3Q7YN2P+jaHX8vUbgog== + +renderkid@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/renderkid/-/renderkid-3.0.0.tgz#5fd823e4d6951d37358ecc9a58b1f06836b6268a" + integrity sha512-q/7VIQA8lmM1hF+jn+sFSPWGlMkSAeNYcPLmDQx2zzuiDfaLrOmumR8iaUKlenFgh0XRPIUeSPlH3A+AW3Z5pg== + dependencies: + css-select "^4.1.3" + dom-converter "^0.2.0" + htmlparser2 "^6.1.0" + lodash "^4.17.21" + strip-ansi "^6.0.1" + +require-from-string@^2.0.2: + version "2.0.2" + resolved "https://registry.yarnpkg.com/require-from-string/-/require-from-string-2.0.2.tgz#89a7fdd938261267318eafe14f9c32e598c36909" + integrity sha512-Xf0nWe6RseziFMu+Ap9biiUbmplq6S9/p+7w7YXP/JBHhrUDDUhwa+vANyubuqfZWTveU//DYVGsDG7RKL/vEw== + +requires-port@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/requires-port/-/requires-port-1.0.0.tgz#925d2601d39ac485e091cf0da5c6e694dc3dcaff" + integrity sha512-KigOCHcocU3XODJxsu8i/j8T9tzT4adHiecwORRQ0ZZFcp7ahwXuRU1m+yuO90C5ZUyGeGfocHDI14M3L3yDAQ== + +reselect@^4.1.7: + version "4.1.8" + resolved "https://registry.yarnpkg.com/reselect/-/reselect-4.1.8.tgz#3f5dc671ea168dccdeb3e141236f69f02eaec524" + integrity sha512-ab9EmR80F/zQTMNeneUr4cv+jSwPJgIlvEmVwLerwrWVbpLlBuls9XHzIeTFy4cegU2NHBp3va0LKOzU5qFEYQ== + +resolve-cwd@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/resolve-cwd/-/resolve-cwd-3.0.0.tgz#0f0075f1bb2544766cf73ba6a6e2adfebcb13f2d" + integrity sha512-OrZaX2Mb+rJCpH/6CpSqt9xFVpN++x01XnN2ie9g6P5/3xelLAkXWVADpdz1IHD/KFfEXyE6V0U01OQ3UO2rEg== + dependencies: + resolve-from "^5.0.0" + +resolve-from@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/resolve-from/-/resolve-from-4.0.0.tgz#4abcd852ad32dd7baabfe9b40e00a36db5f392e6" + integrity sha512-pb/MYmXstAkysRFx8piNI1tGFNQIFA3vkE3Gq4EuA1dF6gHp/+vgZqsCGJapvy8N3Q+4o7FwvquPJcnZ7RYy4g== + +resolve-from@^5.0.0: + version "5.0.0" + resolved "https://registry.yarnpkg.com/resolve-from/-/resolve-from-5.0.0.tgz#c35225843df8f776df21c57557bc087e9dfdfc69" + integrity sha512-qYg9KP24dD5qka9J47d0aVky0N+b4fTU89LN9iDnjB5waksiC49rvMB0PrUJQGoTmH50XPiqOvAjDfaijGxYZw== + +resolve@^1.14.2, resolve@^1.20.0, resolve@^1.22.1: + version "1.22.2" + resolved "https://registry.yarnpkg.com/resolve/-/resolve-1.22.2.tgz#0ed0943d4e301867955766c9f3e1ae6d01c6845f" + integrity sha512-Sb+mjNHOULsBv818T40qSPeRiuWLyaGMa5ewydRLFimneixmVy2zdivRl+AF6jaYPC8ERxGDmFSiqui6SfPd+g== + dependencies: + is-core-module "^2.11.0" + path-parse "^1.0.7" + supports-preserve-symlinks-flag "^1.0.0" + +restore-cursor@^3.1.0: + version "3.1.0" + resolved "https://registry.yarnpkg.com/restore-cursor/-/restore-cursor-3.1.0.tgz#39f67c54b3a7a58cea5236d95cf0034239631f7e" + integrity sha512-l+sSefzHpj5qimhFSE5a8nufZYAM3sBSVMAPtYkmC+4EH2anSGaEMXSD0izRQbu9nfyQ9y5JrVmp7E8oZrUjvA== + dependencies: + onetime "^5.1.0" + signal-exit "^3.0.2" + +retry@^0.13.1: + version "0.13.1" + resolved "https://registry.yarnpkg.com/retry/-/retry-0.13.1.tgz#185b1587acf67919d63b357349e03537b2484658" + integrity sha512-XQBQ3I8W1Cge0Seh+6gjj03LbmRFWuoszgK9ooCpwYIrhhoO80pfq4cUkU5DkknwfOfFteRwlZ56PYOGYyFWdg== + +reusify@^1.0.4: + version "1.0.4" + resolved "https://registry.yarnpkg.com/reusify/-/reusify-1.0.4.tgz#90da382b1e126efc02146e90845a88db12925d76" + integrity sha512-U9nH88a3fc/ekCF1l0/UP1IosiuIjyTh7hBvXVMHYgVcfGvt897Xguj2UOLDeI5BG2m7/uwyaLVT6fbtCwTyzw== + +rfdc@^1.3.0: + version "1.3.0" + resolved "https://registry.yarnpkg.com/rfdc/-/rfdc-1.3.0.tgz#d0b7c441ab2720d05dc4cf26e01c89631d9da08b" + integrity sha512-V2hovdzFbOi77/WajaSMXk2OLm+xNIeQdMMuB7icj7bk6zi2F8GGAxigcnDFpJHbNyNcgyJDiP+8nOrY5cZGrA== + +rimraf@^3.0.2: + version "3.0.2" + resolved "https://registry.yarnpkg.com/rimraf/-/rimraf-3.0.2.tgz#f1a5402ba6220ad52cc1282bac1ae3aa49fd061a" + integrity sha512-JZkJMZkAGFFPP2YqXZXPbMlMBgsxzE8ILs4lMIX/2o0L9UBw9O/Y3o6wFw/i9YLapcUJWwqbi3kdxIPdC62TIA== + dependencies: + glob "^7.1.3" + +run-parallel@^1.1.9: + version "1.2.0" + resolved "https://registry.yarnpkg.com/run-parallel/-/run-parallel-1.2.0.tgz#66d1368da7bdf921eb9d95bd1a9229e7f21a43ee" + integrity sha512-5l4VyZR86LZ/lDxZTR6jqL8AFE2S0IFLMP26AbjsLVADxHdhB/c0GUsH+y39UfCi3dzz8OlQuPmnaJOMoDHQBA== + dependencies: + queue-microtask "^1.2.2" + +rxjs@^7.8.0: + version "7.8.1" + resolved "https://registry.yarnpkg.com/rxjs/-/rxjs-7.8.1.tgz#6f6f3d99ea8044291efd92e7c7fcf562c4057543" + integrity sha512-AA3TVj+0A2iuIoQkWEK/tqFjBq2j+6PO6Y0zJcvzLAFhEFIO3HL0vls9hWLncZbAAbK0mar7oZ4V079I/qPMxg== + dependencies: + tslib "^2.1.0" + +safe-buffer@5.1.2, safe-buffer@~5.1.0, safe-buffer@~5.1.1: + version "5.1.2" + resolved "https://registry.yarnpkg.com/safe-buffer/-/safe-buffer-5.1.2.tgz#991ec69d296e0313747d59bdfd2b745c35f8828d" + integrity sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g== + +safe-buffer@5.2.1, safe-buffer@>=5.1.0, safe-buffer@^5.1.0, safe-buffer@~5.2.0: + version "5.2.1" + resolved "https://registry.yarnpkg.com/safe-buffer/-/safe-buffer-5.2.1.tgz#1eaf9fa9bdb1fdd4ec75f58f9cdb4e6b7827eec6" + integrity sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ== + +safe-regex-test@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/safe-regex-test/-/safe-regex-test-1.0.0.tgz#793b874d524eb3640d1873aad03596db2d4f2295" + integrity sha512-JBUUzyOgEwXQY1NuPtvcj/qcBDbDmEvWufhlnXZIm75DEHp+afM1r1ujJpJsV/gSM4t59tpDyPi1sd6ZaPFfsA== + dependencies: + call-bind "^1.0.2" + get-intrinsic "^1.1.3" + is-regex "^1.1.4" + +"safer-buffer@>= 2.1.2 < 3": + version "2.1.2" + resolved "https://registry.yarnpkg.com/safer-buffer/-/safer-buffer-2.1.2.tgz#44fa161b0187b9549dd84bb91802f9bd8385cd6a" + integrity sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg== + +schema-utils@^3.1.1, schema-utils@^3.2.0: + version "3.3.0" + resolved "https://registry.yarnpkg.com/schema-utils/-/schema-utils-3.3.0.tgz#f50a88877c3c01652a15b622ae9e9795df7a60fe" + integrity sha512-pN/yOAvcC+5rQ5nERGuwrjLlYvLTbCibnZ1I7B1LaiAz9BRBlE9GMgE/eqV30P7aJQUf7Ddimy/RsbYO/GrVGg== + dependencies: + "@types/json-schema" "^7.0.8" + ajv "^6.12.5" + ajv-keywords "^3.5.2" + +schema-utils@^4.0.0: + version "4.2.0" + resolved "https://registry.yarnpkg.com/schema-utils/-/schema-utils-4.2.0.tgz#70d7c93e153a273a805801882ebd3bff20d89c8b" + integrity sha512-L0jRsrPpjdckP3oPug3/VxNKt2trR8TcabrM6FOAAlvC/9Phcmm+cuAgTlxBqdBR1WJx7Naj9WHw+aOmheSVbw== + dependencies: + "@types/json-schema" "^7.0.9" + ajv "^8.9.0" + ajv-formats "^2.1.1" + ajv-keywords "^5.1.0" + +select-hose@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/select-hose/-/select-hose-2.0.0.tgz#625d8658f865af43ec962bfc376a37359a4994ca" + integrity sha512-mEugaLK+YfkijB4fx0e6kImuJdCIt2LxCRcbEYPqRGCs4F2ogyfZU5IAZRdjCP8JPq2AtdNoC/Dux63d9Kiryg== + +selfsigned@^2.1.1: + version "2.1.1" + resolved "https://registry.yarnpkg.com/selfsigned/-/selfsigned-2.1.1.tgz#18a7613d714c0cd3385c48af0075abf3f266af61" + integrity sha512-GSL3aowiF7wa/WtSFwnUrludWFoNhftq8bUkH9pkzjpN2XSPOAYEgg6e0sS9s0rZwgJzJiQRPU18A6clnoW5wQ== + dependencies: + node-forge "^1" + +semver@^6.0.0, semver@^6.1.1, semver@^6.1.2, semver@^6.3.0: + version "6.3.0" + resolved "https://registry.yarnpkg.com/semver/-/semver-6.3.0.tgz#ee0a64c8af5e8ceea67687b133761e1becbd1d3d" + integrity sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw== + +semver@^7.3.8: + version "7.5.3" + resolved "https://registry.yarnpkg.com/semver/-/semver-7.5.3.tgz#161ce8c2c6b4b3bdca6caadc9fa3317a4c4fe88e" + integrity sha512-QBlUtyVk/5EeHbi7X0fw6liDZc7BBmEaSYn01fMU1OUYbf6GPsbTtd8WmnqbI20SeycoHSeiybkE/q1Q+qlThQ== + dependencies: + lru-cache "^6.0.0" + +send@0.18.0: + version "0.18.0" + resolved "https://registry.yarnpkg.com/send/-/send-0.18.0.tgz#670167cc654b05f5aa4a767f9113bb371bc706be" + integrity sha512-qqWzuOjSFOuqPjFe4NOsMLafToQQwBSOEpS+FwEt3A2V3vKubTquT3vmLTQpFgMXp8AlFWFuP1qKaJZOtPpVXg== + dependencies: + debug "2.6.9" + depd "2.0.0" + destroy "1.2.0" + encodeurl "~1.0.2" + escape-html "~1.0.3" + etag "~1.8.1" + fresh "0.5.2" + http-errors "2.0.0" + mime "1.6.0" + ms "2.1.3" + on-finished "2.4.1" + range-parser "~1.2.1" + statuses "2.0.1" + +serialize-javascript@^6.0.1: + version "6.0.1" + resolved "https://registry.yarnpkg.com/serialize-javascript/-/serialize-javascript-6.0.1.tgz#b206efb27c3da0b0ab6b52f48d170b7996458e5c" + integrity sha512-owoXEFjWRllis8/M1Q+Cw5k8ZH40e3zhp/ovX+Xr/vi1qj6QesbyXXViFbpNvWvPNAD62SutwEXavefrLJWj7w== + dependencies: + randombytes "^2.1.0" + +serve-index@^1.9.1: + version "1.9.1" + resolved "https://registry.yarnpkg.com/serve-index/-/serve-index-1.9.1.tgz#d3768d69b1e7d82e5ce050fff5b453bea12a9239" + integrity sha512-pXHfKNP4qujrtteMrSBb0rc8HJ9Ms/GrXwcUtUtD5s4ewDJI8bT3Cz2zTVRMKtri49pLx2e0Ya8ziP5Ya2pZZw== + dependencies: + accepts "~1.3.4" + batch "0.6.1" + debug "2.6.9" + escape-html "~1.0.3" + http-errors "~1.6.2" + mime-types "~2.1.17" + parseurl "~1.3.2" + +serve-static@1.15.0: + version "1.15.0" + resolved "https://registry.yarnpkg.com/serve-static/-/serve-static-1.15.0.tgz#faaef08cffe0a1a62f60cad0c4e513cff0ac9540" + integrity sha512-XGuRDNjXUijsUL0vl6nSD7cwURuzEgglbOaFuZM9g3kwDXOWVTck0jLzjPzGD+TazWbboZYu52/9/XPdUgne9g== + dependencies: + encodeurl "~1.0.2" + escape-html "~1.0.3" + parseurl "~1.3.3" + send "0.18.0" + +setprototypeof@1.1.0: + version "1.1.0" + resolved "https://registry.yarnpkg.com/setprototypeof/-/setprototypeof-1.1.0.tgz#d0bd85536887b6fe7c0d818cb962d9d91c54e656" + integrity sha512-BvE/TwpZX4FXExxOxZyRGQQv651MSwmWKZGqvmPcRIjDqWub67kTKuIMx43cZZrS/cBBzwBcNDWoFxt2XEFIpQ== + +setprototypeof@1.2.0: + version "1.2.0" + resolved "https://registry.yarnpkg.com/setprototypeof/-/setprototypeof-1.2.0.tgz#66c9a24a73f9fc28cbe66b09fed3d33dcaf1b424" + integrity sha512-E5LDX7Wrp85Kil5bhZv46j8jOeboKq5JMmYM3gVGdGH8xFpPWXUMsNrlODCrkoxMEeNi/XZIwuRvY4XNwYMJpw== + +shallow-clone@^3.0.0: + version "3.0.1" + resolved "https://registry.yarnpkg.com/shallow-clone/-/shallow-clone-3.0.1.tgz#8f2981ad92531f55035b01fb230769a40e02efa3" + integrity sha512-/6KqX+GVUdqPuPPd2LxDDxzX6CAbjJehAAOKlNpqqUpAqPM6HeL8f+o3a+JsyGjn2lv0WY8UsTgUJjU9Ok55NA== + dependencies: + kind-of "^6.0.2" + +shebang-command@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/shebang-command/-/shebang-command-2.0.0.tgz#ccd0af4f8835fbdc265b82461aaf0c36663f34ea" + integrity sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA== + dependencies: + shebang-regex "^3.0.0" + +shebang-regex@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/shebang-regex/-/shebang-regex-3.0.0.tgz#ae16f1644d873ecad843b0307b143362d4c42172" + integrity sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A== + +shell-quote@^1.7.3: + version "1.8.1" + resolved "https://registry.yarnpkg.com/shell-quote/-/shell-quote-1.8.1.tgz#6dbf4db75515ad5bac63b4f1894c3a154c766680" + integrity sha512-6j1W9l1iAs/4xYBI1SYOVZyFcCis9b4KCLQ8fgAGG07QvzaRLVVRQvAy85yNmmZSjYjg4MWh4gNvlPujU/5LpA== + +side-channel@^1.0.4: + version "1.0.4" + resolved "https://registry.yarnpkg.com/side-channel/-/side-channel-1.0.4.tgz#efce5c8fdc104ee751b25c58d4290011fa5ea2cf" + integrity sha512-q5XPytqFEIKHkGdiMIrY10mvLRvnQh42/+GoBlFW3b2LXLE2xxJpZFdm94we0BaoV3RwJyGqg5wS7epxTv0Zvw== + dependencies: + call-bind "^1.0.0" + get-intrinsic "^1.0.2" + object-inspect "^1.9.0" + +signal-exit@^3.0.2, signal-exit@^3.0.3, signal-exit@^3.0.7: + version "3.0.7" + resolved "https://registry.yarnpkg.com/signal-exit/-/signal-exit-3.0.7.tgz#a9a1767f8af84155114eaabd73f99273c8f59ad9" + integrity sha512-wnD2ZE+l+SPC/uoS0vXeE9L1+0wuaMqKlfz9AMUo38JsyLSBWSFcHR1Rri62LZc12vLr1gb3jl7iwQhgwpAbGQ== + +slice-ansi@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/slice-ansi/-/slice-ansi-3.0.0.tgz#31ddc10930a1b7e0b67b08c96c2f49b77a789787" + integrity sha512-pSyv7bSTC7ig9Dcgbw9AuRNUb5k5V6oDudjZoMBSr13qpLBG7tB+zgCkARjq7xIUgdz5P1Qe8u+rSGdouOOIyQ== + dependencies: + ansi-styles "^4.0.0" + astral-regex "^2.0.0" + is-fullwidth-code-point "^3.0.0" + +slice-ansi@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/slice-ansi/-/slice-ansi-4.0.0.tgz#500e8dd0fd55b05815086255b3195adf2a45fe6b" + integrity sha512-qMCMfhY040cVHT43K9BFygqYbUPFZKHOg7K73mtTWJRb8pyP3fzf4Ixd5SzdEJQ6MRUg/WBnOLxghZtKKurENQ== + dependencies: + ansi-styles "^4.0.0" + astral-regex "^2.0.0" + is-fullwidth-code-point "^3.0.0" + +slice-ansi@^5.0.0: + version "5.0.0" + resolved "https://registry.yarnpkg.com/slice-ansi/-/slice-ansi-5.0.0.tgz#b73063c57aa96f9cd881654b15294d95d285c42a" + integrity sha512-FC+lgizVPfie0kkhqUScwRu1O/lF6NOgJmlCgK+/LYxDCTk8sGelYaHDhFcDN+Sn3Cv+3VSa4Byeo+IMCzpMgQ== + dependencies: + ansi-styles "^6.0.0" + is-fullwidth-code-point "^4.0.0" + +sockjs@^0.3.24: + version "0.3.24" + resolved "https://registry.yarnpkg.com/sockjs/-/sockjs-0.3.24.tgz#c9bc8995f33a111bea0395ec30aa3206bdb5ccce" + integrity sha512-GJgLTZ7vYb/JtPSSZ10hsOYIvEYsjbNU+zPdIHcUaWVNUEPivzxku31865sSSud0Da0W4lEeOPlmw93zLQchuQ== + dependencies: + faye-websocket "^0.11.3" + uuid "^8.3.2" + websocket-driver "^0.7.4" + +source-map-js@^1.0.2: + version "1.0.2" + resolved "https://registry.yarnpkg.com/source-map-js/-/source-map-js-1.0.2.tgz#adbc361d9c62df380125e7f161f71c826f1e490c" + integrity sha512-R0XvVJ9WusLiqTCEiGCmICCMplcCkIwwR11mOSD9CR5u+IXYdiseeEuXCVAjS54zqwkLcPNnmU4OeJ6tUrWhDw== + +source-map-support@~0.5.20: + version "0.5.21" + resolved "https://registry.yarnpkg.com/source-map-support/-/source-map-support-0.5.21.tgz#04fe7c7f9e1ed2d662233c28cb2b35b9f63f6e4f" + integrity sha512-uBHU3L3czsIyYXKX88fdrGovxdSCoTGDRZ6SYXtSRxLZUzHg5P/66Ht6uoUlHu9EZod+inXhKo3qQgwXUT/y1w== + dependencies: + buffer-from "^1.0.0" + source-map "^0.6.0" + +source-map@^0.6.0, source-map@~0.6.0: + version "0.6.1" + resolved "https://registry.yarnpkg.com/source-map/-/source-map-0.6.1.tgz#74722af32e9614e9c287a8d0bbde48b5e2f1a263" + integrity sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g== + +spdy-transport@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/spdy-transport/-/spdy-transport-3.0.0.tgz#00d4863a6400ad75df93361a1608605e5dcdcf31" + integrity sha512-hsLVFE5SjA6TCisWeJXFKniGGOpBgMLmerfO2aCyCU5s7nJ/rpAepqmFifv/GCbSbueEeAJJnmSQ2rKC/g8Fcw== + dependencies: + debug "^4.1.0" + detect-node "^2.0.4" + hpack.js "^2.1.6" + obuf "^1.1.2" + readable-stream "^3.0.6" + wbuf "^1.7.3" + +spdy@^4.0.2: + version "4.0.2" + resolved "https://registry.yarnpkg.com/spdy/-/spdy-4.0.2.tgz#b74f466203a3eda452c02492b91fb9e84a27677b" + integrity sha512-r46gZQZQV+Kl9oItvl1JZZqJKGr+oEkB08A6BzkiR7593/7IbtuncXHd2YoYeTsG4157ZssMu9KYvUHLcjcDoA== + dependencies: + debug "^4.1.0" + handle-thing "^2.0.0" + http-deceiver "^1.2.7" + select-hose "^2.0.0" + spdy-transport "^3.0.0" + +statuses@2.0.1: + version "2.0.1" + resolved "https://registry.yarnpkg.com/statuses/-/statuses-2.0.1.tgz#55cb000ccf1d48728bd23c685a063998cf1a1b63" + integrity sha512-RwNA9Z/7PrK06rYLIzFMlaF+l73iwpzsqRIFgbMLbTcLD6cOao82TaWefPXQvB2fOC4AjuYSEndS7N/mTCbkdQ== + +"statuses@>= 1.4.0 < 2": + version "1.5.0" + resolved "https://registry.yarnpkg.com/statuses/-/statuses-1.5.0.tgz#161c7dac177659fd9811f43771fa99381478628c" + integrity sha512-OpZ3zP+jT1PI7I8nemJX4AKmAX070ZkYPVWV/AaKTJl+tXCTGyVdC1a4SL8RUQYEwk/f34ZX8UTykN68FwrqAA== + +string-argv@^0.3.1: + version "0.3.2" + resolved "https://registry.yarnpkg.com/string-argv/-/string-argv-0.3.2.tgz#2b6d0ef24b656274d957d54e0a4bbf6153dc02b6" + integrity sha512-aqD2Q0144Z+/RqG52NeHEkZauTAUWJO8c6yTftGJKO3Tja5tUgIfmIl6kExvhtxSDP7fXB6DvzkfMpCd/F3G+Q== + +string-width@^4.1.0, string-width@^4.2.0: + version "4.2.3" + resolved "https://registry.yarnpkg.com/string-width/-/string-width-4.2.3.tgz#269c7117d27b05ad2e536830a8ec895ef9c6d010" + integrity sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g== + dependencies: + emoji-regex "^8.0.0" + is-fullwidth-code-point "^3.0.0" + strip-ansi "^6.0.1" + +string-width@^5.0.0: + version "5.1.2" + resolved "https://registry.yarnpkg.com/string-width/-/string-width-5.1.2.tgz#14f8daec6d81e7221d2a357e668cab73bdbca794" + integrity sha512-HnLOCR3vjcY8beoNLtcjZ5/nxn2afmME6lhrDrebokqMap+XbeW8n9TXpPDOqdGK5qcI3oT0GKTW6wC7EMiVqA== + dependencies: + eastasianwidth "^0.2.0" + emoji-regex "^9.2.2" + strip-ansi "^7.0.1" + +string.prototype.trim@^1.2.7: + version "1.2.7" + resolved "https://registry.yarnpkg.com/string.prototype.trim/-/string.prototype.trim-1.2.7.tgz#a68352740859f6893f14ce3ef1bb3037f7a90533" + integrity sha512-p6TmeT1T3411M8Cgg9wBTMRtY2q9+PNy9EV1i2lIXUN/btt763oIfxwN3RR8VU6wHX8j/1CFy0L+YuThm6bgOg== + dependencies: + call-bind "^1.0.2" + define-properties "^1.1.4" + es-abstract "^1.20.4" + +string.prototype.trimend@^1.0.6: + version "1.0.6" + resolved "https://registry.yarnpkg.com/string.prototype.trimend/-/string.prototype.trimend-1.0.6.tgz#c4a27fa026d979d79c04f17397f250a462944533" + integrity sha512-JySq+4mrPf9EsDBEDYMOb/lM7XQLulwg5R/m1r0PXEFqrV0qHvl58sdTilSXtKOflCsK2E8jxf+GKC0T07RWwQ== + dependencies: + call-bind "^1.0.2" + define-properties "^1.1.4" + es-abstract "^1.20.4" + +string.prototype.trimstart@^1.0.6: + version "1.0.6" + resolved "https://registry.yarnpkg.com/string.prototype.trimstart/-/string.prototype.trimstart-1.0.6.tgz#e90ab66aa8e4007d92ef591bbf3cd422c56bdcf4" + integrity sha512-omqjMDaY92pbn5HOX7f9IccLA+U1tA9GvtU4JrodiXFfYB7jPzzHpRzpglLAjtUV6bB557zwClJezTqnAiYnQA== + dependencies: + call-bind "^1.0.2" + define-properties "^1.1.4" + es-abstract "^1.20.4" + +string_decoder@^1.1.1: + version "1.3.0" + resolved "https://registry.yarnpkg.com/string_decoder/-/string_decoder-1.3.0.tgz#42f114594a46cf1a8e30b0a84f56c78c3edac21e" + integrity sha512-hkRX8U1WjJFd8LsDJ2yQ/wWWxaopEsABU1XfkM8A+j0+85JAGppt16cr1Whg6KIbb4okU6Mql6BOj+uup/wKeA== + dependencies: + safe-buffer "~5.2.0" + +string_decoder@~1.1.1: + version "1.1.1" + resolved "https://registry.yarnpkg.com/string_decoder/-/string_decoder-1.1.1.tgz#9cf1611ba62685d7030ae9e4ba34149c3af03fc8" + integrity sha512-n/ShnvDi6FHbbVfviro+WojiFzv+s8MPMHBczVePfUpDJLwoLT0ht1l4YwBCbi8pJAveEEdnkHyPyTP/mzRfwg== + dependencies: + safe-buffer "~5.1.0" + +strip-ansi@^6.0.0, strip-ansi@^6.0.1: + version "6.0.1" + resolved "https://registry.yarnpkg.com/strip-ansi/-/strip-ansi-6.0.1.tgz#9e26c63d30f53443e9489495b2105d37b67a85d9" + integrity sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A== + dependencies: + ansi-regex "^5.0.1" + +strip-ansi@^7.0.1: + version "7.1.0" + resolved "https://registry.yarnpkg.com/strip-ansi/-/strip-ansi-7.1.0.tgz#d5b6568ca689d8561370b0707685d22434faff45" + integrity sha512-iq6eVVI64nQQTRYq2KtEg2d2uU7LElhTJwsH4YzIHZshxlgZms/wIc4VoDQTlG/IvVIrBKG06CrZnp0qv7hkcQ== + dependencies: + ansi-regex "^6.0.1" + +strip-bom@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/strip-bom/-/strip-bom-3.0.0.tgz#2334c18e9c759f7bdd56fdef7e9ae3d588e68ed3" + integrity sha512-vavAMRXOgBVNF6nyEEmL3DBK19iRpDcoIwW+swQ+CbGiu7lju6t+JklA1MHweoWtadgt4ISVUsXLyDq34ddcwA== + +strip-final-newline@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/strip-final-newline/-/strip-final-newline-2.0.0.tgz#89b852fb2fcbe936f6f4b3187afb0a12c1ab58ad" + integrity sha512-BrpvfNAE3dcvq7ll3xVumzjKjZQ5tI1sEUIKr3Uoks0XUl45St3FlatVqef9prk4jRDzhW6WZg+3bk93y6pLjA== + +strip-final-newline@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/strip-final-newline/-/strip-final-newline-3.0.0.tgz#52894c313fbff318835280aed60ff71ebf12b8fd" + integrity sha512-dOESqjYr96iWYylGObzd39EuNTa5VJxyvVAEm5Jnh7KGo75V43Hk1odPQkNDyXNmUR6k+gEiDVXnjB8HJ3crXw== + +strip-json-comments@^3.1.0, strip-json-comments@^3.1.1: + version "3.1.1" + resolved "https://registry.yarnpkg.com/strip-json-comments/-/strip-json-comments-3.1.1.tgz#31f1281b3832630434831c310c01cccda8cbe006" + integrity sha512-6fPc+R4ihwqP6N/aIv2f1gMH8lOVtWQHoqC4yK6oSDVVocumAsfCqjkXnqiYMhmMwS/mEHLp7Vehlt3ql6lEig== + +style-loader@^3.3.3: + version "3.3.3" + resolved "https://registry.yarnpkg.com/style-loader/-/style-loader-3.3.3.tgz#bba8daac19930169c0c9c96706749a597ae3acff" + integrity sha512-53BiGLXAcll9maCYtZi2RCQZKa8NQQai5C4horqKyRmHj9H7QmcUyucrH+4KW/gBQbXM2AsB0axoEcFZPlfPcw== + +supports-color@^5.3.0: + version "5.5.0" + resolved "https://registry.yarnpkg.com/supports-color/-/supports-color-5.5.0.tgz#e2e69a44ac8772f78a1ec0b35b689df6530efc8f" + integrity sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow== + dependencies: + has-flag "^3.0.0" + +supports-color@^7.1.0: + version "7.2.0" + resolved "https://registry.yarnpkg.com/supports-color/-/supports-color-7.2.0.tgz#1b7dcdcb32b8138801b3e478ba6a51caa89648da" + integrity sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw== + dependencies: + has-flag "^4.0.0" + +supports-color@^8.0.0: + version "8.1.1" + resolved "https://registry.yarnpkg.com/supports-color/-/supports-color-8.1.1.tgz#cd6fc17e28500cff56c1b86c0a7fd4a54a73005c" + integrity sha512-MpUEN2OodtUzxvKQl72cUF7RQ5EiHsGvSsVG0ia9c5RbWGL2CI4C7EpPS8UTBIplnlzZiNuV56w+FuNxy3ty2Q== + dependencies: + has-flag "^4.0.0" + +supports-preserve-symlinks-flag@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/supports-preserve-symlinks-flag/-/supports-preserve-symlinks-flag-1.0.0.tgz#6eda4bd344a3c94aea376d4cc31bc77311039e09" + integrity sha512-ot0WnXS9fgdkgIcePe6RHNk1WA8+muPa6cSjeR3V8K27q9BB1rTE3R1p7Hv0z1ZyAc8s6Vvv8DIyWf681MAt0w== + +tapable@^2.0.0, tapable@^2.1.1, tapable@^2.2.0: + version "2.2.1" + resolved "https://registry.yarnpkg.com/tapable/-/tapable-2.2.1.tgz#1967a73ef4060a82f12ab96af86d52fdb76eeca0" + integrity sha512-GNzQvQTOIP6RyTfE2Qxb8ZVlNmw0n88vp1szwWRimP02mnTsx3Wtn5qRdqY9w2XduFNUgvOwhNnQsjwCp+kqaQ== + +terser-webpack-plugin@^5.3.7: + version "5.3.9" + resolved "https://registry.yarnpkg.com/terser-webpack-plugin/-/terser-webpack-plugin-5.3.9.tgz#832536999c51b46d468067f9e37662a3b96adfe1" + integrity sha512-ZuXsqE07EcggTWQjXUj+Aot/OMcD0bMKGgF63f7UxYcu5/AJF53aIpK1YoP5xR9l6s/Hy2b+t1AM0bLNPRuhwA== + dependencies: + "@jridgewell/trace-mapping" "^0.3.17" + jest-worker "^27.4.5" + schema-utils "^3.1.1" + serialize-javascript "^6.0.1" + terser "^5.16.8" + +terser@^5.10.0, terser@^5.16.8: + version "5.18.2" + resolved "https://registry.yarnpkg.com/terser/-/terser-5.18.2.tgz#ff3072a0faf21ffd38f99acc9a0ddf7b5f07b948" + integrity sha512-Ah19JS86ypbJzTzvUCX7KOsEIhDaRONungA4aYBjEP3JZRf4ocuDzTg4QWZnPn9DEMiMYGJPiSOy7aykoCc70w== + dependencies: + "@jridgewell/source-map" "^0.3.3" + acorn "^8.8.2" + commander "^2.20.0" + source-map-support "~0.5.20" + +text-table@^0.2.0: + version "0.2.0" + resolved "https://registry.yarnpkg.com/text-table/-/text-table-0.2.0.tgz#7f5ee823ae805207c00af2df4a84ec3fcfa570b4" + integrity sha512-N+8UisAXDGk8PFXP4HAzVR9nbfmVJ3zYLAWiTIoqC5v5isinhr+r5uaO8+7r3BMfuNIufIsA7RdpVgacC2cSpw== + +through@^2.3.8: + version "2.3.8" + resolved "https://registry.yarnpkg.com/through/-/through-2.3.8.tgz#0dd4c9ffaabc357960b1b724115d7e0e86a2e1f5" + integrity sha512-w89qg7PI8wAdvX60bMDP+bFoD5Dvhm9oLheFp5O4a2QF0cSBGsBX4qZmadPMvVqlLJBBci+WqGGOAPvcDeNSVg== + +thunky@^1.0.2: + version "1.1.0" + resolved "https://registry.yarnpkg.com/thunky/-/thunky-1.1.0.tgz#5abaf714a9405db0504732bbccd2cedd9ef9537d" + integrity sha512-eHY7nBftgThBqOyHGVN+l8gF0BucP09fMo0oO/Lb0w1OF80dJv+lDVpXG60WMQvkcxAkNybKsrEIE3ZtKGmPrA== + +to-fast-properties@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/to-fast-properties/-/to-fast-properties-2.0.0.tgz#dc5e698cbd079265bc73e0377681a4e4e83f616e" + integrity sha512-/OaKK0xYrs3DmxRYqL/yDc+FxFUVYhDlXMhRmv3z915w2HF1tnN1omB354j8VUGO/hbRzyD6Y3sA7v7GS/ceog== + +to-regex-range@^5.0.1: + version "5.0.1" + resolved "https://registry.yarnpkg.com/to-regex-range/-/to-regex-range-5.0.1.tgz#1648c44aae7c8d988a326018ed72f5b4dd0392e4" + integrity sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ== + dependencies: + is-number "^7.0.0" + +toidentifier@1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/toidentifier/-/toidentifier-1.0.1.tgz#3be34321a88a820ed1bd80dfaa33e479fbb8dd35" + integrity sha512-o5sSPKEkg/DIQNmH43V0/uerLrpzVedkUh8tGNvaeXpfpuwjKenlSox/2O/BTlZUtEe+JG7s5YhEz608PlAHRA== + +tsconfig-paths@^3.14.1: + version "3.14.2" + resolved "https://registry.yarnpkg.com/tsconfig-paths/-/tsconfig-paths-3.14.2.tgz#6e32f1f79412decd261f92d633a9dc1cfa99f088" + integrity sha512-o/9iXgCYc5L/JxCHPe3Hvh8Q/2xm5Z+p18PESBU6Ff33695QnCHBEjcytY2q19ua7Mbl/DavtBOLq+oG0RCL+g== + dependencies: + "@types/json5" "^0.0.29" + json5 "^1.0.2" + minimist "^1.2.6" + strip-bom "^3.0.0" + +tslib@^2.0.3, tslib@^2.1.0: + version "2.6.0" + resolved "https://registry.yarnpkg.com/tslib/-/tslib-2.6.0.tgz#b295854684dbda164e181d259a22cd779dcd7bc3" + integrity sha512-7At1WUettjcSRHXCyYtTselblcHl9PJFFVKiCAy/bY97+BPZXSQ2wbq0P9s8tK2G7dFQfNnlJnPAiArVBVBsfA== + +type-check@^0.4.0, type-check@~0.4.0: + version "0.4.0" + resolved "https://registry.yarnpkg.com/type-check/-/type-check-0.4.0.tgz#07b8203bfa7056c0657050e3ccd2c37730bab8f1" + integrity sha512-XleUoc9uwGXqjWwXaUTZAmzMcFZ5858QA2vvx1Ur5xIcixXIP+8LnFDgRplU30us6teqdlskFfu+ae4K79Ooew== + dependencies: + prelude-ls "^1.2.1" + +type-fest@^0.20.2: + version "0.20.2" + resolved "https://registry.yarnpkg.com/type-fest/-/type-fest-0.20.2.tgz#1bf207f4b28f91583666cb5fbd327887301cd5f4" + integrity sha512-Ne+eE4r0/iWnpAxD852z3A+N0Bt5RN//NjJwRd2VFHEmrywxf5vsZlh4R6lixl6B+wz/8d+maTSAkN1FIkI3LQ== + +type-fest@^0.21.3: + version "0.21.3" + resolved "https://registry.yarnpkg.com/type-fest/-/type-fest-0.21.3.tgz#d260a24b0198436e133fa26a524a6d65fa3b2e37" + integrity sha512-t0rzBq87m3fVcduHDUFhKmyyX+9eo6WQjZvf51Ea/M0Q7+T374Jp1aUiyUl0GKxp8M/OETVHSDvmkyPgvX+X2w== + +type-is@~1.6.18: + version "1.6.18" + resolved "https://registry.yarnpkg.com/type-is/-/type-is-1.6.18.tgz#4e552cd05df09467dcbc4ef739de89f2cf37c131" + integrity sha512-TkRKr9sUTxEH8MdfuCSP7VizJyzRNMjj2J2do2Jr3Kym598JVdEksuzPQCnlFPW4ky9Q+iA+ma9BGm06XQBy8g== + dependencies: + media-typer "0.3.0" + mime-types "~2.1.24" + +typed-array-length@^1.0.4: + version "1.0.4" + resolved "https://registry.yarnpkg.com/typed-array-length/-/typed-array-length-1.0.4.tgz#89d83785e5c4098bec72e08b319651f0eac9c1bb" + integrity sha512-KjZypGq+I/H7HI5HlOoGHkWUUGq+Q0TPhQurLbyrVrvnKTBgzLhIJ7j6J/XTQOi0d1RjyZ0wdas8bKs2p0x3Ng== + dependencies: + call-bind "^1.0.2" + for-each "^0.3.3" + is-typed-array "^1.1.9" + +unbox-primitive@^1.0.2: + version "1.0.2" + resolved "https://registry.yarnpkg.com/unbox-primitive/-/unbox-primitive-1.0.2.tgz#29032021057d5e6cdbd08c5129c226dff8ed6f9e" + integrity sha512-61pPlCD9h51VoreyJ0BReideM3MDKMKnh6+V9L08331ipq6Q8OFXZYiqP6n/tbHx4s5I9uRhcye6BrbkizkBDw== + dependencies: + call-bind "^1.0.2" + has-bigints "^1.0.2" + has-symbols "^1.0.3" + which-boxed-primitive "^1.0.2" + +unicode-canonical-property-names-ecmascript@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/unicode-canonical-property-names-ecmascript/-/unicode-canonical-property-names-ecmascript-2.0.0.tgz#301acdc525631670d39f6146e0e77ff6bbdebddc" + integrity sha512-yY5PpDlfVIU5+y/BSCxAJRBIS1Zc2dDG3Ujq+sR0U+JjUevW2JhocOF+soROYDSaAezOzOKuyyixhD6mBknSmQ== + +unicode-match-property-ecmascript@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/unicode-match-property-ecmascript/-/unicode-match-property-ecmascript-2.0.0.tgz#54fd16e0ecb167cf04cf1f756bdcc92eba7976c3" + integrity sha512-5kaZCrbp5mmbz5ulBkDkbY0SsPOjKqVS35VpL9ulMPfSl0J0Xsm+9Evphv9CoIZFwre7aJoa94AY6seMKGVN5Q== + dependencies: + unicode-canonical-property-names-ecmascript "^2.0.0" + unicode-property-aliases-ecmascript "^2.0.0" + +unicode-match-property-value-ecmascript@^2.1.0: + version "2.1.0" + resolved "https://registry.yarnpkg.com/unicode-match-property-value-ecmascript/-/unicode-match-property-value-ecmascript-2.1.0.tgz#cb5fffdcd16a05124f5a4b0bf7c3770208acbbe0" + integrity sha512-qxkjQt6qjg/mYscYMC0XKRn3Rh0wFPlfxB0xkt9CfyTvpX1Ra0+rAmdX2QyAobptSEvuy4RtpPRui6XkV+8wjA== + +unicode-property-aliases-ecmascript@^2.0.0: + version "2.1.0" + resolved "https://registry.yarnpkg.com/unicode-property-aliases-ecmascript/-/unicode-property-aliases-ecmascript-2.1.0.tgz#43d41e3be698bd493ef911077c9b131f827e8ccd" + integrity sha512-6t3foTQI9qne+OZoVQB/8x8rk2k1eVy1gRXhV3oFQ5T6R1dqQ1xtin3XqSlx3+ATBkliTaR/hHyJBm+LVPNM8w== + +unpipe@1.0.0, unpipe@~1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/unpipe/-/unpipe-1.0.0.tgz#b2bf4ee8514aae6165b4817829d21b2ef49904ec" + integrity sha512-pjy2bYhSsufwWlKwPc+l3cN7+wuJlK6uz0YdJEOlQDbl6jo/YlPi4mb8agUkVC8BF7V8NuzeyPNqRksA3hztKQ== + +update-browserslist-db@^1.0.11: + version "1.0.11" + resolved "https://registry.yarnpkg.com/update-browserslist-db/-/update-browserslist-db-1.0.11.tgz#9a2a641ad2907ae7b3616506f4b977851db5b940" + integrity sha512-dCwEFf0/oT85M1fHBg4F0jtLwJrutGoHSQXCh7u4o2t1drG+c0a9Flnqww6XUKSfQMPpJBRjU8d4RXB09qtvaA== + dependencies: + escalade "^3.1.1" + picocolors "^1.0.0" + +uri-js@^4.2.2: + version "4.4.1" + resolved "https://registry.yarnpkg.com/uri-js/-/uri-js-4.4.1.tgz#9b1a52595225859e55f669d928f88c6c57f2a77e" + integrity sha512-7rKUyy33Q1yc98pQ1DAmLtwX109F7TIfWlW1Ydo8Wl1ii1SeHieeh0HHfPeL2fMXK6z0s8ecKs9frCuLJvndBg== + dependencies: + punycode "^2.1.0" + +util-deprecate@^1.0.1, util-deprecate@^1.0.2, util-deprecate@~1.0.1: + version "1.0.2" + resolved "https://registry.yarnpkg.com/util-deprecate/-/util-deprecate-1.0.2.tgz#450d4dc9fa70de732762fbd2d4a28981419a0ccf" + integrity sha512-EPD5q1uXyFxJpCrLnCc1nHnq3gOa6DZBocAIiI2TaSCA7VCJ1UJDMagCzIkXNsUYfD1daK//LTEQ8xiIbrHtcw== + +utila@~0.4: + version "0.4.0" + resolved "https://registry.yarnpkg.com/utila/-/utila-0.4.0.tgz#8a16a05d445657a3aea5eecc5b12a4fa5379772c" + integrity sha512-Z0DbgELS9/L/75wZbro8xAnT50pBVFQZ+hUEueGDU5FN51YSCYM+jdxsfCiHjwNP/4LCDD0i/graKpeBnOXKRA== + +utils-merge@1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/utils-merge/-/utils-merge-1.0.1.tgz#9f95710f50a267947b2ccc124741c1028427e713" + integrity sha512-pMZTvIkT1d+TFGvDOqodOclx0QWkkgi6Tdoa8gC8ffGAAqz9pzPTZWAybbsHHoED/ztMtkv/VoYTYyShUn81hA== + +uuid@^8.3.2: + version "8.3.2" + resolved "https://registry.yarnpkg.com/uuid/-/uuid-8.3.2.tgz#80d5b5ced271bb9af6c445f21a1a04c606cefbe2" + integrity sha512-+NYs2QeMWy+GWFOEm9xnn6HCDp0l7QBD7ml8zLUmJ+93Q5NF0NocErnwkTkXVFNiX3/fpC6afS8Dhb/gz7R7eg== + +vary@~1.1.2: + version "1.1.2" + resolved "https://registry.yarnpkg.com/vary/-/vary-1.1.2.tgz#2299f02c6ded30d4a5961b0b9f74524a18f634fc" + integrity sha512-BNGbWLfd0eUPabhkXUVm0j8uuvREyTh5ovRa/dyow/BqAbZJyC+5fU+IzQOzmAKzYqYRAISoRhdQr3eIZ/PXqg== + +watchpack@^2.4.0: + version "2.4.0" + resolved "https://registry.yarnpkg.com/watchpack/-/watchpack-2.4.0.tgz#fa33032374962c78113f93c7f2fb4c54c9862a5d" + integrity sha512-Lcvm7MGST/4fup+ifyKi2hjyIAwcdI4HRgtvTpIUxBRhB+RFtUh8XtDOxUfctVCnhVi+QQj49i91OyvzkJl6cg== + dependencies: + glob-to-regexp "^0.4.1" + graceful-fs "^4.1.2" + +wbuf@^1.1.0, wbuf@^1.7.3: + version "1.7.3" + resolved "https://registry.yarnpkg.com/wbuf/-/wbuf-1.7.3.tgz#c1d8d149316d3ea852848895cb6a0bfe887b87df" + integrity sha512-O84QOnr0icsbFGLS0O3bI5FswxzRr8/gHwWkDlQFskhSPryQXvrTMxjxGP4+iWYoauLoBvfDpkrOauZ+0iZpDA== + dependencies: + minimalistic-assert "^1.0.0" + +webpack-cli@^5.1.4: + version "5.1.4" + resolved "https://registry.yarnpkg.com/webpack-cli/-/webpack-cli-5.1.4.tgz#c8e046ba7eaae4911d7e71e2b25b776fcc35759b" + integrity sha512-pIDJHIEI9LR0yxHXQ+Qh95k2EvXpWzZ5l+d+jIo+RdSm9MiHfzazIxwwni/p7+x4eJZuvG1AJwgC4TNQ7NRgsg== + dependencies: + "@discoveryjs/json-ext" "^0.5.0" + "@webpack-cli/configtest" "^2.1.1" + "@webpack-cli/info" "^2.0.2" + "@webpack-cli/serve" "^2.0.5" + colorette "^2.0.14" + commander "^10.0.1" + cross-spawn "^7.0.3" + envinfo "^7.7.3" + fastest-levenshtein "^1.0.12" + import-local "^3.0.2" + interpret "^3.1.1" + rechoir "^0.8.0" + webpack-merge "^5.7.3" + +webpack-dev-middleware@^5.3.1: + version "5.3.3" + resolved "https://registry.yarnpkg.com/webpack-dev-middleware/-/webpack-dev-middleware-5.3.3.tgz#efae67c2793908e7311f1d9b06f2a08dcc97e51f" + integrity sha512-hj5CYrY0bZLB+eTO+x/j67Pkrquiy7kWepMHmUMoPsmcUaeEnQJqFzHJOyxgWlq746/wUuA64p9ta34Kyb01pA== + dependencies: + colorette "^2.0.10" + memfs "^3.4.3" + mime-types "^2.1.31" + range-parser "^1.2.1" + schema-utils "^4.0.0" + +webpack-dev-server@^4.15.1: + version "4.15.1" + resolved "https://registry.yarnpkg.com/webpack-dev-server/-/webpack-dev-server-4.15.1.tgz#8944b29c12760b3a45bdaa70799b17cb91b03df7" + integrity sha512-5hbAst3h3C3L8w6W4P96L5vaV0PxSmJhxZvWKYIdgxOQm8pNZ5dEOmmSLBVpP85ReeyRt6AS1QJNyo/oFFPeVA== + dependencies: + "@types/bonjour" "^3.5.9" + "@types/connect-history-api-fallback" "^1.3.5" + "@types/express" "^4.17.13" + "@types/serve-index" "^1.9.1" + "@types/serve-static" "^1.13.10" + "@types/sockjs" "^0.3.33" + "@types/ws" "^8.5.5" + ansi-html-community "^0.0.8" + bonjour-service "^1.0.11" + chokidar "^3.5.3" + colorette "^2.0.10" + compression "^1.7.4" + connect-history-api-fallback "^2.0.0" + default-gateway "^6.0.3" + express "^4.17.3" + graceful-fs "^4.2.6" + html-entities "^2.3.2" + http-proxy-middleware "^2.0.3" + ipaddr.js "^2.0.1" + launch-editor "^2.6.0" + open "^8.0.9" + p-retry "^4.5.0" + rimraf "^3.0.2" + schema-utils "^4.0.0" + selfsigned "^2.1.1" + serve-index "^1.9.1" + sockjs "^0.3.24" + spdy "^4.0.2" + webpack-dev-middleware "^5.3.1" + ws "^8.13.0" + +webpack-merge@^5.7.3: + version "5.9.0" + resolved "https://registry.yarnpkg.com/webpack-merge/-/webpack-merge-5.9.0.tgz#dc160a1c4cf512ceca515cc231669e9ddb133826" + integrity sha512-6NbRQw4+Sy50vYNTw7EyOn41OZItPiXB8GNv3INSoe3PSFaHJEz3SHTrYVaRm2LilNGnFUzh0FAwqPEmU/CwDg== + dependencies: + clone-deep "^4.0.1" + wildcard "^2.0.0" + +webpack-sources@^3.2.3: + version "3.2.3" + resolved "https://registry.yarnpkg.com/webpack-sources/-/webpack-sources-3.2.3.tgz#2d4daab8451fd4b240cc27055ff6a0c2ccea0cde" + integrity sha512-/DyMEOrDgLKKIG0fmvtz+4dUX/3Ghozwgm6iPp8KRhvn+eQf9+Q7GWxVNMk3+uCPWfdXYC4ExGBckIXdFEfH1w== + +webpack@^5.88.1: + version "5.88.1" + resolved "https://registry.yarnpkg.com/webpack/-/webpack-5.88.1.tgz#21eba01e81bd5edff1968aea726e2fbfd557d3f8" + integrity sha512-FROX3TxQnC/ox4N+3xQoWZzvGXSuscxR32rbzjpXgEzWudJFEJBpdlkkob2ylrv5yzzufD1zph1OoFsLtm6stQ== + dependencies: + "@types/eslint-scope" "^3.7.3" + "@types/estree" "^1.0.0" + "@webassemblyjs/ast" "^1.11.5" + "@webassemblyjs/wasm-edit" "^1.11.5" + "@webassemblyjs/wasm-parser" "^1.11.5" + acorn "^8.7.1" + acorn-import-assertions "^1.9.0" + browserslist "^4.14.5" + chrome-trace-event "^1.0.2" + enhanced-resolve "^5.15.0" + es-module-lexer "^1.2.1" + eslint-scope "5.1.1" + events "^3.2.0" + glob-to-regexp "^0.4.1" + graceful-fs "^4.2.9" + json-parse-even-better-errors "^2.3.1" + loader-runner "^4.2.0" + mime-types "^2.1.27" + neo-async "^2.6.2" + schema-utils "^3.2.0" + tapable "^2.1.1" + terser-webpack-plugin "^5.3.7" + watchpack "^2.4.0" + webpack-sources "^3.2.3" + +websocket-driver@>=0.5.1, websocket-driver@^0.7.4: + version "0.7.4" + resolved "https://registry.yarnpkg.com/websocket-driver/-/websocket-driver-0.7.4.tgz#89ad5295bbf64b480abcba31e4953aca706f5760" + integrity sha512-b17KeDIQVjvb0ssuSDF2cYXSg2iztliJ4B9WdsuB6J952qCPKmnVq4DyW5motImXHDC1cBT/1UezrJVsKw5zjg== + dependencies: + http-parser-js ">=0.5.1" + safe-buffer ">=5.1.0" + websocket-extensions ">=0.1.1" + +websocket-extensions@>=0.1.1: + version "0.1.4" + resolved "https://registry.yarnpkg.com/websocket-extensions/-/websocket-extensions-0.1.4.tgz#7f8473bc839dfd87608adb95d7eb075211578a42" + integrity sha512-OqedPIGOfsDlo31UNwYbCFMSaO9m9G/0faIHj5/dZFDMFqPTcx6UwqyOy3COEaEOg/9VsGIpdqn62W5KhoKSpg== + +which-boxed-primitive@^1.0.2: + version "1.0.2" + resolved "https://registry.yarnpkg.com/which-boxed-primitive/-/which-boxed-primitive-1.0.2.tgz#13757bc89b209b049fe5d86430e21cf40a89a8e6" + integrity sha512-bwZdv0AKLpplFY2KZRX6TvyuN7ojjr7lwkg6ml0roIy9YeuSr7JS372qlNW18UQYzgYK9ziGcerWqZOmEn9VNg== + dependencies: + is-bigint "^1.0.1" + is-boolean-object "^1.1.0" + is-number-object "^1.0.4" + is-string "^1.0.5" + is-symbol "^1.0.3" + +which-typed-array@^1.1.9: + version "1.1.9" + resolved "https://registry.yarnpkg.com/which-typed-array/-/which-typed-array-1.1.9.tgz#307cf898025848cf995e795e8423c7f337efbde6" + integrity sha512-w9c4xkx6mPidwp7180ckYWfMmvxpjlZuIudNtDf4N/tTAUB8VJbX25qZoAsrtGuYNnGw3pa0AXgbGKRB8/EceA== + dependencies: + available-typed-arrays "^1.0.5" + call-bind "^1.0.2" + for-each "^0.3.3" + gopd "^1.0.1" + has-tostringtag "^1.0.0" + is-typed-array "^1.1.10" + +which@^2.0.1: + version "2.0.2" + resolved "https://registry.yarnpkg.com/which/-/which-2.0.2.tgz#7c6a8dd0a636a0327e10b59c9286eee93f3f51b1" + integrity sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA== + dependencies: + isexe "^2.0.0" + +wildcard@^2.0.0: + version "2.0.1" + resolved "https://registry.yarnpkg.com/wildcard/-/wildcard-2.0.1.tgz#5ab10d02487198954836b6349f74fff961e10f67" + integrity sha512-CC1bOL87PIWSBhDcTrdeLo6eGT7mCFtrg0uIJtqJUFyK+eJnzl8A1niH56uu7KMa5XFrtiV+AQuHO3n7DsHnLQ== + +wrap-ansi@^6.2.0: + version "6.2.0" + resolved "https://registry.yarnpkg.com/wrap-ansi/-/wrap-ansi-6.2.0.tgz#e9393ba07102e6c91a3b221478f0257cd2856e53" + integrity sha512-r6lPcBGxZXlIcymEu7InxDMhdW0KDxpLgoFLcguasxCaJ/SOIZwINatK9KY/tf+ZrlywOKU0UDj3ATXUBfxJXA== + dependencies: + ansi-styles "^4.0.0" + string-width "^4.1.0" + strip-ansi "^6.0.0" + +wrap-ansi@^7.0.0: + version "7.0.0" + resolved "https://registry.yarnpkg.com/wrap-ansi/-/wrap-ansi-7.0.0.tgz#67e145cff510a6a6984bdf1152911d69d2eb9e43" + integrity sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q== + dependencies: + ansi-styles "^4.0.0" + string-width "^4.1.0" + strip-ansi "^6.0.0" + +wrappy@1: + version "1.0.2" + resolved "https://registry.yarnpkg.com/wrappy/-/wrappy-1.0.2.tgz#b5243d8f3ec1aa35f1364605bc0d1036e30ab69f" + integrity sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ== + +ws@^8.13.0: + version "8.13.0" + resolved "https://registry.yarnpkg.com/ws/-/ws-8.13.0.tgz#9a9fb92f93cf41512a0735c8f4dd09b8a1211cd0" + integrity sha512-x9vcZYTrFPC7aSIbj7sRCYo7L/Xb8Iy+pW0ng0wt2vCJv7M9HOMy0UoN3rr+IFC7hb7vXoqS+P9ktyLLLhO+LA== + +yallist@^3.0.2: + version "3.1.1" + resolved "https://registry.yarnpkg.com/yallist/-/yallist-3.1.1.tgz#dbb7daf9bfd8bac9ab45ebf602b8cbad0d5d08fd" + integrity sha512-a4UGQaWPH59mOXUYnAG2ewncQS4i4F43Tv3JoAM+s2VDAmS9NsK8GpDMLrCHPksFT7h3K6TOoUNn2pb7RoXx4g== + +yallist@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/yallist/-/yallist-4.0.0.tgz#9bb92790d9c0effec63be73519e11a35019a3a72" + integrity sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A== + +yaml@^2.2.2: + version "2.3.1" + resolved "https://registry.yarnpkg.com/yaml/-/yaml-2.3.1.tgz#02fe0975d23cd441242aa7204e09fc28ac2ac33b" + integrity sha512-2eHWfjaoXgTBC2jNM1LRef62VQa0umtvRiDSk6HSzW7RvS5YtkabJrwYLLEKWBc8a5U2PTSCs+dJjUTJdlHsWQ== + +yocto-queue@^0.1.0: + version "0.1.0" + resolved "https://registry.yarnpkg.com/yocto-queue/-/yocto-queue-0.1.0.tgz#0294eb3dee05028d31ee1a5fa2c556a6aaf10a1b" + integrity sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q== From efd3ed839717c2e254754560daa66762891d0a75 Mon Sep 17 00:00:00 2001 From: kutta97 Date: Sat, 1 Jul 2023 03:41:35 +0900 Subject: [PATCH 02/87] =?UTF-8?q?chore:=20alias=20=EC=84=A4=EC=A0=95?= =?UTF-8?q?=EC=9D=84=20=EC=9C=84=ED=95=B4=20eslint=20'import/no-unresolved?= =?UTF-8?q?'=20=EC=98=B5=EC=85=98=20off?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .eslintrc.js | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/.eslintrc.js b/.eslintrc.js index f11add17..e46890d2 100644 --- a/.eslintrc.js +++ b/.eslintrc.js @@ -9,5 +9,7 @@ module.exports = { sourceType: 'module', }, plugins: [], - rules: {}, + rules: { + 'import/no-unresolved': 'off', + }, }; From 7c9293a10730c6a1565dd7fb96ad5e2f058cb66e Mon Sep 17 00:00:00 2001 From: kutta97 Date: Sat, 1 Jul 2023 03:42:31 +0900 Subject: [PATCH 03/87] =?UTF-8?q?feat:=20style=20reset=20=EC=B6=94?= =?UTF-8?q?=EA=B0=80?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- src/main.js | 1 + src/styleReset.css | 129 +++++++++++++++++++++++++++++++++++++++++++++ 2 files changed, 130 insertions(+) create mode 100644 src/styleReset.css diff --git a/src/main.js b/src/main.js index dd63fdb4..03451956 100644 --- a/src/main.js +++ b/src/main.js @@ -1,3 +1,4 @@ +import './styleReset.css'; import App from './App'; const initApp = ($target) => new App({ $target }); diff --git a/src/styleReset.css b/src/styleReset.css new file mode 100644 index 00000000..a3f76817 --- /dev/null +++ b/src/styleReset.css @@ -0,0 +1,129 @@ +/* http://meyerweb.com/eric/tools/css/reset/ + v2.0 | 20110126 + License: none (public domain) +*/ + +html, +body, +div, +span, +applet, +object, +iframe, +h1, +h2, +h3, +h4, +h5, +h6, +p, +blockquote, +pre, +a, +abbr, +acronym, +address, +big, +cite, +code, +del, +dfn, +em, +img, +ins, +kbd, +q, +s, +samp, +small, +strike, +strong, +sub, +sup, +tt, +var, +b, +u, +i, +center, +dl, +dt, +dd, +ol, +ul, +li, +fieldset, +form, +label, +legend, +table, +caption, +tbody, +tfoot, +thead, +tr, +th, +td, +article, +aside, +canvas, +details, +embed, +figure, +figcaption, +footer, +header, +hgroup, +menu, +nav, +output, +ruby, +section, +summary, +time, +mark, +audio, +video { + margin: 0; + padding: 0; + border: 0; + font-size: 100%; + font: inherit; + vertical-align: baseline; +} +/* HTML5 display-role reset for older browsers */ +article, +aside, +details, +figcaption, +figure, +footer, +header, +hgroup, +menu, +nav, +section { + display: block; +} +body { + line-height: 1; +} +ol, +ul { + list-style: none; +} +blockquote, +q { + quotes: none; +} +blockquote:before, +blockquote:after, +q:before, +q:after { + content: ''; + content: none; +} +table { + border-collapse: collapse; + border-spacing: 0; +} From b9444f2712b7f4564608d6aa171aca23345b6367 Mon Sep 17 00:00:00 2001 From: kutta97 Date: Sat, 1 Jul 2023 03:43:04 +0900 Subject: [PATCH 04/87] =?UTF-8?q?feat:=20NotionSidebar=20=EC=BB=B4?= =?UTF-8?q?=ED=8F=AC=EB=84=8C=ED=8A=B8=20=EC=B6=94=EA=B0=80?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- src/components/NotionSidebar.css | 29 +++++++++++++++++++++++++++++ src/components/NotionSidebar.js | 27 +++++++++++++++++++++++++++ 2 files changed, 56 insertions(+) create mode 100644 src/components/NotionSidebar.css create mode 100644 src/components/NotionSidebar.js diff --git a/src/components/NotionSidebar.css b/src/components/NotionSidebar.css new file mode 100644 index 00000000..0c778d69 --- /dev/null +++ b/src/components/NotionSidebar.css @@ -0,0 +1,29 @@ +.notion-sidebar-container { + display: flex; + flex-direction: column; + width: 300px; + height: 100%; + border-right: 1px solid #f5f6f5; + + background-color: #fbfbfa; +} + +.notion-sidebar-container ul { + width: 100%; + height: 100%; + padding: 10px 5px; +} + +.notion-sidebar-container ul li { + width: 100%; + padding: 5px 10px; + + color: #19171199; + font-size: 14px; + font-weight: 600; + + &:hover { + background-color: #ededeb; + cursor: pointer; + } +} diff --git a/src/components/NotionSidebar.js b/src/components/NotionSidebar.js new file mode 100644 index 00000000..dce4089f --- /dev/null +++ b/src/components/NotionSidebar.js @@ -0,0 +1,27 @@ +import './NotionSidebar.css'; + +export default class NotionSidebar { + constructor({ $target }) { + const $sidebarContainer = document.createElement('nav'); + $sidebarContainer.className = 'notion-sidebar-container'; + $target.appendChild($sidebarContainer); + + this.$sidebar = document.createElement('ul'); + $sidebarContainer.appendChild(this.$sidebar); + } + + setState(nextState) { + this.state = nextState; + this.render(); + } + + render() { + this.$sidebar.innerHTML = this.state + .map( + (document) => ` +
  • ${document.title}
  • + ` + ) + .join(''); + } +} From 5d6c17019b845f7f8a7540d6e45de3efcbfb52d6 Mon Sep 17 00:00:00 2001 From: kutta97 Date: Sat, 1 Jul 2023 03:43:22 +0900 Subject: [PATCH 05/87] =?UTF-8?q?feat:=20api.js=20util=20=ED=95=A8?= =?UTF-8?q?=EC=88=98=20=EC=B6=94=EA=B0=80?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- src/utils/api.js | 27 +++++++++++++++++++++++++++ 1 file changed, 27 insertions(+) create mode 100644 src/utils/api.js diff --git a/src/utils/api.js b/src/utils/api.js new file mode 100644 index 00000000..53a027b0 --- /dev/null +++ b/src/utils/api.js @@ -0,0 +1,27 @@ +const API_END_POINT = 'https://kdt-frontend.programmers.co.kr'; + +const DEFAULT_OPTION = { + headers: { + 'Content-Type': 'application/json', + 'x-username': 'roto', + }, +}; + +const request = async (url, options = {}) => { + const data = await fetch(`${API_END_POINT}${url}`, { + ...options, + ...DEFAULT_OPTION, + }) + .then((response) => { + if (!response.ok) throw new Error('API request error'); + + return response.json(); + }) + .catch((e) => { + console.error(e); + }); + + return data; +}; + +export default request; From aff88e9be41f57767ceed89feaa48fdb416e0756 Mon Sep 17 00:00:00 2001 From: kutta97 Date: Sat, 1 Jul 2023 03:43:45 +0900 Subject: [PATCH 06/87] =?UTF-8?q?feat:=20NotionPage=20=ED=8E=98=EC=9D=B4?= =?UTF-8?q?=EC=A7=80=20=EC=BB=B4=ED=8F=AC=EB=84=8C=ED=8A=B8=20=EC=B6=94?= =?UTF-8?q?=EA=B0=80?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- src/pages/NotionPage.css | 5 +++++ src/pages/NotionPage.js | 26 ++++++++++++++++++++++++++ 2 files changed, 31 insertions(+) create mode 100644 src/pages/NotionPage.css create mode 100644 src/pages/NotionPage.js diff --git a/src/pages/NotionPage.css b/src/pages/NotionPage.css new file mode 100644 index 00000000..4e8d6471 --- /dev/null +++ b/src/pages/NotionPage.css @@ -0,0 +1,5 @@ +.notion-page { + width: 100%; + height: 100%; + display: flex; +} diff --git a/src/pages/NotionPage.js b/src/pages/NotionPage.js new file mode 100644 index 00000000..af654c39 --- /dev/null +++ b/src/pages/NotionPage.js @@ -0,0 +1,26 @@ +import './NotionPage.css'; + +import NotionSidebar from '@components/NotionSidebar'; +import request from '@utils/api'; + +export default class NotionPage { + constructor({ $target }) { + this.$target = $target; + this.$page = document.createElement('div'); + this.$page.className = 'notion-page'; + + this.$sidebar = new NotionSidebar({ $target: this.$page }); + + this.setState(); + } + + async setState() { + const documents = await request('/documents'); + this.$sidebar.setState(documents); + this.render(); + } + + render() { + this.$target.appendChild(this.$page); + } +} From 81518af3ac790cc90f1bcc26df7bfa1bf5106e94 Mon Sep 17 00:00:00 2001 From: kutta97 Date: Sat, 1 Jul 2023 03:44:15 +0900 Subject: [PATCH 07/87] =?UTF-8?q?feat:=20App.js=20=EC=BB=B4=ED=8F=AC?= =?UTF-8?q?=EB=84=8C=ED=8A=B8=EC=97=90=20NotionPage=20=EC=BB=B4=ED=8F=AC?= =?UTF-8?q?=EB=84=8C=ED=8A=B8=20=EC=B6=94=EA=B0=80?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- src/App.css | 8 ++++++++ src/App.js | 5 ++--- 2 files changed, 10 insertions(+), 3 deletions(-) diff --git a/src/App.css b/src/App.css index 460f268a..af715397 100644 --- a/src/App.css +++ b/src/App.css @@ -1,7 +1,15 @@ * { box-sizing: border-box; } +html, body { + width: 100%; + height: 100%; +} +#app { + width: 100%; + height: 100%; + background-color: #fff; color: #37352f; font-family: 'Pretendard'; diff --git a/src/App.js b/src/App.js index a4490565..bab05276 100644 --- a/src/App.js +++ b/src/App.js @@ -1,9 +1,8 @@ +import NotionPage from '@pages/NotionPage'; import './App.css'; export default class App { constructor({ $target }) { - const $title = document.createElement('h1'); - $title.textContent = 'Notion with Vanilla JS'; - $target.appendChild($title); + this.$notionPage = new NotionPage({ $target }); } } From 3340684e1c0a520f99e2c49c472a5e4e72f38a99 Mon Sep 17 00:00:00 2001 From: kutta97 Date: Sun, 2 Jul 2023 00:21:07 +0900 Subject: [PATCH 08/87] =?UTF-8?q?chore:=20webpack-dev-server=20=EC=9D=98?= =?UTF-8?q?=20historyApiFallback=20=EC=84=A4=EC=A0=95=20=EC=B6=94=EA=B0=80?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- webpack.config.js | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/webpack.config.js b/webpack.config.js index 6e03f094..a97d5dd5 100644 --- a/webpack.config.js +++ b/webpack.config.js @@ -19,6 +19,7 @@ module.exports = (_, argv) => { output: { path: path.resolve(__dirname, './dist'), + publicPath: '/', filename: 'index.js', clean: true, }, @@ -26,6 +27,9 @@ module.exports = (_, argv) => { devServer: { port: 3000, hot: true, + historyApiFallback: { + index: '/index.html', + }, }, devtool: isDevelopment ? 'eval-source-map' : 'source-map', From a77667aa24eb6b96b078b61659ccc6aa46a1f9b6 Mon Sep 17 00:00:00 2001 From: kutta97 Date: Sun, 2 Jul 2023 00:47:42 +0900 Subject: [PATCH 09/87] =?UTF-8?q?chore:=20@trivago/prettier-plugin-sort-im?= =?UTF-8?q?ports=20Prettier=20=ED=94=8C=EB=9F=AC=EA=B7=B8=EC=9D=B8=20?= =?UTF-8?q?=EC=B6=94=EA=B0=80?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .prettierrc.js | 10 +++++++ package.json | 1 + yarn.lock | 73 +++++++++++++++++++++++++++++++++++++++++++------- 3 files changed, 75 insertions(+), 9 deletions(-) diff --git a/.prettierrc.js b/.prettierrc.js index f1284c64..7786ed3b 100644 --- a/.prettierrc.js +++ b/.prettierrc.js @@ -3,4 +3,14 @@ module.exports = { tabWidth: 2, semi: true, singleQuote: true, + + importOrder: [ + '^@consts/(.*)$', + '^@utils/(.*)$', + '^@components/(.*)$', + '^@pages/(.*)$', + '^[./]', + ], + importOrderSeparation: true, + importOrderSortSpecifiers: true, }; diff --git a/package.json b/package.json index bc696a25..c287e15e 100644 --- a/package.json +++ b/package.json @@ -19,6 +19,7 @@ "devDependencies": { "@babel/core": "^7.22.5", "@babel/preset-env": "^7.22.5", + "@trivago/prettier-plugin-sort-imports": "^4.1.1", "babel-loader": "^9.1.2", "babel-plugin-module-resolver": "^5.0.0", "css-loader": "^6.8.1", diff --git a/yarn.lock b/yarn.lock index 79253558..17d61303 100644 --- a/yarn.lock +++ b/yarn.lock @@ -15,7 +15,7 @@ "@jridgewell/gen-mapping" "^0.3.0" "@jridgewell/trace-mapping" "^0.3.9" -"@babel/code-frame@^7.22.5": +"@babel/code-frame@^7.16.7", "@babel/code-frame@^7.22.5": version "7.22.5" resolved "https://registry.yarnpkg.com/@babel/code-frame/-/code-frame-7.22.5.tgz#234d98e1551960604f1246e6475891a570ad5658" integrity sha512-Xmwn266vad+6DAqEB2A6V/CcZVp62BbwVmcOJc2RPuwih1kw02TjQvWVWlcKGbBPd+8/0V5DEkOcizRGYsspYQ== @@ -48,7 +48,16 @@ json5 "^2.2.2" semver "^6.3.0" -"@babel/generator@^7.22.5": +"@babel/generator@7.17.7": + version "7.17.7" + resolved "https://registry.yarnpkg.com/@babel/generator/-/generator-7.17.7.tgz#8da2599beb4a86194a3b24df6c085931d9ee45ad" + integrity sha512-oLcVCTeIFadUoArDTwpluncplrYBmTCCZZgXCbgNGvOBBiSDDK3eWO4b/+eOTli5tKv1lg+a5/NAXg+nTcei1w== + dependencies: + "@babel/types" "^7.17.0" + jsesc "^2.5.1" + source-map "^0.5.0" + +"@babel/generator@^7.17.3", "@babel/generator@^7.22.5": version "7.22.5" resolved "https://registry.yarnpkg.com/@babel/generator/-/generator-7.22.5.tgz#1e7bf768688acfb05cf30b2369ef855e82d984f7" integrity sha512-+lcUbnTRhd0jOewtFSedLyiPsD5tswKkbgcezOqqWFUVNEwoUTlpPOBmvhG7OXWLR4jMdv0czPGH5XbflnD1EA== @@ -119,12 +128,12 @@ resolve "^1.14.2" semver "^6.1.2" -"@babel/helper-environment-visitor@^7.22.5": +"@babel/helper-environment-visitor@^7.16.7", "@babel/helper-environment-visitor@^7.22.5": version "7.22.5" resolved "https://registry.yarnpkg.com/@babel/helper-environment-visitor/-/helper-environment-visitor-7.22.5.tgz#f06dd41b7c1f44e1f8da6c4055b41ab3a09a7e98" integrity sha512-XGmhECfVA/5sAt+H+xpSg0mfrHq6FzNr9Oxh7PSEBBRUb/mL7Kz3NICXb194rCqAEdxkhPT1a88teizAFyvk8Q== -"@babel/helper-function-name@^7.22.5": +"@babel/helper-function-name@^7.16.7", "@babel/helper-function-name@^7.22.5": version "7.22.5" resolved "https://registry.yarnpkg.com/@babel/helper-function-name/-/helper-function-name-7.22.5.tgz#ede300828905bb15e582c037162f99d5183af1be" integrity sha512-wtHSq6jMRE3uF2otvfuD3DIvVhOsSNshQl0Qrd7qC9oQJzHvOL4qQXlQn2916+CXGywIjpGuIkoyZRRxHPiNQQ== @@ -132,7 +141,7 @@ "@babel/template" "^7.22.5" "@babel/types" "^7.22.5" -"@babel/helper-hoist-variables@^7.22.5": +"@babel/helper-hoist-variables@^7.16.7", "@babel/helper-hoist-variables@^7.22.5": version "7.22.5" resolved "https://registry.yarnpkg.com/@babel/helper-hoist-variables/-/helper-hoist-variables-7.22.5.tgz#c01a007dac05c085914e8fb652b339db50d823bb" integrity sha512-wGjk9QZVzvknA6yKIUURb8zY3grXCcOZt+/7Wcy8O2uctxhplmUPkOdlgoNhmdVee2c92JXbf1xpMtVNbfoxRw== @@ -215,7 +224,7 @@ dependencies: "@babel/types" "^7.22.5" -"@babel/helper-split-export-declaration@^7.22.5": +"@babel/helper-split-export-declaration@^7.16.7", "@babel/helper-split-export-declaration@^7.22.5": version "7.22.5" resolved "https://registry.yarnpkg.com/@babel/helper-split-export-declaration/-/helper-split-export-declaration-7.22.5.tgz#88cf11050edb95ed08d596f7a044462189127a08" integrity sha512-thqK5QFghPKWLhAV321lxF95yCg2K3Ob5yw+M3VHWfdia0IkPXUtoLH8x/6Fh486QUvzhb8YOWHChTVen2/PoQ== @@ -227,7 +236,7 @@ resolved "https://registry.yarnpkg.com/@babel/helper-string-parser/-/helper-string-parser-7.22.5.tgz#533f36457a25814cf1df6488523ad547d784a99f" integrity sha512-mM4COjgZox8U+JcXQwPijIZLElkgEpO5rsERVDJTc2qfCDfERyob6k5WegS14SX18IIjv+XD+GrqNumY5JRCDw== -"@babel/helper-validator-identifier@^7.22.5": +"@babel/helper-validator-identifier@^7.16.7", "@babel/helper-validator-identifier@^7.22.5": version "7.22.5" resolved "https://registry.yarnpkg.com/@babel/helper-validator-identifier/-/helper-validator-identifier-7.22.5.tgz#9544ef6a33999343c8740fa51350f30eeaaaf193" integrity sha512-aJXu+6lErq8ltp+JhkJUfk1MTGyuA4v7f3pA+BJ5HLfNC6nAQ0Cpi9uOquUj8Hehg0aUiHzWQbOVJGao6ztBAQ== @@ -265,7 +274,7 @@ chalk "^2.0.0" js-tokens "^4.0.0" -"@babel/parser@^7.22.5": +"@babel/parser@^7.17.3", "@babel/parser@^7.20.5", "@babel/parser@^7.22.5": version "7.22.5" resolved "https://registry.yarnpkg.com/@babel/parser/-/parser-7.22.5.tgz#721fd042f3ce1896238cf1b341c77eb7dee7dbea" integrity sha512-DFZMC9LJUG9PLOclRC32G63UXwzqS2koQC8dkx+PLdmt1xSePYpbT/NbsrJy8Q/muXz7o/h/d4A7Fuyixm559Q== @@ -932,6 +941,22 @@ "@babel/parser" "^7.22.5" "@babel/types" "^7.22.5" +"@babel/traverse@7.17.3": + version "7.17.3" + resolved "https://registry.yarnpkg.com/@babel/traverse/-/traverse-7.17.3.tgz#0ae0f15b27d9a92ba1f2263358ea7c4e7db47b57" + integrity sha512-5irClVky7TxRWIRtxlh2WPUUOLhcPN06AGgaQSB8AEwuyEBgJVuJ5imdHm5zxk8w0QS5T+tDfnDxAlhWjpb7cw== + dependencies: + "@babel/code-frame" "^7.16.7" + "@babel/generator" "^7.17.3" + "@babel/helper-environment-visitor" "^7.16.7" + "@babel/helper-function-name" "^7.16.7" + "@babel/helper-hoist-variables" "^7.16.7" + "@babel/helper-split-export-declaration" "^7.16.7" + "@babel/parser" "^7.17.3" + "@babel/types" "^7.17.0" + debug "^4.1.0" + globals "^11.1.0" + "@babel/traverse@^7.22.5": version "7.22.5" resolved "https://registry.yarnpkg.com/@babel/traverse/-/traverse-7.22.5.tgz#44bd276690db6f4940fdb84e1cb4abd2f729ccd1" @@ -948,7 +973,15 @@ debug "^4.1.0" globals "^11.1.0" -"@babel/types@^7.22.5", "@babel/types@^7.4.4": +"@babel/types@7.17.0": + version "7.17.0" + resolved "https://registry.yarnpkg.com/@babel/types/-/types-7.17.0.tgz#a826e368bccb6b3d84acd76acad5c0d87342390b" + integrity sha512-TmKSNO4D5rzhL5bjWFcVHHLETzfQ/AmbKpKPOSjlP0WoHZ6L911fgoOKY4Alp/emzG4cHJdyN49zpgkbXFEHHw== + dependencies: + "@babel/helper-validator-identifier" "^7.16.7" + to-fast-properties "^2.0.0" + +"@babel/types@^7.17.0", "@babel/types@^7.22.5", "@babel/types@^7.4.4": version "7.22.5" resolved "https://registry.yarnpkg.com/@babel/types/-/types-7.22.5.tgz#cd93eeaab025880a3a47ec881f4b096a5b786fbe" integrity sha512-zo3MIHGOkPOfoRXitsgHLjEXmlDaD/5KU1Uzuc9GNiZPhSqVxVRtxuPaSBZDsYZ9qV88AjtMtWW7ww98loJ9KA== @@ -1081,6 +1114,18 @@ "@nodelib/fs.scandir" "2.1.5" fastq "^1.6.0" +"@trivago/prettier-plugin-sort-imports@^4.1.1": + version "4.1.1" + resolved "https://registry.yarnpkg.com/@trivago/prettier-plugin-sort-imports/-/prettier-plugin-sort-imports-4.1.1.tgz#71c3c1ae770c3738b6fc85710714844477574ffd" + integrity sha512-dQ2r2uzNr1x6pJsuh/8x0IRA3CBUB+pWEW3J/7N98axqt7SQSm+2fy0FLNXvXGg77xEDC7KHxJlHfLYyi7PDcw== + dependencies: + "@babel/generator" "7.17.7" + "@babel/parser" "^7.20.5" + "@babel/traverse" "7.17.3" + "@babel/types" "7.17.0" + javascript-natural-sort "0.7.1" + lodash "^4.17.21" + "@types/body-parser@*": version "1.19.2" resolved "https://registry.yarnpkg.com/@types/body-parser/-/body-parser-1.19.2.tgz#aea2059e28b7658639081347ac4fab3de166e6f0" @@ -3229,6 +3274,11 @@ isobject@^3.0.1: resolved "https://registry.yarnpkg.com/isobject/-/isobject-3.0.1.tgz#4e431e92b11a9731636aa1f9c8d1ccbcfdab78df" integrity sha512-WhB9zCku7EGTj/HQQRz5aUQEUeoQZH2bWcltRErOpymJ4boYE6wL9Tbr23krRPSZ+C5zqNSrSw+Cc7sZZ4b7vg== +javascript-natural-sort@0.7.1: + version "0.7.1" + resolved "https://registry.yarnpkg.com/javascript-natural-sort/-/javascript-natural-sort-0.7.1.tgz#f9e2303d4507f6d74355a73664d1440fb5a0ef59" + integrity sha512-nO6jcEfZWQXDhOiBtG2KvKyEptz7RVbpGP4vTD2hLBdmNQSsCiicO2Ioinv6UI4y9ukqnBpy+XZ9H6uLNgJTlw== + jest-worker@^27.4.5: version "27.5.1" resolved "https://registry.yarnpkg.com/jest-worker/-/jest-worker-27.5.1.tgz#8d146f0900e8973b106b6f73cc1e9a8cb86f8db0" @@ -4369,6 +4419,11 @@ source-map-support@~0.5.20: buffer-from "^1.0.0" source-map "^0.6.0" +source-map@^0.5.0: + version "0.5.7" + resolved "https://registry.yarnpkg.com/source-map/-/source-map-0.5.7.tgz#8a039d2d1021d22d1ea14c80d8ea468ba2ef3fcc" + integrity sha512-LbrmJOMUSdEVxIKvdcJzQC+nQhe8FUZQTXQy6+I75skNgn3OoQ0DZA8YnFa7gp8tqtL3KPf1kmo0R5DoApeSGQ== + source-map@^0.6.0, source-map@~0.6.0: version "0.6.1" resolved "https://registry.yarnpkg.com/source-map/-/source-map-0.6.1.tgz#74722af32e9614e9c287a8d0bbde48b5e2f1a263" From 4bd6e4f966e86472095dcabe377d4fef3e42c234 Mon Sep 17 00:00:00 2001 From: kutta97 Date: Sun, 2 Jul 2023 01:02:56 +0900 Subject: [PATCH 10/87] =?UTF-8?q?feat:=20App.js=20=EC=97=90=20Route=20?= =?UTF-8?q?=EC=B2=98=EB=A6=AC=20=EA=B8=B0=EB=8A=A5=20=EC=B6=94=EA=B0=80?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- src/App.js | 22 ++++++++++++++++++++++ src/main.js | 2 +- src/utils/pathname.js | 6 ++++++ src/utils/router.js | 32 ++++++++++++++++++++++++++++++++ 4 files changed, 61 insertions(+), 1 deletion(-) create mode 100644 src/utils/pathname.js create mode 100644 src/utils/router.js diff --git a/src/App.js b/src/App.js index bab05276..7696c6c4 100644 --- a/src/App.js +++ b/src/App.js @@ -1,8 +1,30 @@ +import Pathname from '@utils/pathname'; +import { initRouter } from '@utils/router'; + import NotionPage from '@pages/NotionPage'; + import './App.css'; export default class App { constructor({ $target }) { this.$notionPage = new NotionPage({ $target }); + + initRouter(this.route.bind(this)); + this.route(); + } + + route() { + const { pathname } = window.location; + + if (Pathname.isRoot(pathname)) { + return this.$notionPage.setState({ documentId: 'empty' }); + } + + if (Pathname.isDocument(pathname)) { + const [, , documentId] = pathname.split('/'); + return this.$notionPage.setState({ documentId }); + } + + return null; } } diff --git a/src/main.js b/src/main.js index 03451956..1c7db75c 100644 --- a/src/main.js +++ b/src/main.js @@ -1,5 +1,5 @@ -import './styleReset.css'; import App from './App'; +import './styleReset.css'; const initApp = ($target) => new App({ $target }); diff --git a/src/utils/pathname.js b/src/utils/pathname.js new file mode 100644 index 00000000..f5557753 --- /dev/null +++ b/src/utils/pathname.js @@ -0,0 +1,6 @@ +const Pathname = { + isRoot: (pathname) => pathname === '/', + isDocument: (pathname) => /\/documents\/\d/.test(pathname), +}; + +export default Pathname; diff --git a/src/utils/router.js b/src/utils/router.js new file mode 100644 index 00000000..6f817c43 --- /dev/null +++ b/src/utils/router.js @@ -0,0 +1,32 @@ +const POPSTATE_EVENT = 'popstate'; +const HISTORY_PUSH_EVENT = 'history-push'; + +const initPopstateEventListener = (onRoute) => { + window.addEventListener(POPSTATE_EVENT, () => { + onRoute(); + }); +}; + +const initHistoryPushEventListener = (onRoute) => { + window.addEventListener(HISTORY_PUSH_EVENT, (e) => { + const { nextUrl } = e.detail; + + if (!nextUrl) return; + + window.history.pushState(null, null, nextUrl); + onRoute(); + }); +}; + +export const initRouter = (onRoute) => { + initPopstateEventListener(onRoute); + initHistoryPushEventListener(onRoute); +}; + +export const push = (nextUrl) => { + window.dispatchEvent( + new CustomEvent(HISTORY_PUSH_EVENT, { + detail: { nextUrl }, + }) + ); +}; From ad85a515dfd3173bc20230eb82a41effeb05a30e Mon Sep 17 00:00:00 2001 From: kutta97 Date: Sun, 2 Jul 2023 01:04:08 +0900 Subject: [PATCH 11/87] =?UTF-8?q?feat:=20NotionSidebar=20click=20event=20?= =?UTF-8?q?=EC=B6=94=EA=B0=80?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- src/components/{ => NotionSidebar}/NotionSidebar.css | 0 src/components/{ => NotionSidebar}/NotionSidebar.js | 11 +++++++++++ 2 files changed, 11 insertions(+) rename src/components/{ => NotionSidebar}/NotionSidebar.css (100%) rename src/components/{ => NotionSidebar}/NotionSidebar.js (72%) diff --git a/src/components/NotionSidebar.css b/src/components/NotionSidebar/NotionSidebar.css similarity index 100% rename from src/components/NotionSidebar.css rename to src/components/NotionSidebar/NotionSidebar.css diff --git a/src/components/NotionSidebar.js b/src/components/NotionSidebar/NotionSidebar.js similarity index 72% rename from src/components/NotionSidebar.js rename to src/components/NotionSidebar/NotionSidebar.js index dce4089f..473eaa5e 100644 --- a/src/components/NotionSidebar.js +++ b/src/components/NotionSidebar/NotionSidebar.js @@ -1,3 +1,5 @@ +import { push } from '@utils/router'; + import './NotionSidebar.css'; export default class NotionSidebar { @@ -8,6 +10,15 @@ export default class NotionSidebar { this.$sidebar = document.createElement('ul'); $sidebarContainer.appendChild(this.$sidebar); + + this.$sidebar.addEventListener('click', (e) => { + const $li = e.target.closest('li'); + + if ($li) { + const { id } = $li.dataset; + push(`/documents/${id}`); + } + }); } setState(nextState) { From fd78d48a0cbccec96a8c83536b3fab1b99764b62 Mon Sep 17 00:00:00 2001 From: kutta97 Date: Sun, 2 Jul 2023 01:04:56 +0900 Subject: [PATCH 12/87] =?UTF-8?q?feat:=20NotionEditor=20=EC=B6=94=EA=B0=80?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../Editor/Content/NotionEditorContent.css | 0 .../Editor/Content/NotionEditorContent.js | 29 +++++++++++++++++++ src/components/Editor/NotionEditor.css | 20 +++++++++++++ src/components/Editor/NotionEditor.js | 23 +++++++++++++++ .../Editor/Title/NotionEditorTitle.css | 0 .../Editor/Title/NotionEditorTitle.js | 22 ++++++++++++++ 6 files changed, 94 insertions(+) create mode 100644 src/components/Editor/Content/NotionEditorContent.css create mode 100644 src/components/Editor/Content/NotionEditorContent.js create mode 100644 src/components/Editor/NotionEditor.css create mode 100644 src/components/Editor/NotionEditor.js create mode 100644 src/components/Editor/Title/NotionEditorTitle.css create mode 100644 src/components/Editor/Title/NotionEditorTitle.js diff --git a/src/components/Editor/Content/NotionEditorContent.css b/src/components/Editor/Content/NotionEditorContent.css new file mode 100644 index 00000000..e69de29b diff --git a/src/components/Editor/Content/NotionEditorContent.js b/src/components/Editor/Content/NotionEditorContent.js new file mode 100644 index 00000000..4961a3bc --- /dev/null +++ b/src/components/Editor/Content/NotionEditorContent.js @@ -0,0 +1,29 @@ +import './NotionEditorContent.css'; + +export default class NotionEditorContent { + constructor({ $target }) { + this.$content = document.createElement('div'); + + this.$content.name = 'content'; + this.$content.contentEditable = true; + + $target.appendChild(this.$content); + } + + setState(nextState) { + this.state = nextState; + this.render(); + } + + render() { + const { content } = this.state; + + const renderedContent = + content + ?.split('\n') + ?.map((line) => line) + ?.join('
    ') ?? ''; + + this.$content.innerHTML = renderedContent; + } +} diff --git a/src/components/Editor/NotionEditor.css b/src/components/Editor/NotionEditor.css new file mode 100644 index 00000000..bd191ac4 --- /dev/null +++ b/src/components/Editor/NotionEditor.css @@ -0,0 +1,20 @@ +.notion-editor { + display: flex; + flex-direction: column; + width: 100%; + height: 100%; + padding: 20px 10px; +} + +.notion-editor input { + font-size: 20px; +} + +.notion-editor div { + display: flex; + flex-direction: column; + width: 100%; + height: 100%; + + padding: 20px; +} diff --git a/src/components/Editor/NotionEditor.js b/src/components/Editor/NotionEditor.js new file mode 100644 index 00000000..1d9cfc4d --- /dev/null +++ b/src/components/Editor/NotionEditor.js @@ -0,0 +1,23 @@ +import NotionEditorTitle from '@components//Editor/Title/NotionEditorTitle'; +import NotionEditorContent from '@components/Editor/Content/NotionEditorContent'; + +import './NotionEditor.css'; + +export default class NotionEditor { + constructor({ $target }) { + this.$editor = document.createElement('section'); + this.$editor.className = 'notion-editor'; + $target.appendChild(this.$editor); + + this.$title = new NotionEditorTitle({ $target: this.$editor }); + this.$content = new NotionEditorContent({ $target: this.$editor }); + } + + setState(nextState) { + this.state = nextState; + + const { title, content } = this.state; + this.$title.setState({ title }); + this.$content.setState({ content }); + } +} diff --git a/src/components/Editor/Title/NotionEditorTitle.css b/src/components/Editor/Title/NotionEditorTitle.css new file mode 100644 index 00000000..e69de29b diff --git a/src/components/Editor/Title/NotionEditorTitle.js b/src/components/Editor/Title/NotionEditorTitle.js new file mode 100644 index 00000000..5a9fbab4 --- /dev/null +++ b/src/components/Editor/Title/NotionEditorTitle.js @@ -0,0 +1,22 @@ +import './NotionEditorTitle.css'; + +export default class NotionEditorTitle { + constructor({ $target }) { + this.$title = document.createElement('input'); + + this.$title.name = 'title'; + this.$title.placeholder = 'undefined'; + + $target.appendChild(this.$title); + } + + setState(nextState) { + this.state = nextState; + this.render(); + } + + render() { + const { title } = this.state; + this.$title.value = title ?? ''; + } +} From a3a2398f3dfb9e9b65a347e90d71c57617b38694 Mon Sep 17 00:00:00 2001 From: kutta97 Date: Sun, 2 Jul 2023 01:05:17 +0900 Subject: [PATCH 13/87] =?UTF-8?q?feat:=20NotionPage=20=EC=97=90=20NotionDo?= =?UTF-8?q?cument=20=EC=B6=94=EA=B0=80?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../NotionDocument/NotionDocument.css | 0 .../NotionDocument/NotionDocument.js | 16 ++++++++++++ src/pages/NotionPage.js | 26 +++++++++++-------- 3 files changed, 31 insertions(+), 11 deletions(-) create mode 100644 src/components/NotionDocument/NotionDocument.css create mode 100644 src/components/NotionDocument/NotionDocument.js diff --git a/src/components/NotionDocument/NotionDocument.css b/src/components/NotionDocument/NotionDocument.css new file mode 100644 index 00000000..e69de29b diff --git a/src/components/NotionDocument/NotionDocument.js b/src/components/NotionDocument/NotionDocument.js new file mode 100644 index 00000000..0d78f021 --- /dev/null +++ b/src/components/NotionDocument/NotionDocument.js @@ -0,0 +1,16 @@ +import NotionEditor from '@components/Editor/NotionEditor'; + +import './NotionDocument.css'; + +export default class NotionDocument { + constructor({ $target }) { + this.$editor = new NotionEditor({ $target }); + } + + setState(nextState) { + this.state = nextState; + + const { title, content } = this.state; + this.$editor.setState({ title, content }); + } +} diff --git a/src/pages/NotionPage.js b/src/pages/NotionPage.js index af654c39..02ca34bd 100644 --- a/src/pages/NotionPage.js +++ b/src/pages/NotionPage.js @@ -1,26 +1,30 @@ -import './NotionPage.css'; - -import NotionSidebar from '@components/NotionSidebar'; import request from '@utils/api'; +import NotionDocument from '@components/NotionDocument/NotionDocument'; +import NotionSidebar from '@components/NotionSidebar/NotionSidebar'; + +import './NotionPage.css'; + export default class NotionPage { constructor({ $target }) { - this.$target = $target; this.$page = document.createElement('div'); this.$page.className = 'notion-page'; + $target.appendChild(this.$page); this.$sidebar = new NotionSidebar({ $target: this.$page }); - - this.setState(); + this.$document = new NotionDocument({ $target: this.$page }); } - async setState() { + async setState(nextState) { + this.state = nextState; + const documents = await request('/documents'); this.$sidebar.setState(documents); - this.render(); - } - render() { - this.$target.appendChild(this.$page); + const { documentId } = this.state; + if (documentId !== 'empty') { + const document = await request(`/documents/${documentId}`); + this.$document.setState(document); + } } } From bf25a26f77872d758e71e58f67a3b61ccaa22557 Mon Sep 17 00:00:00 2001 From: kutta97 Date: Sun, 2 Jul 2023 01:15:22 +0900 Subject: [PATCH 14/87] =?UTF-8?q?feat:=20root=20url=20=EC=A0=91=EC=86=8D?= =?UTF-8?q?=20=EC=8B=9C,=20Empty=20Editor=20=EB=A5=BC=20=EB=B3=B4=EC=97=AC?= =?UTF-8?q?=EC=A3=BC=EB=8F=84=EB=A1=9D=20=EC=88=98=EC=A0=95?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- src/pages/NotionPage.js | 17 +++++++++++++---- 1 file changed, 13 insertions(+), 4 deletions(-) diff --git a/src/pages/NotionPage.js b/src/pages/NotionPage.js index 02ca34bd..062693a2 100644 --- a/src/pages/NotionPage.js +++ b/src/pages/NotionPage.js @@ -21,10 +21,19 @@ export default class NotionPage { const documents = await request('/documents'); this.$sidebar.setState(documents); + const document = await this.fetchDocument(); + this.$document.setState(document); + } + + async fetchDocument() { const { documentId } = this.state; - if (documentId !== 'empty') { - const document = await request(`/documents/${documentId}`); - this.$document.setState(document); - } + + if (documentId === 'empty') + return { + title: '', + content: '', + }; + const document = await request(`/documents/${documentId}`); + return document; } } From d9a90c48959e9912a97b1ce339694f909922f9c3 Mon Sep 17 00:00:00 2001 From: kutta97 Date: Sun, 2 Jul 2023 02:24:52 +0900 Subject: [PATCH 15/87] =?UTF-8?q?feat:=20localStorage=20=EC=B6=94=EA=B0=80?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- src/utils/localStorage.js | 26 ++++++++++++++++++++++++++ 1 file changed, 26 insertions(+) create mode 100644 src/utils/localStorage.js diff --git a/src/utils/localStorage.js b/src/utils/localStorage.js new file mode 100644 index 00000000..32bf8c51 --- /dev/null +++ b/src/utils/localStorage.js @@ -0,0 +1,26 @@ +const storage = window.localStorage; + +export const setItem = (key, value) => { + try { + storage.setItem(key, JSON.stringify(value)); + } catch (e) { + console.error(e); + } +}; + +export const getItem = (key, defaultValue) => { + try { + const storedValue = storage.getItem(key); + return storedValue ? JSON.parse(storedValue) : defaultValue; + } catch (e) { + console.error(e); + return defaultValue; + } +}; +export const removeItem = (key) => { + try { + storage.removeItem(key); + } catch (e) { + console.error(e); + } +}; From 07c40b75fd698351230ee3dffcadbfb074c5e96b Mon Sep 17 00:00:00 2001 From: kutta97 Date: Sun, 2 Jul 2023 02:25:47 +0900 Subject: [PATCH 16/87] =?UTF-8?q?feat:=20NotionDocument=20=EC=97=90=20onEd?= =?UTF-8?q?it=20=EC=9D=B4=EB=B2=A4=ED=8A=B8=20=ED=95=B8=EB=93=A4=EB=9F=AC?= =?UTF-8?q?=20=EC=B6=94=EA=B0=80=20=EB=B0=8F=20=EB=94=94=EB=B0=94=EC=9A=B4?= =?UTF-8?q?=EC=8A=A4=20=EC=A0=81=EC=9A=A9?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- src/components/Editor/NotionEditor.js | 21 ++++++++++++++- .../NotionDocument/NotionDocument.js | 27 ++++++++++++++++++- 2 files changed, 46 insertions(+), 2 deletions(-) diff --git a/src/components/Editor/NotionEditor.js b/src/components/Editor/NotionEditor.js index 1d9cfc4d..6cfb6c6b 100644 --- a/src/components/Editor/NotionEditor.js +++ b/src/components/Editor/NotionEditor.js @@ -4,13 +4,32 @@ import NotionEditorContent from '@components/Editor/Content/NotionEditorContent' import './NotionEditor.css'; export default class NotionEditor { - constructor({ $target }) { + constructor({ $target, onEdit }) { this.$editor = document.createElement('section'); this.$editor.className = 'notion-editor'; $target.appendChild(this.$editor); this.$title = new NotionEditorTitle({ $target: this.$editor }); this.$content = new NotionEditorContent({ $target: this.$editor }); + + this.initEventListener(onEdit); + } + + getNextStateWithTarget(target) { + const name = target.getAttribute('name'); + const value = name === 'title' ? target.value : target.innerHTML; + + return { + ...this.state, + [name]: value, + }; + } + + initEventListener(callback) { + this.$editor.addEventListener('input', (e) => { + const nextState = this.getNextStateWithTarget(e.target); + callback?.(nextState); + }); } setState(nextState) { diff --git a/src/components/NotionDocument/NotionDocument.js b/src/components/NotionDocument/NotionDocument.js index 0d78f021..cb73c176 100644 --- a/src/components/NotionDocument/NotionDocument.js +++ b/src/components/NotionDocument/NotionDocument.js @@ -1,10 +1,35 @@ +import { setItem } from '@utils/localStorage'; + import NotionEditor from '@components/Editor/NotionEditor'; import './NotionDocument.css'; export default class NotionDocument { constructor({ $target }) { - this.$editor = new NotionEditor({ $target }); + this.$editor = new NotionEditor({ + $target, + onEdit: this.onEdit.bind(this), + }); + + this.timer = null; + } + + onEdit(document) { + if (this.timer !== null) { + clearTimeout(this.timer); + } + this.timer = setTimeout(async () => { + setItem(`temp-post-${document.id}`, { + ...document, + tempSaveDate: new Date(), + }); + }, 1000); + + const nextState = { + ...this.state, + ...document, + }; + this.setState(nextState); } setState(nextState) { From 654a0ab199889e845795f018af5f81f05f772d1f Mon Sep 17 00:00:00 2001 From: kutta97 Date: Sun, 2 Jul 2023 03:23:23 +0900 Subject: [PATCH 17/87] =?UTF-8?q?feat:=20HTTP=20=EA=B0=81=20Method=20reque?= =?UTF-8?q?st=20=ED=95=A8=EC=88=98=ED=99=94?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- src/utils/api.js | 45 +++++++++++++++++++++++++++++++++++++++------ 1 file changed, 39 insertions(+), 6 deletions(-) diff --git a/src/utils/api.js b/src/utils/api.js index 53a027b0..4b5b56c5 100644 --- a/src/utils/api.js +++ b/src/utils/api.js @@ -1,16 +1,16 @@ const API_END_POINT = 'https://kdt-frontend.programmers.co.kr'; -const DEFAULT_OPTION = { - headers: { +const getHeader = () => { + const username = 'roto'; + return { 'Content-Type': 'application/json', - 'x-username': 'roto', - }, + 'x-username': username, + }; }; const request = async (url, options = {}) => { const data = await fetch(`${API_END_POINT}${url}`, { ...options, - ...DEFAULT_OPTION, }) .then((response) => { if (!response.ok) throw new Error('API request error'); @@ -24,4 +24,37 @@ const request = async (url, options = {}) => { return data; }; -export default request; +const get = async (url) => + request(url, { + headers: getHeader(), + method: 'GET', + }); + +const post = async (url, data) => + request(url, { + headers: getHeader(), + method: 'POST', + body: JSON.stringify(data), + }); + +const put = async (url, data) => + request(url, { + headers: getHeader(), + method: 'PUT', + body: JSON.stringify(data), + }); + +const del = async (url) => + request(url, { + headers: getHeader(), + method: 'DELETE', + }); + +const api = { + get, + post, + put, + del, +}; + +export default api; From 1d5437e90b1ec9878885521a8241e8751e9c1710 Mon Sep 17 00:00:00 2001 From: kutta97 Date: Sun, 2 Jul 2023 03:24:27 +0900 Subject: [PATCH 18/87] =?UTF-8?q?feat:=20document=20api=20=EC=9A=94?= =?UTF-8?q?=EC=B2=AD=20=ED=95=A8=EC=88=98=ED=99=94?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .prettierrc.js | 1 + babel.config.js | 1 + jsconfig.json | 1 + src/api/document.js | 12 ++++++++++++ webpack.config.js | 1 + 5 files changed, 16 insertions(+) create mode 100644 src/api/document.js diff --git a/.prettierrc.js b/.prettierrc.js index 7786ed3b..073fdfec 100644 --- a/.prettierrc.js +++ b/.prettierrc.js @@ -6,6 +6,7 @@ module.exports = { importOrder: [ '^@consts/(.*)$', + '^@api/(.*)$', '^@utils/(.*)$', '^@components/(.*)$', '^@pages/(.*)$', diff --git a/babel.config.js b/babel.config.js index b7687a42..a4556599 100644 --- a/babel.config.js +++ b/babel.config.js @@ -6,6 +6,7 @@ module.exports = { { root: ['./src'], alias: { + '@api': './src/api', '@components': './src/components', '@pages': './src/pages', '@consts': './src/consts', diff --git a/jsconfig.json b/jsconfig.json index e11d9b86..b4784344 100644 --- a/jsconfig.json +++ b/jsconfig.json @@ -2,6 +2,7 @@ "compilerOptions": { "baseUrl": ".", "paths": { + "@api/*": ["./src/api/*"], "@utils/*": ["./src/utils/*"], "@components/*": ["./src/components/*"], "@pages/*": ["./src/pages/*"], diff --git a/src/api/document.js b/src/api/document.js new file mode 100644 index 00000000..fb81d924 --- /dev/null +++ b/src/api/document.js @@ -0,0 +1,12 @@ +import api from '@utils/api'; + +export const getDocumentList = async () => api.get('/documents'); + +export const getDocument = async (documentId) => + api.get(`/documents/${documentId}`); + +export const createDocument = async (document) => + api.post('/documents', document); + +export const deleteDocument = async (documentId) => + api.del(`/documents/${documentId}`); diff --git a/webpack.config.js b/webpack.config.js index a97d5dd5..ca93b209 100644 --- a/webpack.config.js +++ b/webpack.config.js @@ -10,6 +10,7 @@ module.exports = (_, argv) => { resolve: { extensions: ['.js'], alias: { + '@api': path.resolve(__dirname, 'src/api'), '@components': path.resolve(__dirname, 'src/components'), '@consts': path.resolve(__dirname, 'src/consts'), '@pages': path.resolve(__dirname, 'src/pages'), From b8b56604b09373263f0706ac097aaceb27723e38 Mon Sep 17 00:00:00 2001 From: kutta97 Date: Sun, 2 Jul 2023 03:25:05 +0900 Subject: [PATCH 19/87] =?UTF-8?q?refactor:=20request=20=EB=8C=80=EC=8B=A0?= =?UTF-8?q?=20=EC=BB=A4=EC=8A=A4=ED=85=80=20api=20=EC=9A=94=EC=B2=AD=20?= =?UTF-8?q?=ED=95=A8=EC=88=98=20=EC=82=AC=EC=9A=A9?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- src/pages/NotionPage.js | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/src/pages/NotionPage.js b/src/pages/NotionPage.js index 062693a2..307f99fb 100644 --- a/src/pages/NotionPage.js +++ b/src/pages/NotionPage.js @@ -1,4 +1,4 @@ -import request from '@utils/api'; +import { getDocument, getDocumentList } from '@api/document'; import NotionDocument from '@components/NotionDocument/NotionDocument'; import NotionSidebar from '@components/NotionSidebar/NotionSidebar'; @@ -18,7 +18,7 @@ export default class NotionPage { async setState(nextState) { this.state = nextState; - const documents = await request('/documents'); + const documents = await getDocumentList(); this.$sidebar.setState(documents); const document = await this.fetchDocument(); @@ -33,7 +33,7 @@ export default class NotionPage { title: '', content: '', }; - const document = await request(`/documents/${documentId}`); + const document = await getDocument(documentId); return document; } } From 9178385bb36302ebb7f067437bb14c5aa6246e21 Mon Sep 17 00:00:00 2001 From: kutta97 Date: Sun, 2 Jul 2023 03:44:17 +0900 Subject: [PATCH 20/87] =?UTF-8?q?feat:=20Editor=20=EC=BB=B4=ED=8F=AC?= =?UTF-8?q?=EB=84=8C=ED=8A=B8=20textArea=20=EB=A1=9C=20=EB=B3=80=EA=B2=BD?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../Editor/Content/NotionEditorContent.css | 9 +++++++++ .../Editor/Content/NotionEditorContent.js | 12 +++--------- src/components/Editor/NotionEditor.css | 13 ------------- src/components/Editor/NotionEditor.js | 6 +++--- src/components/Editor/Title/NotionEditorTitle.css | 4 ++++ src/components/Editor/Title/NotionEditorTitle.js | 3 +++ 6 files changed, 22 insertions(+), 25 deletions(-) diff --git a/src/components/Editor/Content/NotionEditorContent.css b/src/components/Editor/Content/NotionEditorContent.css index e69de29b..a125b7ef 100644 --- a/src/components/Editor/Content/NotionEditorContent.css +++ b/src/components/Editor/Content/NotionEditorContent.css @@ -0,0 +1,9 @@ +.notion-editor-content { + display: flex; + flex-direction: column; + width: 100%; + height: 100%; + padding: 20px; + + font-size: 20px; +} diff --git a/src/components/Editor/Content/NotionEditorContent.js b/src/components/Editor/Content/NotionEditorContent.js index 4961a3bc..f0c02e4c 100644 --- a/src/components/Editor/Content/NotionEditorContent.js +++ b/src/components/Editor/Content/NotionEditorContent.js @@ -2,10 +2,10 @@ import './NotionEditorContent.css'; export default class NotionEditorContent { constructor({ $target }) { - this.$content = document.createElement('div'); + this.$content = document.createElement('textarea'); + this.$content.className = 'notion-editor-content'; this.$content.name = 'content'; - this.$content.contentEditable = true; $target.appendChild(this.$content); } @@ -18,12 +18,6 @@ export default class NotionEditorContent { render() { const { content } = this.state; - const renderedContent = - content - ?.split('\n') - ?.map((line) => line) - ?.join('
    ') ?? ''; - - this.$content.innerHTML = renderedContent; + this.$content.innerHTML = content; } } diff --git a/src/components/Editor/NotionEditor.css b/src/components/Editor/NotionEditor.css index bd191ac4..7f0c744e 100644 --- a/src/components/Editor/NotionEditor.css +++ b/src/components/Editor/NotionEditor.css @@ -5,16 +5,3 @@ height: 100%; padding: 20px 10px; } - -.notion-editor input { - font-size: 20px; -} - -.notion-editor div { - display: flex; - flex-direction: column; - width: 100%; - height: 100%; - - padding: 20px; -} diff --git a/src/components/Editor/NotionEditor.js b/src/components/Editor/NotionEditor.js index 6cfb6c6b..aca7aec5 100644 --- a/src/components/Editor/NotionEditor.js +++ b/src/components/Editor/NotionEditor.js @@ -12,12 +12,12 @@ export default class NotionEditor { this.$title = new NotionEditorTitle({ $target: this.$editor }); this.$content = new NotionEditorContent({ $target: this.$editor }); - this.initEventListener(onEdit); + this.setEvent(onEdit); } getNextStateWithTarget(target) { const name = target.getAttribute('name'); - const value = name === 'title' ? target.value : target.innerHTML; + const { value } = target; return { ...this.state, @@ -25,7 +25,7 @@ export default class NotionEditor { }; } - initEventListener(callback) { + setEvent(callback) { this.$editor.addEventListener('input', (e) => { const nextState = this.getNextStateWithTarget(e.target); callback?.(nextState); diff --git a/src/components/Editor/Title/NotionEditorTitle.css b/src/components/Editor/Title/NotionEditorTitle.css index e69de29b..81dc0ad3 100644 --- a/src/components/Editor/Title/NotionEditorTitle.css +++ b/src/components/Editor/Title/NotionEditorTitle.css @@ -0,0 +1,4 @@ +.notion-editor-title { + font-size: 25px; + padding: 5px 10px; +} diff --git a/src/components/Editor/Title/NotionEditorTitle.js b/src/components/Editor/Title/NotionEditorTitle.js index 5a9fbab4..f350824f 100644 --- a/src/components/Editor/Title/NotionEditorTitle.js +++ b/src/components/Editor/Title/NotionEditorTitle.js @@ -4,6 +4,8 @@ export default class NotionEditorTitle { constructor({ $target }) { this.$title = document.createElement('input'); + this.$title.className = 'notion-editor-title'; + this.$title.type = 'text'; this.$title.name = 'title'; this.$title.placeholder = 'undefined'; @@ -17,6 +19,7 @@ export default class NotionEditorTitle { render() { const { title } = this.state; + this.$title.value = title ?? ''; } } From 9a0b7e00796823c016a76ea1b6882078f91215a3 Mon Sep 17 00:00:00 2001 From: kutta97 Date: Sun, 2 Jul 2023 03:44:48 +0900 Subject: [PATCH 21/87] =?UTF-8?q?refactor:=20=EC=9D=B4=EB=B2=A4=ED=8A=B8?= =?UTF-8?q?=20=EB=A6=AC=EC=8A=A4=EB=84=88=EB=8A=94=20setEvent=20=EB=A1=9C?= =?UTF-8?q?=20=EB=93=B1=EB=A1=9D=ED=95=98=EB=8F=84=EB=A1=9D=20=EB=A6=AC?= =?UTF-8?q?=ED=8C=A9=ED=86=A0=EB=A7=81?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- src/components/NotionSidebar/NotionSidebar.js | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/src/components/NotionSidebar/NotionSidebar.js b/src/components/NotionSidebar/NotionSidebar.js index 473eaa5e..5341c27f 100644 --- a/src/components/NotionSidebar/NotionSidebar.js +++ b/src/components/NotionSidebar/NotionSidebar.js @@ -11,6 +11,10 @@ export default class NotionSidebar { this.$sidebar = document.createElement('ul'); $sidebarContainer.appendChild(this.$sidebar); + this.setEvent(); + } + + setEvent() { this.$sidebar.addEventListener('click', (e) => { const $li = e.target.closest('li'); From d84464578587d36a295809c42d208f5ce093ab5c Mon Sep 17 00:00:00 2001 From: kutta97 Date: Sun, 2 Jul 2023 04:45:41 +0900 Subject: [PATCH 22/87] =?UTF-8?q?feat:=20history=20pushstate,=20replacesta?= =?UTF-8?q?te=20=ED=95=A8=EC=88=98=ED=99=94?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- src/components/NotionSidebar/NotionSidebar.js | 4 +-- src/utils/router.js | 25 ++++++++++++++++++- 2 files changed, 26 insertions(+), 3 deletions(-) diff --git a/src/components/NotionSidebar/NotionSidebar.js b/src/components/NotionSidebar/NotionSidebar.js index 5341c27f..ab8f839b 100644 --- a/src/components/NotionSidebar/NotionSidebar.js +++ b/src/components/NotionSidebar/NotionSidebar.js @@ -1,4 +1,4 @@ -import { push } from '@utils/router'; +import { history } from '@utils/router'; import './NotionSidebar.css'; @@ -20,7 +20,7 @@ export default class NotionSidebar { if ($li) { const { id } = $li.dataset; - push(`/documents/${id}`); + history.push(`/documents/${id}`); } }); } diff --git a/src/utils/router.js b/src/utils/router.js index 6f817c43..e1446931 100644 --- a/src/utils/router.js +++ b/src/utils/router.js @@ -1,5 +1,6 @@ const POPSTATE_EVENT = 'popstate'; const HISTORY_PUSH_EVENT = 'history-push'; +const HISTORY_REPLACE_EVENT = 'history-replace'; const initPopstateEventListener = (onRoute) => { window.addEventListener(POPSTATE_EVENT, () => { @@ -18,15 +19,37 @@ const initHistoryPushEventListener = (onRoute) => { }); }; +const initHistoryReplaceEventListener = (onRoute) => { + window.addEventListener(HISTORY_REPLACE_EVENT, (e) => { + const { nextUrl } = e.detail; + + if (!nextUrl) return; + + window.history.replaceState(null, null, nextUrl); + onRoute(); + }); +}; + export const initRouter = (onRoute) => { initPopstateEventListener(onRoute); initHistoryPushEventListener(onRoute); + initHistoryReplaceEventListener(onRoute); }; -export const push = (nextUrl) => { +const push = (nextUrl) => { window.dispatchEvent( new CustomEvent(HISTORY_PUSH_EVENT, { detail: { nextUrl }, }) ); }; + +const replace = (nextUrl) => { + window.dispatchEvent( + new CustomEvent(HISTORY_REPLACE_EVENT, { + detail: { nextUrl }, + }) + ); +}; + +export const history = { push, replace }; From b118e69b5b97d01ac786ee9f6b00a63489d9e222 Mon Sep 17 00:00:00 2001 From: kutta97 Date: Sun, 2 Jul 2023 04:47:53 +0900 Subject: [PATCH 23/87] =?UTF-8?q?feat:=20document=20=EC=88=98=EC=A0=95=20A?= =?UTF-8?q?PI=20=EC=B6=94=EA=B0=80?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- src/api/document.js | 3 +++ 1 file changed, 3 insertions(+) diff --git a/src/api/document.js b/src/api/document.js index fb81d924..ffc77bf1 100644 --- a/src/api/document.js +++ b/src/api/document.js @@ -8,5 +8,8 @@ export const getDocument = async (documentId) => export const createDocument = async (document) => api.post('/documents', document); +export const updateDocument = async (documentId, document) => + api.put(`/documents/${documentId}`, document); + export const deleteDocument = async (documentId) => api.del(`/documents/${documentId}`); From c8632d344bba9b1ec596364143ff20cd27291baf Mon Sep 17 00:00:00 2001 From: kutta97 Date: Sun, 2 Jul 2023 04:48:31 +0900 Subject: [PATCH 24/87] =?UTF-8?q?feat:=20NotionDocument=20=EC=88=98?= =?UTF-8?q?=EC=A0=95=20=EC=8B=9C,=20=EC=84=9C=EB=B2=84=EC=99=80=20Sync=20?= =?UTF-8?q?=EB=B0=8F=20root=20=EC=97=90=EC=84=9C=20=EC=97=90=EB=94=94?= =?UTF-8?q?=ED=84=B0=20=EC=95=88=EB=B3=B4=EC=9D=B4=EB=8A=94=20=EA=B8=B0?= =?UTF-8?q?=EB=8A=A5=20=EC=B6=94=EA=B0=80?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../NotionDocument/NotionDocument.css | 3 ++ .../NotionDocument/NotionDocument.js | 49 +++++++++++++++---- src/pages/NotionPage.js | 16 ++---- 3 files changed, 48 insertions(+), 20 deletions(-) diff --git a/src/components/NotionDocument/NotionDocument.css b/src/components/NotionDocument/NotionDocument.css index e69de29b..8060d48c 100644 --- a/src/components/NotionDocument/NotionDocument.css +++ b/src/components/NotionDocument/NotionDocument.css @@ -0,0 +1,3 @@ +.notion-document { + width: 100%; +} diff --git a/src/components/NotionDocument/NotionDocument.js b/src/components/NotionDocument/NotionDocument.js index cb73c176..1d2cf5a3 100644 --- a/src/components/NotionDocument/NotionDocument.js +++ b/src/components/NotionDocument/NotionDocument.js @@ -1,4 +1,7 @@ -import { setItem } from '@utils/localStorage'; +import { createDocument, updateDocument } from '@api/document'; + +import { removeItem, setItem } from '@utils/localStorage'; +import { history } from '@utils/router'; import NotionEditor from '@components/Editor/NotionEditor'; @@ -6,8 +9,13 @@ import './NotionDocument.css'; export default class NotionDocument { constructor({ $target }) { + this.$document = document.createElement('div'); + this.$document.className = 'notion-document'; + + $target.appendChild(this.$document); + this.$editor = new NotionEditor({ - $target, + $target: this.$document, onEdit: this.onEdit.bind(this), }); @@ -23,19 +31,42 @@ export default class NotionDocument { ...document, tempSaveDate: new Date(), }); + + const isNew = this.state.id === 'new'; + if (isNew) { + const createdDocument = await createDocument({ title: document.title }); + history.replace(`/documents/${createdDocument.id}`); + removeItem('temp-post-new'); + + this.setState({ + ...this.state, + ...createdDocument, + }); + } else { + await updateDocument(document.id, document); + this.setState({ + ...this.state, + ...document, + }); + } }, 1000); + } - const nextState = { + setState(nextState) { + this.state = { ...this.state, - ...document, + ...nextState, }; - this.setState(nextState); + + const { id, title, content } = this.state; + this.$editor.setState({ id, title, content }); + + this.render(); } - setState(nextState) { - this.state = nextState; + render() { + const { isVisible } = this.state; - const { title, content } = this.state; - this.$editor.setState({ title, content }); + this.$document.style.visibility = isVisible ? 'visible' : 'hidden'; } } diff --git a/src/pages/NotionPage.js b/src/pages/NotionPage.js index 307f99fb..b2951cf4 100644 --- a/src/pages/NotionPage.js +++ b/src/pages/NotionPage.js @@ -21,19 +21,13 @@ export default class NotionPage { const documents = await getDocumentList(); this.$sidebar.setState(documents); - const document = await this.fetchDocument(); - this.$document.setState(document); - } - - async fetchDocument() { const { documentId } = this.state; + if (documentId === 'empty') { + this.$document.setState({ isVisible: false }); + return; + } - if (documentId === 'empty') - return { - title: '', - content: '', - }; const document = await getDocument(documentId); - return document; + this.$document.setState({ ...document, isVisible: true }); } } From cd8d697b490898cdecb74ecaeaac9b550ae763aa Mon Sep 17 00:00:00 2001 From: kutta97 Date: Sun, 2 Jul 2023 04:51:06 +0900 Subject: [PATCH 25/87] =?UTF-8?q?refactor:=20root=20=ED=8E=98=EC=9D=B4?= =?UTF-8?q?=EC=A7=80=20documentId=20=EB=A5=BC=20null=20=EB=A1=9C=20?= =?UTF-8?q?=EC=88=98=EC=A0=95?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- src/App.js | 2 +- src/pages/NotionPage.js | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/src/App.js b/src/App.js index 7696c6c4..48011fdd 100644 --- a/src/App.js +++ b/src/App.js @@ -17,7 +17,7 @@ export default class App { const { pathname } = window.location; if (Pathname.isRoot(pathname)) { - return this.$notionPage.setState({ documentId: 'empty' }); + return this.$notionPage.setState({ documentId: null }); } if (Pathname.isDocument(pathname)) { diff --git a/src/pages/NotionPage.js b/src/pages/NotionPage.js index b2951cf4..f79ec7c6 100644 --- a/src/pages/NotionPage.js +++ b/src/pages/NotionPage.js @@ -22,7 +22,7 @@ export default class NotionPage { this.$sidebar.setState(documents); const { documentId } = this.state; - if (documentId === 'empty') { + if (documentId === null) { this.$document.setState({ isVisible: false }); return; } From d67ace4daad3eeb8e1ac1aec8975e70d1625e3f1 Mon Sep 17 00:00:00 2001 From: kutta97 Date: Sun, 2 Jul 2023 04:55:28 +0900 Subject: [PATCH 26/87] =?UTF-8?q?fix:=20=ED=8E=B8=EC=A7=91=EA=B8=B0=20cont?= =?UTF-8?q?ent=20state=20=EB=9E=9C=EB=8D=94=EB=A7=81=20=EB=90=98=EC=A7=80?= =?UTF-8?q?=20=EC=95=8A=EB=8A=94=20=EB=B2=84=EA=B7=B8=20=EC=88=98=EC=A0=95?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- src/components/Editor/Content/NotionEditorContent.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/components/Editor/Content/NotionEditorContent.js b/src/components/Editor/Content/NotionEditorContent.js index f0c02e4c..c69ed29b 100644 --- a/src/components/Editor/Content/NotionEditorContent.js +++ b/src/components/Editor/Content/NotionEditorContent.js @@ -18,6 +18,6 @@ export default class NotionEditorContent { render() { const { content } = this.state; - this.$content.innerHTML = content; + this.$content.value = content; } } From e8bc6bb33d1086244c6b4d9c74356b24b2077261 Mon Sep 17 00:00:00 2001 From: kutta97 Date: Sun, 2 Jul 2023 05:02:23 +0900 Subject: [PATCH 27/87] =?UTF-8?q?feat:=20Editor=20=EC=9D=98=20onEdit=20?= =?UTF-8?q?=ED=95=B8=EB=93=A4=EB=9F=AC=EB=8A=94=20=EB=AC=B8=EC=84=9C=20?= =?UTF-8?q?=EC=88=98=EC=A0=95=20=EC=A4=91=EC=97=90=EB=A7=8C=20=ED=98=B8?= =?UTF-8?q?=EC=B6=9C=EB=90=98=EB=8F=84=EB=A1=9D=20=EB=B3=80=EA=B2=BD?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../NotionDocument/NotionDocument.js | 27 +++++-------------- 1 file changed, 7 insertions(+), 20 deletions(-) diff --git a/src/components/NotionDocument/NotionDocument.js b/src/components/NotionDocument/NotionDocument.js index 1d2cf5a3..9da2bbb9 100644 --- a/src/components/NotionDocument/NotionDocument.js +++ b/src/components/NotionDocument/NotionDocument.js @@ -1,7 +1,6 @@ -import { createDocument, updateDocument } from '@api/document'; +import { updateDocument } from '@api/document'; -import { removeItem, setItem } from '@utils/localStorage'; -import { history } from '@utils/router'; +import { setItem } from '@utils/localStorage'; import NotionEditor from '@components/Editor/NotionEditor'; @@ -32,23 +31,11 @@ export default class NotionDocument { tempSaveDate: new Date(), }); - const isNew = this.state.id === 'new'; - if (isNew) { - const createdDocument = await createDocument({ title: document.title }); - history.replace(`/documents/${createdDocument.id}`); - removeItem('temp-post-new'); - - this.setState({ - ...this.state, - ...createdDocument, - }); - } else { - await updateDocument(document.id, document); - this.setState({ - ...this.state, - ...document, - }); - } + const updatedDocument = await updateDocument(document.id, document); + this.setState({ + ...this.state, + ...updatedDocument, + }); }, 1000); } From 0af7d84e50d39505d2d33c5d825adeeca05db08f Mon Sep 17 00:00:00 2001 From: kutta97 Date: Sun, 2 Jul 2023 05:40:37 +0900 Subject: [PATCH 28/87] =?UTF-8?q?feat:=20Button=20=EC=BB=B4=ED=8F=AC?= =?UTF-8?q?=EB=84=8C=ED=8A=B8=20=EC=B6=94=EA=B0=80?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- src/components/Button/Button.js | 18 ++++++++++++++++++ 1 file changed, 18 insertions(+) create mode 100644 src/components/Button/Button.js diff --git a/src/components/Button/Button.js b/src/components/Button/Button.js new file mode 100644 index 00000000..18cfa50b --- /dev/null +++ b/src/components/Button/Button.js @@ -0,0 +1,18 @@ +export default class Button { + constructor({ $target, className, textContent, onClick }) { + this.$createButton = document.createElement('button'); + + this.$createButton.className = className; + this.$createButton.textContent = textContent; + + $target.appendChild(this.$createButton); + + this.setEvent(onClick); + } + + setEvent(callback) { + this.$createButton.addEventListener('click', () => { + callback?.(); + }); + } +} From 5e1061f2c50654dd9bd43369ee7e37872d5be4ee Mon Sep 17 00:00:00 2001 From: kutta97 Date: Sun, 2 Jul 2023 05:41:44 +0900 Subject: [PATCH 29/87] =?UTF-8?q?feat:=20=EB=AC=B8=EC=84=9C=20=EB=AA=A9?= =?UTF-8?q?=EB=A1=9D=20=EB=B3=B4=EC=97=AC=EC=A3=BC=EB=8A=94=20=ED=99=94?= =?UTF-8?q?=EB=A9=B4=20DocumentList=20=EB=A1=9C=20=EB=B6=84=EB=A6=AC?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- src/components/DocumentList/DocumentList.css | 19 +++++++ src/components/DocumentList/DocumentList.js | 41 ++++++++++++++ .../NotionSidebar/NotionSidebar.css | 24 ++------- src/components/NotionSidebar/NotionSidebar.js | 53 +++++++++---------- src/pages/NotionPage.js | 4 +- 5 files changed, 92 insertions(+), 49 deletions(-) create mode 100644 src/components/DocumentList/DocumentList.css create mode 100644 src/components/DocumentList/DocumentList.js diff --git a/src/components/DocumentList/DocumentList.css b/src/components/DocumentList/DocumentList.css new file mode 100644 index 00000000..7dd810de --- /dev/null +++ b/src/components/DocumentList/DocumentList.css @@ -0,0 +1,19 @@ +.document-list { + width: 100%; + height: 100%; + padding: 10px 5px; +} + +.document-list li { + width: 100%; + padding: 5px 10px; + + color: #19171199; + font-size: 14px; + font-weight: 600; + + &:hover { + background-color: #ededeb; + cursor: pointer; + } +} diff --git a/src/components/DocumentList/DocumentList.js b/src/components/DocumentList/DocumentList.js new file mode 100644 index 00000000..c41f2fe4 --- /dev/null +++ b/src/components/DocumentList/DocumentList.js @@ -0,0 +1,41 @@ +import { history } from '@utils/router'; + +import './DocumentList.css'; + +export default class DocumentList { + constructor({ $target }) { + this.$documentList = document.createElement('ul'); + this.$documentList.className = 'document-list'; + + $target.appendChild(this.$documentList); + + this.setEvent(); + } + + setEvent() { + this.$documentList.addEventListener('click', (e) => { + const $li = e.target.closest('li'); + + if ($li) { + const { id } = $li.dataset; + history.push(`/documents/${id}`); + } + }); + } + + setState(nextState) { + this.state = nextState; + this.render(); + } + + render() { + const { documentList } = this.state; + this.$documentList.innerHTML = documentList + .map( + (document) => ` +
  • ${document.title}
  • + ` + ) + .join(''); + } +} diff --git a/src/components/NotionSidebar/NotionSidebar.css b/src/components/NotionSidebar/NotionSidebar.css index 0c778d69..f6b8d7e1 100644 --- a/src/components/NotionSidebar/NotionSidebar.css +++ b/src/components/NotionSidebar/NotionSidebar.css @@ -1,29 +1,15 @@ -.notion-sidebar-container { +.notion-sidebar { display: flex; flex-direction: column; width: 300px; height: 100%; + padding: 20px 0; border-right: 1px solid #f5f6f5; background-color: #fbfbfa; } -.notion-sidebar-container ul { - width: 100%; - height: 100%; - padding: 10px 5px; -} - -.notion-sidebar-container ul li { - width: 100%; - padding: 5px 10px; - - color: #19171199; - font-size: 14px; - font-weight: 600; - - &:hover { - background-color: #ededeb; - cursor: pointer; - } +.document-create-button { + height: 30px; + margin: 5px 10px; } diff --git a/src/components/NotionSidebar/NotionSidebar.js b/src/components/NotionSidebar/NotionSidebar.js index ab8f839b..cca1b03d 100644 --- a/src/components/NotionSidebar/NotionSidebar.js +++ b/src/components/NotionSidebar/NotionSidebar.js @@ -1,42 +1,39 @@ +import { createDocument } from '@api/document'; + import { history } from '@utils/router'; +import Button from '@components/Button/Button'; +import DocumentList from '@components/DocumentList/DocumentList'; + import './NotionSidebar.css'; export default class NotionSidebar { constructor({ $target }) { - const $sidebarContainer = document.createElement('nav'); - $sidebarContainer.className = 'notion-sidebar-container'; - $target.appendChild($sidebarContainer); - - this.$sidebar = document.createElement('ul'); - $sidebarContainer.appendChild(this.$sidebar); - - this.setEvent(); - } - - setEvent() { - this.$sidebar.addEventListener('click', (e) => { - const $li = e.target.closest('li'); - - if ($li) { - const { id } = $li.dataset; - history.push(`/documents/${id}`); - } + this.$sidebar = document.createElement('nav'); + this.$sidebar.className = 'notion-sidebar'; + + $target.appendChild(this.$sidebar); + + this.$createButton = new Button({ + $target: this.$sidebar, + className: 'document-create-button', + textContent: 'add a document', + onClick: () => { + this.handleCreateButtonClick(); + }, }); + this.$documentList = new DocumentList({ $target: this.$sidebar }); } + handleCreateButtonClick = async () => { + const newDocument = await createDocument({ title: 'Untitled' }); + history.push(`/documents/${newDocument.id}`); + }; + setState(nextState) { this.state = nextState; - this.render(); - } - render() { - this.$sidebar.innerHTML = this.state - .map( - (document) => ` -
  • ${document.title}
  • - ` - ) - .join(''); + const { documentList } = this.state; + this.$documentList.setState({ documentList }); } } diff --git a/src/pages/NotionPage.js b/src/pages/NotionPage.js index f79ec7c6..bc11347f 100644 --- a/src/pages/NotionPage.js +++ b/src/pages/NotionPage.js @@ -18,8 +18,8 @@ export default class NotionPage { async setState(nextState) { this.state = nextState; - const documents = await getDocumentList(); - this.$sidebar.setState(documents); + const documentList = await getDocumentList(); + this.$sidebar.setState({ documentList }); const { documentId } = this.state; if (documentId === null) { From a4b493f02e52d5b39f6541994d24a9c120dfbfa0 Mon Sep 17 00:00:00 2001 From: kutta97 Date: Sun, 2 Jul 2023 05:42:11 +0900 Subject: [PATCH 30/87] =?UTF-8?q?chore:=20class=20method=20=EC=95=88?= =?UTF-8?q?=EC=97=90=EC=84=9C=20=EC=9E=84=EC=8B=9C=20=EB=B3=80=EC=88=98=20?= =?UTF-8?q?=EC=82=AC=EC=9A=A9=EC=9D=84=20=EC=9C=84=ED=95=B4=20class-method?= =?UTF-8?q?s-use-this=20=EC=98=B5=EC=85=98=20Off?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .eslintrc.js | 1 + 1 file changed, 1 insertion(+) diff --git a/.eslintrc.js b/.eslintrc.js index e46890d2..1525140e 100644 --- a/.eslintrc.js +++ b/.eslintrc.js @@ -11,5 +11,6 @@ module.exports = { plugins: [], rules: { 'import/no-unresolved': 'off', + 'class-methods-use-this': 'off', }, }; From 6c2cbb29835b1f2201f0b513040f81c36757d542 Mon Sep 17 00:00:00 2001 From: kutta97 Date: Sun, 2 Jul 2023 19:52:55 +0900 Subject: [PATCH 31/87] =?UTF-8?q?feat:=20Sidebar=20=EC=97=90=EC=84=9C=20Ro?= =?UTF-8?q?ot=20Document=20=EC=9D=98=20=ED=95=98=EC=9C=84=20Document=20?= =?UTF-8?q?=EB=9E=9C=EB=8D=94=EB=A7=81=ED=95=98=EB=8A=94=20=EA=B8=B0?= =?UTF-8?q?=EB=8A=A5=20=EC=B6=94=EA=B0=80?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- src/components/DocumentList/DocumentList.css | 13 ++++++- src/components/DocumentList/DocumentList.js | 37 +++++++++++++++---- .../NotionSidebar/NotionSidebar.css | 2 +- 3 files changed, 43 insertions(+), 9 deletions(-) diff --git a/src/components/DocumentList/DocumentList.css b/src/components/DocumentList/DocumentList.css index 7dd810de..b7ac5749 100644 --- a/src/components/DocumentList/DocumentList.css +++ b/src/components/DocumentList/DocumentList.css @@ -5,8 +5,12 @@ } .document-list li { + display: flex; + flex-direction: row; + align-items: center; width: 100%; - padding: 5px 10px; + padding: 2px 0; + border-radius: 3px; color: #19171199; font-size: 14px; @@ -17,3 +21,10 @@ cursor: pointer; } } + +.document-list li a { + width: 100%; + overflow: hidden; + text-overflow: ellipsis; + white-space: nowrap; +} diff --git a/src/components/DocumentList/DocumentList.js b/src/components/DocumentList/DocumentList.js index c41f2fe4..3d8baf56 100644 --- a/src/components/DocumentList/DocumentList.js +++ b/src/components/DocumentList/DocumentList.js @@ -28,14 +28,37 @@ export default class DocumentList { this.render(); } + createDocumentItem(item, depth = 1) { + const $div = document.createElement('div'); + + const $li = document.createElement('li'); + $li.dataset.id = item.id; + $li.innerHTML = ` + ${item.title} + + `; + $li.style.paddingLeft = `${depth * 5}px`; + + $div.appendChild($li); + + const { documents: childItems } = item; + + if (childItems.length > 0) { + const $ul = document.createElement('ul'); + childItems.forEach((childItem) => { + $ul.appendChild(this.createDocumentItem(childItem, depth + 1)); + }); + $div.appendChild($ul); + } + + return $div; + } + render() { const { documentList } = this.state; - this.$documentList.innerHTML = documentList - .map( - (document) => ` -
  • ${document.title}
  • - ` - ) - .join(''); + documentList.forEach((document) => { + const $documentItem = this.createDocumentItem(document); + this.$documentList.appendChild($documentItem); + }); } } diff --git a/src/components/NotionSidebar/NotionSidebar.css b/src/components/NotionSidebar/NotionSidebar.css index f6b8d7e1..ff55e9ed 100644 --- a/src/components/NotionSidebar/NotionSidebar.css +++ b/src/components/NotionSidebar/NotionSidebar.css @@ -1,7 +1,7 @@ .notion-sidebar { display: flex; flex-direction: column; - width: 300px; + width: 250px; height: 100%; padding: 20px 0; border-right: 1px solid #f5f6f5; From 1cd97db83db6b253bba19b1e49b4ce9315bb1d5f Mon Sep 17 00:00:00 2001 From: kutta97 Date: Sun, 2 Jul 2023 20:21:33 +0900 Subject: [PATCH 32/87] =?UTF-8?q?feat:=20root=20Document=20=ED=95=98?= =?UTF-8?q?=EC=9C=84=20=EB=AC=B8=EC=84=9C=20=EC=83=9D=EC=84=B1=ED=95=98?= =?UTF-8?q?=EB=8A=94=20=EA=B8=B0=EB=8A=A5=20=EC=B6=94=EA=B0=80?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- src/components/DocumentList/DocumentList.css | 2 ++ src/components/DocumentList/DocumentList.js | 24 ++++++++++++++++---- 2 files changed, 21 insertions(+), 5 deletions(-) diff --git a/src/components/DocumentList/DocumentList.css b/src/components/DocumentList/DocumentList.css index b7ac5749..9200e8f9 100644 --- a/src/components/DocumentList/DocumentList.css +++ b/src/components/DocumentList/DocumentList.css @@ -2,6 +2,8 @@ width: 100%; height: 100%; padding: 10px 5px; + + overflow-y: scroll; } .document-list li { diff --git a/src/components/DocumentList/DocumentList.js b/src/components/DocumentList/DocumentList.js index 3d8baf56..5c6f37cd 100644 --- a/src/components/DocumentList/DocumentList.js +++ b/src/components/DocumentList/DocumentList.js @@ -1,3 +1,5 @@ +import { createDocument } from '@api/document'; + import { history } from '@utils/router'; import './DocumentList.css'; @@ -12,14 +14,25 @@ export default class DocumentList { this.setEvent(); } + handleCreateIndsideButtonClick = async (id) => { + const newDocument = await createDocument({ title: 'Untitled', parent: id }); + history.push(`/documents/${newDocument.id}`); + }; + setEvent() { this.$documentList.addEventListener('click', (e) => { const $li = e.target.closest('li'); - if ($li) { - const { id } = $li.dataset; - history.push(`/documents/${id}`); + if (!$li) return; + + const { id } = $li.dataset; + const { className } = e.target; + + if (className === 'document-create-inside-button') { + this.handleCreateIndsideButtonClick(id); + return; } + history.push(`/documents/${id}`); }); } @@ -35,9 +48,9 @@ export default class DocumentList { $li.dataset.id = item.id; $li.innerHTML = ` ${item.title} - + `; - $li.style.paddingLeft = `${depth * 5}px`; + $li.style.paddingLeft = `${depth * 10}px`; $div.appendChild($li); @@ -56,6 +69,7 @@ export default class DocumentList { render() { const { documentList } = this.state; + this.$documentList.innerHTML = ''; documentList.forEach((document) => { const $documentItem = this.createDocumentItem(document); this.$documentList.appendChild($documentItem); From 311c3fb0704706e2e9116f5c33243513453f21a5 Mon Sep 17 00:00:00 2001 From: kutta97 Date: Sun, 2 Jul 2023 20:50:47 +0900 Subject: [PATCH 33/87] =?UTF-8?q?feat:=20DocumentListItem=20=EC=BB=B4?= =?UTF-8?q?=ED=8F=AC=EB=84=8C=ED=8A=B8=20=EB=B6=84=EB=A6=AC?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- src/components/DocumentList/DocumentList.css | 25 ------------------- src/components/DocumentList/DocumentList.js | 23 ++++++++--------- .../DocumentList/Item/DocumentListItem.css | 24 ++++++++++++++++++ .../DocumentList/Item/DocumentListItem.js | 25 +++++++++++++++++++ 4 files changed, 59 insertions(+), 38 deletions(-) create mode 100644 src/components/DocumentList/Item/DocumentListItem.css create mode 100644 src/components/DocumentList/Item/DocumentListItem.js diff --git a/src/components/DocumentList/DocumentList.css b/src/components/DocumentList/DocumentList.css index 9200e8f9..b8d9ffd0 100644 --- a/src/components/DocumentList/DocumentList.css +++ b/src/components/DocumentList/DocumentList.css @@ -5,28 +5,3 @@ overflow-y: scroll; } - -.document-list li { - display: flex; - flex-direction: row; - align-items: center; - width: 100%; - padding: 2px 0; - border-radius: 3px; - - color: #19171199; - font-size: 14px; - font-weight: 600; - - &:hover { - background-color: #ededeb; - cursor: pointer; - } -} - -.document-list li a { - width: 100%; - overflow: hidden; - text-overflow: ellipsis; - white-space: nowrap; -} diff --git a/src/components/DocumentList/DocumentList.js b/src/components/DocumentList/DocumentList.js index 5c6f37cd..3937af54 100644 --- a/src/components/DocumentList/DocumentList.js +++ b/src/components/DocumentList/DocumentList.js @@ -2,6 +2,8 @@ import { createDocument } from '@api/document'; import { history } from '@utils/router'; +import DocumentListItem from '@components/DocumentList/Item/DocumentListItem'; + import './DocumentList.css'; export default class DocumentList { @@ -20,13 +22,12 @@ export default class DocumentList { }; setEvent() { - this.$documentList.addEventListener('click', (e) => { - const $li = e.target.closest('li'); - + this.$documentList.addEventListener('click', ({ target }) => { + const $li = target.closest('li'); if (!$li) return; const { id } = $li.dataset; - const { className } = e.target; + const { className } = target; if (className === 'document-create-inside-button') { this.handleCreateIndsideButtonClick(id); @@ -44,15 +45,11 @@ export default class DocumentList { createDocumentItem(item, depth = 1) { const $div = document.createElement('div'); - const $li = document.createElement('li'); - $li.dataset.id = item.id; - $li.innerHTML = ` - ${item.title} - - `; - $li.style.paddingLeft = `${depth * 10}px`; - - $div.appendChild($li); + const $documentItem = new DocumentListItem({ $target: $div }); + $documentItem.setState({ + documentItem: item, + depth, + }); const { documents: childItems } = item; diff --git a/src/components/DocumentList/Item/DocumentListItem.css b/src/components/DocumentList/Item/DocumentListItem.css new file mode 100644 index 00000000..37f98264 --- /dev/null +++ b/src/components/DocumentList/Item/DocumentListItem.css @@ -0,0 +1,24 @@ +.document-list li { + display: flex; + flex-direction: row; + align-items: center; + width: 100%; + padding: 2px 0; + border-radius: 3px; + + color: #19171199; + font-size: 14px; + font-weight: 600; + + &:hover { + background-color: #ededeb; + cursor: pointer; + } +} + +.document-list li a { + width: 100%; + overflow: hidden; + text-overflow: ellipsis; + white-space: nowrap; +} diff --git a/src/components/DocumentList/Item/DocumentListItem.js b/src/components/DocumentList/Item/DocumentListItem.js new file mode 100644 index 00000000..dc48139b --- /dev/null +++ b/src/components/DocumentList/Item/DocumentListItem.js @@ -0,0 +1,25 @@ +import './DocumentListItem.css'; + +export default class DocumentListItem { + constructor({ $target }) { + this.$documentListItem = document.createElement('li'); + + $target.appendChild(this.$documentListItem); + } + + setState(nextState) { + this.state = nextState; + this.render(); + } + + render() { + const { documentItem, depth } = this.state; + + this.$documentListItem.dataset.id = documentItem.id; + this.$documentListItem.innerHTML = ` + ${documentItem.title} + + `; + this.$documentListItem.style.paddingLeft = `${depth * 10}px`; + } +} From be413bf981b79eaab76546dfc39f992d05d320cb Mon Sep 17 00:00:00 2001 From: kutta97 Date: Sun, 2 Jul 2023 22:26:40 +0900 Subject: [PATCH 34/87] =?UTF-8?q?feat:=20Sidebar=20list=20item=20=EC=97=90?= =?UTF-8?q?=20=EB=AC=B8=EC=84=9C=20=EC=82=AD=EC=A0=9C=20=EA=B8=B0=EB=8A=A5?= =?UTF-8?q?=20=EB=B2=84=ED=8A=BC=20=EC=B6=94=EA=B0=80?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- src/components/DocumentList/DocumentList.js | 12 +++++++++++- src/components/DocumentList/Item/DocumentListItem.js | 1 + 2 files changed, 12 insertions(+), 1 deletion(-) diff --git a/src/components/DocumentList/DocumentList.js b/src/components/DocumentList/DocumentList.js index 3937af54..dfc0e8c6 100644 --- a/src/components/DocumentList/DocumentList.js +++ b/src/components/DocumentList/DocumentList.js @@ -1,4 +1,4 @@ -import { createDocument } from '@api/document'; +import { createDocument, deleteDocument } from '@api/document'; import { history } from '@utils/router'; @@ -21,6 +21,11 @@ export default class DocumentList { history.push(`/documents/${newDocument.id}`); }; + hanldeDeleteButtonClick = async (id) => { + await deleteDocument(id); + history.push('/'); + }; + setEvent() { this.$documentList.addEventListener('click', ({ target }) => { const $li = target.closest('li'); @@ -29,6 +34,11 @@ export default class DocumentList { const { id } = $li.dataset; const { className } = target; + if (className === 'document-delete-button') { + this.hanldeDeleteButtonClick(id); + return; + } + if (className === 'document-create-inside-button') { this.handleCreateIndsideButtonClick(id); return; diff --git a/src/components/DocumentList/Item/DocumentListItem.js b/src/components/DocumentList/Item/DocumentListItem.js index dc48139b..a1c37ed4 100644 --- a/src/components/DocumentList/Item/DocumentListItem.js +++ b/src/components/DocumentList/Item/DocumentListItem.js @@ -18,6 +18,7 @@ export default class DocumentListItem { this.$documentListItem.dataset.id = documentItem.id; this.$documentListItem.innerHTML = ` ${documentItem.title} + `; this.$documentListItem.style.paddingLeft = `${depth * 10}px`; From 823d96bd33be4063c795ffa267a6af3ffa629ac3 Mon Sep 17 00:00:00 2001 From: kutta97 Date: Mon, 3 Jul 2023 05:59:20 +0900 Subject: [PATCH 35/87] =?UTF-8?q?refactor:=20Component=20core=20class=20?= =?UTF-8?q?=EC=A0=81=EC=9A=A9?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .prettierrc.js | 1 + babel.config.js | 1 + jsconfig.json | 5 +- src/App.js | 31 ++++++-- src/components/Button/Button.js | 26 ++++--- src/components/DocumentList/DocumentList.js | 58 +++++++------- .../Editor/Content/NotionEditorContent.js | 29 +++---- src/components/Editor/NotionEditor.css | 9 ++- src/components/Editor/NotionEditor.js | 67 +++++++++------- .../Editor/Title/NotionEditorTitle.css | 1 + .../Editor/Title/NotionEditorTitle.js | 34 ++++---- .../NotionDocument/NotionDocument.css | 7 ++ .../NotionDocument/NotionDocument.js | 67 ++++++---------- .../NotionSidebar/NotionSidebar.css | 2 +- src/components/NotionSidebar/NotionSidebar.js | 32 +++++--- src/core/Component.js | 40 ++++++++++ src/main.js | 2 +- src/pages/NotionPage.css | 11 +++ src/pages/NotionPage.js | 78 +++++++++++++++---- src/utils/api.js | 2 +- webpack.config.js | 1 + 21 files changed, 313 insertions(+), 191 deletions(-) create mode 100644 src/core/Component.js diff --git a/.prettierrc.js b/.prettierrc.js index 073fdfec..caae71cd 100644 --- a/.prettierrc.js +++ b/.prettierrc.js @@ -8,6 +8,7 @@ module.exports = { '^@consts/(.*)$', '^@api/(.*)$', '^@utils/(.*)$', + '^@core/(.*)$', '^@components/(.*)$', '^@pages/(.*)$', '^[./]', diff --git a/babel.config.js b/babel.config.js index a4556599..6191f2dd 100644 --- a/babel.config.js +++ b/babel.config.js @@ -7,6 +7,7 @@ module.exports = { root: ['./src'], alias: { '@api': './src/api', + '@core': './src/core', '@components': './src/components', '@pages': './src/pages', '@consts': './src/consts', diff --git a/jsconfig.json b/jsconfig.json index b4784344..ac9e5033 100644 --- a/jsconfig.json +++ b/jsconfig.json @@ -2,11 +2,12 @@ "compilerOptions": { "baseUrl": ".", "paths": { + "@consts/*": ["./src/consts/*"], "@api/*": ["./src/api/*"], "@utils/*": ["./src/utils/*"], + "@core/*": ["./src/core/*"], "@components/*": ["./src/components/*"], - "@pages/*": ["./src/pages/*"], - "@consts/*": ["./src/consts/*"] + "@pages/*": ["./src/pages/*"] } } } diff --git a/src/App.js b/src/App.js index 48011fdd..88f79187 100644 --- a/src/App.js +++ b/src/App.js @@ -1,28 +1,47 @@ import Pathname from '@utils/pathname'; import { initRouter } from '@utils/router'; +import Component from '@core/Component'; + import NotionPage from '@pages/NotionPage'; import './App.css'; -export default class App { - constructor({ $target }) { - this.$notionPage = new NotionPage({ $target }); +export default class App extends Component { + constructor($target, props) { + super($target, props); initRouter(this.route.bind(this)); this.route(); } + template() { + return ` +
    + `; + } + + mount() { + const { documentId } = this; + const $notionPage = this.$target.querySelector('.notion-page'); + this.$notionPage = new NotionPage($notionPage, { documentId }); + } + + get documentId() { + const { pathname } = window.location; + const [, , documentId] = pathname.split('/'); + return documentId ?? null; + } + route() { const { pathname } = window.location; if (Pathname.isRoot(pathname)) { - return this.$notionPage.setState({ documentId: null }); + return this.setState(); } if (Pathname.isDocument(pathname)) { - const [, , documentId] = pathname.split('/'); - return this.$notionPage.setState({ documentId }); + return this.setState(); } return null; diff --git a/src/components/Button/Button.js b/src/components/Button/Button.js index 18cfa50b..8ef92dd6 100644 --- a/src/components/Button/Button.js +++ b/src/components/Button/Button.js @@ -1,18 +1,20 @@ -export default class Button { - constructor({ $target, className, textContent, onClick }) { - this.$createButton = document.createElement('button'); +import Component from '@core/Component'; - this.$createButton.className = className; - this.$createButton.textContent = textContent; - - $target.appendChild(this.$createButton); - - this.setEvent(onClick); +export default class Button extends Component { + template() { + const { className, textContent } = this.props; + return ` + + `; } - setEvent(callback) { - this.$createButton.addEventListener('click', () => { - callback?.(); + setEvent() { + const { className, onClick } = this.props; + + this.addEvent('click', `.${className}`, () => { + onClick(); }); } } diff --git a/src/components/DocumentList/DocumentList.js b/src/components/DocumentList/DocumentList.js index dfc0e8c6..b1735fe7 100644 --- a/src/components/DocumentList/DocumentList.js +++ b/src/components/DocumentList/DocumentList.js @@ -2,34 +2,34 @@ import { createDocument, deleteDocument } from '@api/document'; import { history } from '@utils/router'; +import Component from '@core/Component'; + import DocumentListItem from '@components/DocumentList/Item/DocumentListItem'; import './DocumentList.css'; -export default class DocumentList { - constructor({ $target }) { - this.$documentList = document.createElement('ul'); - this.$documentList.className = 'document-list'; - - $target.appendChild(this.$documentList); - - this.setEvent(); +export default class DocumentList extends Component { + setup() { + this.state = { + documentList: [], + }; } - handleCreateIndsideButtonClick = async (id) => { - const newDocument = await createDocument({ title: 'Untitled', parent: id }); - history.push(`/documents/${newDocument.id}`); - }; + render() { + super.render(); - hanldeDeleteButtonClick = async (id) => { - await deleteDocument(id); - history.push('/'); - }; + const { documentList } = this.props; + if (documentList.length === 0) return; + + documentList.forEach((document) => { + const $documentItem = this.createDocumentItem(document); + this.$target.appendChild($documentItem); + }); + } setEvent() { - this.$documentList.addEventListener('click', ({ target }) => { + this.addEvent('click', 'li', ({ target }) => { const $li = target.closest('li'); - if (!$li) return; const { id } = $li.dataset; const { className } = target; @@ -47,10 +47,15 @@ export default class DocumentList { }); } - setState(nextState) { - this.state = nextState; - this.render(); - } + handleCreateIndsideButtonClick = async (id) => { + const newDocument = await createDocument({ title: 'Untitled', parent: id }); + history.push(`/documents/${newDocument.id}`); + }; + + hanldeDeleteButtonClick = async (id) => { + await deleteDocument(id); + history.push('/'); + }; createDocumentItem(item, depth = 1) { const $div = document.createElement('div'); @@ -73,13 +78,4 @@ export default class DocumentList { return $div; } - - render() { - const { documentList } = this.state; - this.$documentList.innerHTML = ''; - documentList.forEach((document) => { - const $documentItem = this.createDocumentItem(document); - this.$documentList.appendChild($documentItem); - }); - } } diff --git a/src/components/Editor/Content/NotionEditorContent.js b/src/components/Editor/Content/NotionEditorContent.js index c69ed29b..5c75ba0b 100644 --- a/src/components/Editor/Content/NotionEditorContent.js +++ b/src/components/Editor/Content/NotionEditorContent.js @@ -1,23 +1,16 @@ -import './NotionEditorContent.css'; - -export default class NotionEditorContent { - constructor({ $target }) { - this.$content = document.createElement('textarea'); +import Component from '@core/Component'; - this.$content.className = 'notion-editor-content'; - this.$content.name = 'content'; - - $target.appendChild(this.$content); - } - - setState(nextState) { - this.state = nextState; - this.render(); - } +import './NotionEditorContent.css'; - render() { - const { content } = this.state; +export default class NotionEditorContent extends Component { + template() { + const { content } = this.props; - this.$content.value = content; + return ` + + `; } } diff --git a/src/components/Editor/NotionEditor.css b/src/components/Editor/NotionEditor.css index 7f0c744e..53765b0b 100644 --- a/src/components/Editor/NotionEditor.css +++ b/src/components/Editor/NotionEditor.css @@ -1,7 +1,8 @@ -.notion-editor { - display: flex; - flex-direction: column; +.notion-editor-title-wrapper { + width: 100%; +} + +.notion-editor-content-wrapper { width: 100%; height: 100%; - padding: 20px 10px; } diff --git a/src/components/Editor/NotionEditor.js b/src/components/Editor/NotionEditor.js index aca7aec5..d896ba38 100644 --- a/src/components/Editor/NotionEditor.js +++ b/src/components/Editor/NotionEditor.js @@ -1,42 +1,57 @@ +import Component from '@core/Component'; + import NotionEditorTitle from '@components//Editor/Title/NotionEditorTitle'; import NotionEditorContent from '@components/Editor/Content/NotionEditorContent'; import './NotionEditor.css'; -export default class NotionEditor { - constructor({ $target, onEdit }) { - this.$editor = document.createElement('section'); - this.$editor.className = 'notion-editor'; - $target.appendChild(this.$editor); - - this.$title = new NotionEditorTitle({ $target: this.$editor }); - this.$content = new NotionEditorContent({ $target: this.$editor }); +export default class NotionEditor extends Component { + template() { + return ` +
    +
    + `; + } - this.setEvent(onEdit); + mount() { + const { title, content } = this; + const $titleEditor = this.$target.querySelector( + '.notion-editor-title-wrapper' + ); + const $contentEditor = this.$target.querySelector( + '.notion-editor-content-wrapper' + ); + + this.$titleEditor = new NotionEditorTitle($titleEditor, { title }); + this.$contentEditor = new NotionEditorContent($contentEditor, { content }); } - getNextStateWithTarget(target) { - const name = target.getAttribute('name'); - const { value } = target; + get title() { + const { documentData } = this.props; + const { title } = documentData; - return { - ...this.state, - [name]: value, - }; + return title; } - setEvent(callback) { - this.$editor.addEventListener('input', (e) => { - const nextState = this.getNextStateWithTarget(e.target); - callback?.(nextState); - }); + get content() { + const { documentData } = this.props; + const { content } = documentData; + + return content; } - setState(nextState) { - this.state = nextState; + setEvent() { + const { onEdit } = this.props; + + this.$target.addEventListener('input', ({ target }) => { + const { name } = target; + const { documentData } = this.props; - const { title, content } = this.state; - this.$title.setState({ title }); - this.$content.setState({ content }); + if (documentData[name] === undefined) return; + onEdit({ + ...documentData, + [name]: target.value, + }); + }); } } diff --git a/src/components/Editor/Title/NotionEditorTitle.css b/src/components/Editor/Title/NotionEditorTitle.css index 81dc0ad3..e5b9372f 100644 --- a/src/components/Editor/Title/NotionEditorTitle.css +++ b/src/components/Editor/Title/NotionEditorTitle.css @@ -1,4 +1,5 @@ .notion-editor-title { + width: 100%; font-size: 25px; padding: 5px 10px; } diff --git a/src/components/Editor/Title/NotionEditorTitle.js b/src/components/Editor/Title/NotionEditorTitle.js index f350824f..6bc6d97f 100644 --- a/src/components/Editor/Title/NotionEditorTitle.js +++ b/src/components/Editor/Title/NotionEditorTitle.js @@ -1,25 +1,19 @@ -import './NotionEditorTitle.css'; - -export default class NotionEditorTitle { - constructor({ $target }) { - this.$title = document.createElement('input'); +import Component from '@core/Component'; - this.$title.className = 'notion-editor-title'; - this.$title.type = 'text'; - this.$title.name = 'title'; - this.$title.placeholder = 'undefined'; - - $target.appendChild(this.$title); - } - - setState(nextState) { - this.state = nextState; - this.render(); - } +import './NotionEditorTitle.css'; - render() { - const { title } = this.state; +export default class NotionEditorTitle extends Component { + template() { + const { title } = this.props; - this.$title.value = title ?? ''; + return ` + + `; } } diff --git a/src/components/NotionDocument/NotionDocument.css b/src/components/NotionDocument/NotionDocument.css index 8060d48c..4d49d235 100644 --- a/src/components/NotionDocument/NotionDocument.css +++ b/src/components/NotionDocument/NotionDocument.css @@ -1,3 +1,10 @@ .notion-document { width: 100%; } + +.notion-editor-wrapper { + display: flex; + flex-direction: column; + width: 100%; + height: 100%; +} diff --git a/src/components/NotionDocument/NotionDocument.js b/src/components/NotionDocument/NotionDocument.js index 9da2bbb9..999de423 100644 --- a/src/components/NotionDocument/NotionDocument.js +++ b/src/components/NotionDocument/NotionDocument.js @@ -1,59 +1,38 @@ -import { updateDocument } from '@api/document'; - -import { setItem } from '@utils/localStorage'; +import Component from '@core/Component'; import NotionEditor from '@components/Editor/NotionEditor'; import './NotionDocument.css'; -export default class NotionDocument { - constructor({ $target }) { - this.$document = document.createElement('div'); - this.$document.className = 'notion-document'; - - $target.appendChild(this.$document); - - this.$editor = new NotionEditor({ - $target: this.$document, - onEdit: this.onEdit.bind(this), - }); - - this.timer = null; - } - - onEdit(document) { - if (this.timer !== null) { - clearTimeout(this.timer); - } - this.timer = setTimeout(async () => { - setItem(`temp-post-${document.id}`, { - ...document, - tempSaveDate: new Date(), - }); - - const updatedDocument = await updateDocument(document.id, document); - this.setState({ - ...this.state, - ...updatedDocument, - }); - }, 1000); - } - - setState(nextState) { +export default class NotionDocument extends Component { + setup() { this.state = { - ...this.state, - ...nextState, + isVisible: false, }; + } - const { id, title, content } = this.state; - this.$editor.setState({ id, title, content }); - - this.render(); + template() { + return ` +
    +
    +
    + `; } render() { + super.render(); + const { isVisible } = this.state; + this.$target.style.visibility = isVisible ? 'visible' : 'hidden'; + } + + mount() { + const { documentData, onEdit } = this.props; + const $editor = this.$target.querySelector('.notion-editor-wrapper'); - this.$document.style.visibility = isVisible ? 'visible' : 'hidden'; + this.$editor = new NotionEditor($editor, { + documentData, + onEdit, + }); } } diff --git a/src/components/NotionSidebar/NotionSidebar.css b/src/components/NotionSidebar/NotionSidebar.css index ff55e9ed..7d870c56 100644 --- a/src/components/NotionSidebar/NotionSidebar.css +++ b/src/components/NotionSidebar/NotionSidebar.css @@ -1,7 +1,7 @@ .notion-sidebar { display: flex; flex-direction: column; - width: 250px; + width: 100%; height: 100%; padding: 20px 0; border-right: 1px solid #f5f6f5; diff --git a/src/components/NotionSidebar/NotionSidebar.js b/src/components/NotionSidebar/NotionSidebar.js index cca1b03d..18238015 100644 --- a/src/components/NotionSidebar/NotionSidebar.js +++ b/src/components/NotionSidebar/NotionSidebar.js @@ -2,27 +2,39 @@ import { createDocument } from '@api/document'; import { history } from '@utils/router'; +import Component from '@core/Component'; + import Button from '@components/Button/Button'; import DocumentList from '@components/DocumentList/DocumentList'; import './NotionSidebar.css'; -export default class NotionSidebar { - constructor({ $target }) { - this.$sidebar = document.createElement('nav'); - this.$sidebar.className = 'notion-sidebar'; +export default class NotionSidebar extends Component { + template() { + return ` + + `; + } - $target.appendChild(this.$sidebar); + mount() { + const { documentList } = this.props; + const $createButtonWrapper = this.$target.querySelector( + '.create-button-wrapper' + ); + const $documentList = this.$target.querySelector('.document-list'); - this.$createButton = new Button({ - $target: this.$sidebar, + this.$createButton = new Button($createButtonWrapper, { className: 'document-create-button', textContent: 'add a document', onClick: () => { this.handleCreateButtonClick(); }, }); - this.$documentList = new DocumentList({ $target: this.$sidebar }); + + this.$documentList = new DocumentList($documentList, { documentList }); } handleCreateButtonClick = async () => { @@ -31,9 +43,9 @@ export default class NotionSidebar { }; setState(nextState) { - this.state = nextState; + super.setState(nextState); - const { documentList } = this.state; + const { documentList } = this.props; this.$documentList.setState({ documentList }); } } diff --git a/src/core/Component.js b/src/core/Component.js new file mode 100644 index 00000000..a02fb499 --- /dev/null +++ b/src/core/Component.js @@ -0,0 +1,40 @@ +export default class Component { + constructor($target, props) { + this.$target = $target; + this.props = props; + + this.setup(); + this.setEvent(); + this.render(); + this.fetchData(); + } + + setup() {} + + mount() {} + + template() { + return ''; + } + + render() { + this.$target.innerHTML = this.template(); + this.mount(); + } + + setEvent() {} + + setState(newState) { + this.state = { ...this.state, ...newState }; + this.render(); + } + + addEvent(eventType, selector, callback) { + this.$target.addEventListener(eventType, (event) => { + if (!event.target.closest(selector)) return; + callback(event); + }); + } + + fetchData() {} +} diff --git a/src/main.js b/src/main.js index 1c7db75c..f60ab540 100644 --- a/src/main.js +++ b/src/main.js @@ -1,7 +1,7 @@ import App from './App'; import './styleReset.css'; -const initApp = ($target) => new App({ $target }); +const initApp = ($target) => new App($target); const $app = document.querySelector(`#app`); diff --git a/src/pages/NotionPage.css b/src/pages/NotionPage.css index 4e8d6471..e2442599 100644 --- a/src/pages/NotionPage.css +++ b/src/pages/NotionPage.css @@ -3,3 +3,14 @@ height: 100%; display: flex; } + +.notion-sidebar-wrapper { + width: 250px; + height: 100%; +} + +.notion-document-wrapper { + width: 100%; + height: 100%; + padding: 20px 10px; +} diff --git a/src/pages/NotionPage.js b/src/pages/NotionPage.js index bc11347f..e5ba3367 100644 --- a/src/pages/NotionPage.js +++ b/src/pages/NotionPage.js @@ -1,33 +1,81 @@ -import { getDocument, getDocumentList } from '@api/document'; +import { getDocument, getDocumentList, updateDocument } from '@api/document'; + +import { setItem } from '@utils/localStorage'; + +import Component from '@core/Component'; import NotionDocument from '@components/NotionDocument/NotionDocument'; import NotionSidebar from '@components/NotionSidebar/NotionSidebar'; import './NotionPage.css'; -export default class NotionPage { - constructor({ $target }) { - this.$page = document.createElement('div'); - this.$page.className = 'notion-page'; - $target.appendChild(this.$page); +export default class NotionPage extends Component { + timer = null; - this.$sidebar = new NotionSidebar({ $target: this.$page }); - this.$document = new NotionDocument({ $target: this.$page }); + setup() { + this.state = { + documentList: [], + documentData: {}, + }; } - async setState(nextState) { - this.state = nextState; - + async fetchData() { const documentList = await getDocumentList(); - this.$sidebar.setState({ documentList }); + this.setState({ documentList }); - const { documentId } = this.state; + const { documentId } = this.props; if (documentId === null) { this.$document.setState({ isVisible: false }); return; } + const documentData = await getDocument(documentId); + + this.setState({ documentData }); + this.$document.setState({ isVisible: true }); + } + + template() { + return ` + +
    + `; + } + + mount() { + const { documentList, documentData } = this; + const $sidebar = this.$target.querySelector('.notion-sidebar-wrapper'); + const $document = this.$target.querySelector('.notion-document-wrapper'); + + this.$sidebar = new NotionSidebar($sidebar, { documentList }); + this.$document = new NotionDocument($document, { + documentData, + onEdit: this.handleEdit.bind(this), + }); + } + + handleEdit(document) { + if (this.timer !== null) { + clearTimeout(this.timer); + } + this.timer = setTimeout(async () => { + setItem(`temp-post-${document.id}`, { + ...document, + tempSaveDate: new Date(), + }); + + const updatedDocument = await updateDocument(document.id, document); + this.setState({ documentData: updatedDocument }); + this.$document.setState({ isVisible: true }); + }, 1000); + } + + get documentList() { + const { documentList } = this.state; + return documentList; + } - const document = await getDocument(documentId); - this.$document.setState({ ...document, isVisible: true }); + get documentData() { + const { documentData } = this.state; + return documentData; } } diff --git a/src/utils/api.js b/src/utils/api.js index 4b5b56c5..e2b19044 100644 --- a/src/utils/api.js +++ b/src/utils/api.js @@ -1,7 +1,7 @@ const API_END_POINT = 'https://kdt-frontend.programmers.co.kr'; const getHeader = () => { - const username = 'roto'; + const username = 'kutta97'; return { 'Content-Type': 'application/json', 'x-username': username, diff --git a/webpack.config.js b/webpack.config.js index ca93b209..d6f95cc4 100644 --- a/webpack.config.js +++ b/webpack.config.js @@ -13,6 +13,7 @@ module.exports = (_, argv) => { '@api': path.resolve(__dirname, 'src/api'), '@components': path.resolve(__dirname, 'src/components'), '@consts': path.resolve(__dirname, 'src/consts'), + '@core': path.resolve(__dirname, 'src/core'), '@pages': path.resolve(__dirname, 'src/pages'), '@utils': path.resolve(__dirname, 'src/utils'), }, From 0d9d624e9e58f1f10525dd3481a5240742d53212 Mon Sep 17 00:00:00 2001 From: kutta97 Date: Mon, 3 Jul 2023 06:16:48 +0900 Subject: [PATCH 36/87] =?UTF-8?q?revert:=20"refactor:=20Component=20core?= =?UTF-8?q?=20class=20=EC=A0=81=EC=9A=A9"?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit This reverts commit 823d96bd33be4063c795ffa267a6af3ffa629ac3. --- .prettierrc.js | 1 - babel.config.js | 1 - jsconfig.json | 5 +- src/App.js | 31 ++------ src/components/Button/Button.js | 26 +++---- src/components/DocumentList/DocumentList.js | 58 +++++++------- .../Editor/Content/NotionEditorContent.js | 29 ++++--- src/components/Editor/NotionEditor.css | 9 +-- src/components/Editor/NotionEditor.js | 67 +++++++--------- .../Editor/Title/NotionEditorTitle.css | 1 - .../Editor/Title/NotionEditorTitle.js | 34 ++++---- .../NotionDocument/NotionDocument.css | 7 -- .../NotionDocument/NotionDocument.js | 67 ++++++++++------ .../NotionSidebar/NotionSidebar.css | 2 +- src/components/NotionSidebar/NotionSidebar.js | 32 +++----- src/core/Component.js | 40 ---------- src/main.js | 2 +- src/pages/NotionPage.css | 11 --- src/pages/NotionPage.js | 78 ++++--------------- src/utils/api.js | 2 +- webpack.config.js | 1 - 21 files changed, 191 insertions(+), 313 deletions(-) delete mode 100644 src/core/Component.js diff --git a/.prettierrc.js b/.prettierrc.js index caae71cd..073fdfec 100644 --- a/.prettierrc.js +++ b/.prettierrc.js @@ -8,7 +8,6 @@ module.exports = { '^@consts/(.*)$', '^@api/(.*)$', '^@utils/(.*)$', - '^@core/(.*)$', '^@components/(.*)$', '^@pages/(.*)$', '^[./]', diff --git a/babel.config.js b/babel.config.js index 6191f2dd..a4556599 100644 --- a/babel.config.js +++ b/babel.config.js @@ -7,7 +7,6 @@ module.exports = { root: ['./src'], alias: { '@api': './src/api', - '@core': './src/core', '@components': './src/components', '@pages': './src/pages', '@consts': './src/consts', diff --git a/jsconfig.json b/jsconfig.json index ac9e5033..b4784344 100644 --- a/jsconfig.json +++ b/jsconfig.json @@ -2,12 +2,11 @@ "compilerOptions": { "baseUrl": ".", "paths": { - "@consts/*": ["./src/consts/*"], "@api/*": ["./src/api/*"], "@utils/*": ["./src/utils/*"], - "@core/*": ["./src/core/*"], "@components/*": ["./src/components/*"], - "@pages/*": ["./src/pages/*"] + "@pages/*": ["./src/pages/*"], + "@consts/*": ["./src/consts/*"] } } } diff --git a/src/App.js b/src/App.js index 88f79187..48011fdd 100644 --- a/src/App.js +++ b/src/App.js @@ -1,47 +1,28 @@ import Pathname from '@utils/pathname'; import { initRouter } from '@utils/router'; -import Component from '@core/Component'; - import NotionPage from '@pages/NotionPage'; import './App.css'; -export default class App extends Component { - constructor($target, props) { - super($target, props); +export default class App { + constructor({ $target }) { + this.$notionPage = new NotionPage({ $target }); initRouter(this.route.bind(this)); this.route(); } - template() { - return ` -
    - `; - } - - mount() { - const { documentId } = this; - const $notionPage = this.$target.querySelector('.notion-page'); - this.$notionPage = new NotionPage($notionPage, { documentId }); - } - - get documentId() { - const { pathname } = window.location; - const [, , documentId] = pathname.split('/'); - return documentId ?? null; - } - route() { const { pathname } = window.location; if (Pathname.isRoot(pathname)) { - return this.setState(); + return this.$notionPage.setState({ documentId: null }); } if (Pathname.isDocument(pathname)) { - return this.setState(); + const [, , documentId] = pathname.split('/'); + return this.$notionPage.setState({ documentId }); } return null; diff --git a/src/components/Button/Button.js b/src/components/Button/Button.js index 8ef92dd6..18cfa50b 100644 --- a/src/components/Button/Button.js +++ b/src/components/Button/Button.js @@ -1,20 +1,18 @@ -import Component from '@core/Component'; +export default class Button { + constructor({ $target, className, textContent, onClick }) { + this.$createButton = document.createElement('button'); -export default class Button extends Component { - template() { - const { className, textContent } = this.props; - return ` - - `; - } + this.$createButton.className = className; + this.$createButton.textContent = textContent; + + $target.appendChild(this.$createButton); - setEvent() { - const { className, onClick } = this.props; + this.setEvent(onClick); + } - this.addEvent('click', `.${className}`, () => { - onClick(); + setEvent(callback) { + this.$createButton.addEventListener('click', () => { + callback?.(); }); } } diff --git a/src/components/DocumentList/DocumentList.js b/src/components/DocumentList/DocumentList.js index b1735fe7..dfc0e8c6 100644 --- a/src/components/DocumentList/DocumentList.js +++ b/src/components/DocumentList/DocumentList.js @@ -2,34 +2,34 @@ import { createDocument, deleteDocument } from '@api/document'; import { history } from '@utils/router'; -import Component from '@core/Component'; - import DocumentListItem from '@components/DocumentList/Item/DocumentListItem'; import './DocumentList.css'; -export default class DocumentList extends Component { - setup() { - this.state = { - documentList: [], - }; - } +export default class DocumentList { + constructor({ $target }) { + this.$documentList = document.createElement('ul'); + this.$documentList.className = 'document-list'; - render() { - super.render(); + $target.appendChild(this.$documentList); - const { documentList } = this.props; - if (documentList.length === 0) return; - - documentList.forEach((document) => { - const $documentItem = this.createDocumentItem(document); - this.$target.appendChild($documentItem); - }); + this.setEvent(); } + handleCreateIndsideButtonClick = async (id) => { + const newDocument = await createDocument({ title: 'Untitled', parent: id }); + history.push(`/documents/${newDocument.id}`); + }; + + hanldeDeleteButtonClick = async (id) => { + await deleteDocument(id); + history.push('/'); + }; + setEvent() { - this.addEvent('click', 'li', ({ target }) => { + this.$documentList.addEventListener('click', ({ target }) => { const $li = target.closest('li'); + if (!$li) return; const { id } = $li.dataset; const { className } = target; @@ -47,15 +47,10 @@ export default class DocumentList extends Component { }); } - handleCreateIndsideButtonClick = async (id) => { - const newDocument = await createDocument({ title: 'Untitled', parent: id }); - history.push(`/documents/${newDocument.id}`); - }; - - hanldeDeleteButtonClick = async (id) => { - await deleteDocument(id); - history.push('/'); - }; + setState(nextState) { + this.state = nextState; + this.render(); + } createDocumentItem(item, depth = 1) { const $div = document.createElement('div'); @@ -78,4 +73,13 @@ export default class DocumentList extends Component { return $div; } + + render() { + const { documentList } = this.state; + this.$documentList.innerHTML = ''; + documentList.forEach((document) => { + const $documentItem = this.createDocumentItem(document); + this.$documentList.appendChild($documentItem); + }); + } } diff --git a/src/components/Editor/Content/NotionEditorContent.js b/src/components/Editor/Content/NotionEditorContent.js index 5c75ba0b..c69ed29b 100644 --- a/src/components/Editor/Content/NotionEditorContent.js +++ b/src/components/Editor/Content/NotionEditorContent.js @@ -1,16 +1,23 @@ -import Component from '@core/Component'; - import './NotionEditorContent.css'; -export default class NotionEditorContent extends Component { - template() { - const { content } = this.props; +export default class NotionEditorContent { + constructor({ $target }) { + this.$content = document.createElement('textarea'); + + this.$content.className = 'notion-editor-content'; + this.$content.name = 'content'; + + $target.appendChild(this.$content); + } + + setState(nextState) { + this.state = nextState; + this.render(); + } + + render() { + const { content } = this.state; - return ` - - `; + this.$content.value = content; } } diff --git a/src/components/Editor/NotionEditor.css b/src/components/Editor/NotionEditor.css index 53765b0b..7f0c744e 100644 --- a/src/components/Editor/NotionEditor.css +++ b/src/components/Editor/NotionEditor.css @@ -1,8 +1,7 @@ -.notion-editor-title-wrapper { - width: 100%; -} - -.notion-editor-content-wrapper { +.notion-editor { + display: flex; + flex-direction: column; width: 100%; height: 100%; + padding: 20px 10px; } diff --git a/src/components/Editor/NotionEditor.js b/src/components/Editor/NotionEditor.js index d896ba38..aca7aec5 100644 --- a/src/components/Editor/NotionEditor.js +++ b/src/components/Editor/NotionEditor.js @@ -1,57 +1,42 @@ -import Component from '@core/Component'; - import NotionEditorTitle from '@components//Editor/Title/NotionEditorTitle'; import NotionEditorContent from '@components/Editor/Content/NotionEditorContent'; import './NotionEditor.css'; -export default class NotionEditor extends Component { - template() { - return ` -
    -
    - `; - } - - mount() { - const { title, content } = this; - const $titleEditor = this.$target.querySelector( - '.notion-editor-title-wrapper' - ); - const $contentEditor = this.$target.querySelector( - '.notion-editor-content-wrapper' - ); - - this.$titleEditor = new NotionEditorTitle($titleEditor, { title }); - this.$contentEditor = new NotionEditorContent($contentEditor, { content }); - } +export default class NotionEditor { + constructor({ $target, onEdit }) { + this.$editor = document.createElement('section'); + this.$editor.className = 'notion-editor'; + $target.appendChild(this.$editor); - get title() { - const { documentData } = this.props; - const { title } = documentData; + this.$title = new NotionEditorTitle({ $target: this.$editor }); + this.$content = new NotionEditorContent({ $target: this.$editor }); - return title; + this.setEvent(onEdit); } - get content() { - const { documentData } = this.props; - const { content } = documentData; + getNextStateWithTarget(target) { + const name = target.getAttribute('name'); + const { value } = target; - return content; + return { + ...this.state, + [name]: value, + }; } - setEvent() { - const { onEdit } = this.props; + setEvent(callback) { + this.$editor.addEventListener('input', (e) => { + const nextState = this.getNextStateWithTarget(e.target); + callback?.(nextState); + }); + } - this.$target.addEventListener('input', ({ target }) => { - const { name } = target; - const { documentData } = this.props; + setState(nextState) { + this.state = nextState; - if (documentData[name] === undefined) return; - onEdit({ - ...documentData, - [name]: target.value, - }); - }); + const { title, content } = this.state; + this.$title.setState({ title }); + this.$content.setState({ content }); } } diff --git a/src/components/Editor/Title/NotionEditorTitle.css b/src/components/Editor/Title/NotionEditorTitle.css index e5b9372f..81dc0ad3 100644 --- a/src/components/Editor/Title/NotionEditorTitle.css +++ b/src/components/Editor/Title/NotionEditorTitle.css @@ -1,5 +1,4 @@ .notion-editor-title { - width: 100%; font-size: 25px; padding: 5px 10px; } diff --git a/src/components/Editor/Title/NotionEditorTitle.js b/src/components/Editor/Title/NotionEditorTitle.js index 6bc6d97f..f350824f 100644 --- a/src/components/Editor/Title/NotionEditorTitle.js +++ b/src/components/Editor/Title/NotionEditorTitle.js @@ -1,19 +1,25 @@ -import Component from '@core/Component'; - import './NotionEditorTitle.css'; -export default class NotionEditorTitle extends Component { - template() { - const { title } = this.props; +export default class NotionEditorTitle { + constructor({ $target }) { + this.$title = document.createElement('input'); + + this.$title.className = 'notion-editor-title'; + this.$title.type = 'text'; + this.$title.name = 'title'; + this.$title.placeholder = 'undefined'; + + $target.appendChild(this.$title); + } + + setState(nextState) { + this.state = nextState; + this.render(); + } + + render() { + const { title } = this.state; - return ` - - `; + this.$title.value = title ?? ''; } } diff --git a/src/components/NotionDocument/NotionDocument.css b/src/components/NotionDocument/NotionDocument.css index 4d49d235..8060d48c 100644 --- a/src/components/NotionDocument/NotionDocument.css +++ b/src/components/NotionDocument/NotionDocument.css @@ -1,10 +1,3 @@ .notion-document { width: 100%; } - -.notion-editor-wrapper { - display: flex; - flex-direction: column; - width: 100%; - height: 100%; -} diff --git a/src/components/NotionDocument/NotionDocument.js b/src/components/NotionDocument/NotionDocument.js index 999de423..9da2bbb9 100644 --- a/src/components/NotionDocument/NotionDocument.js +++ b/src/components/NotionDocument/NotionDocument.js @@ -1,38 +1,59 @@ -import Component from '@core/Component'; +import { updateDocument } from '@api/document'; + +import { setItem } from '@utils/localStorage'; import NotionEditor from '@components/Editor/NotionEditor'; import './NotionDocument.css'; -export default class NotionDocument extends Component { - setup() { - this.state = { - isVisible: false, - }; +export default class NotionDocument { + constructor({ $target }) { + this.$document = document.createElement('div'); + this.$document.className = 'notion-document'; + + $target.appendChild(this.$document); + + this.$editor = new NotionEditor({ + $target: this.$document, + onEdit: this.onEdit.bind(this), + }); + + this.timer = null; } - template() { - return ` -
    -
    -
    - `; + onEdit(document) { + if (this.timer !== null) { + clearTimeout(this.timer); + } + this.timer = setTimeout(async () => { + setItem(`temp-post-${document.id}`, { + ...document, + tempSaveDate: new Date(), + }); + + const updatedDocument = await updateDocument(document.id, document); + this.setState({ + ...this.state, + ...updatedDocument, + }); + }, 1000); } - render() { - super.render(); + setState(nextState) { + this.state = { + ...this.state, + ...nextState, + }; - const { isVisible } = this.state; - this.$target.style.visibility = isVisible ? 'visible' : 'hidden'; + const { id, title, content } = this.state; + this.$editor.setState({ id, title, content }); + + this.render(); } - mount() { - const { documentData, onEdit } = this.props; - const $editor = this.$target.querySelector('.notion-editor-wrapper'); + render() { + const { isVisible } = this.state; - this.$editor = new NotionEditor($editor, { - documentData, - onEdit, - }); + this.$document.style.visibility = isVisible ? 'visible' : 'hidden'; } } diff --git a/src/components/NotionSidebar/NotionSidebar.css b/src/components/NotionSidebar/NotionSidebar.css index 7d870c56..ff55e9ed 100644 --- a/src/components/NotionSidebar/NotionSidebar.css +++ b/src/components/NotionSidebar/NotionSidebar.css @@ -1,7 +1,7 @@ .notion-sidebar { display: flex; flex-direction: column; - width: 100%; + width: 250px; height: 100%; padding: 20px 0; border-right: 1px solid #f5f6f5; diff --git a/src/components/NotionSidebar/NotionSidebar.js b/src/components/NotionSidebar/NotionSidebar.js index 18238015..cca1b03d 100644 --- a/src/components/NotionSidebar/NotionSidebar.js +++ b/src/components/NotionSidebar/NotionSidebar.js @@ -2,39 +2,27 @@ import { createDocument } from '@api/document'; import { history } from '@utils/router'; -import Component from '@core/Component'; - import Button from '@components/Button/Button'; import DocumentList from '@components/DocumentList/DocumentList'; import './NotionSidebar.css'; -export default class NotionSidebar extends Component { - template() { - return ` - - `; - } +export default class NotionSidebar { + constructor({ $target }) { + this.$sidebar = document.createElement('nav'); + this.$sidebar.className = 'notion-sidebar'; - mount() { - const { documentList } = this.props; - const $createButtonWrapper = this.$target.querySelector( - '.create-button-wrapper' - ); - const $documentList = this.$target.querySelector('.document-list'); + $target.appendChild(this.$sidebar); - this.$createButton = new Button($createButtonWrapper, { + this.$createButton = new Button({ + $target: this.$sidebar, className: 'document-create-button', textContent: 'add a document', onClick: () => { this.handleCreateButtonClick(); }, }); - - this.$documentList = new DocumentList($documentList, { documentList }); + this.$documentList = new DocumentList({ $target: this.$sidebar }); } handleCreateButtonClick = async () => { @@ -43,9 +31,9 @@ export default class NotionSidebar extends Component { }; setState(nextState) { - super.setState(nextState); + this.state = nextState; - const { documentList } = this.props; + const { documentList } = this.state; this.$documentList.setState({ documentList }); } } diff --git a/src/core/Component.js b/src/core/Component.js deleted file mode 100644 index a02fb499..00000000 --- a/src/core/Component.js +++ /dev/null @@ -1,40 +0,0 @@ -export default class Component { - constructor($target, props) { - this.$target = $target; - this.props = props; - - this.setup(); - this.setEvent(); - this.render(); - this.fetchData(); - } - - setup() {} - - mount() {} - - template() { - return ''; - } - - render() { - this.$target.innerHTML = this.template(); - this.mount(); - } - - setEvent() {} - - setState(newState) { - this.state = { ...this.state, ...newState }; - this.render(); - } - - addEvent(eventType, selector, callback) { - this.$target.addEventListener(eventType, (event) => { - if (!event.target.closest(selector)) return; - callback(event); - }); - } - - fetchData() {} -} diff --git a/src/main.js b/src/main.js index f60ab540..1c7db75c 100644 --- a/src/main.js +++ b/src/main.js @@ -1,7 +1,7 @@ import App from './App'; import './styleReset.css'; -const initApp = ($target) => new App($target); +const initApp = ($target) => new App({ $target }); const $app = document.querySelector(`#app`); diff --git a/src/pages/NotionPage.css b/src/pages/NotionPage.css index e2442599..4e8d6471 100644 --- a/src/pages/NotionPage.css +++ b/src/pages/NotionPage.css @@ -3,14 +3,3 @@ height: 100%; display: flex; } - -.notion-sidebar-wrapper { - width: 250px; - height: 100%; -} - -.notion-document-wrapper { - width: 100%; - height: 100%; - padding: 20px 10px; -} diff --git a/src/pages/NotionPage.js b/src/pages/NotionPage.js index e5ba3367..bc11347f 100644 --- a/src/pages/NotionPage.js +++ b/src/pages/NotionPage.js @@ -1,81 +1,33 @@ -import { getDocument, getDocumentList, updateDocument } from '@api/document'; - -import { setItem } from '@utils/localStorage'; - -import Component from '@core/Component'; +import { getDocument, getDocumentList } from '@api/document'; import NotionDocument from '@components/NotionDocument/NotionDocument'; import NotionSidebar from '@components/NotionSidebar/NotionSidebar'; import './NotionPage.css'; -export default class NotionPage extends Component { - timer = null; +export default class NotionPage { + constructor({ $target }) { + this.$page = document.createElement('div'); + this.$page.className = 'notion-page'; + $target.appendChild(this.$page); - setup() { - this.state = { - documentList: [], - documentData: {}, - }; + this.$sidebar = new NotionSidebar({ $target: this.$page }); + this.$document = new NotionDocument({ $target: this.$page }); } - async fetchData() { + async setState(nextState) { + this.state = nextState; + const documentList = await getDocumentList(); - this.setState({ documentList }); + this.$sidebar.setState({ documentList }); - const { documentId } = this.props; + const { documentId } = this.state; if (documentId === null) { this.$document.setState({ isVisible: false }); return; } - const documentData = await getDocument(documentId); - - this.setState({ documentData }); - this.$document.setState({ isVisible: true }); - } - - template() { - return ` - -
    - `; - } - - mount() { - const { documentList, documentData } = this; - const $sidebar = this.$target.querySelector('.notion-sidebar-wrapper'); - const $document = this.$target.querySelector('.notion-document-wrapper'); - - this.$sidebar = new NotionSidebar($sidebar, { documentList }); - this.$document = new NotionDocument($document, { - documentData, - onEdit: this.handleEdit.bind(this), - }); - } - - handleEdit(document) { - if (this.timer !== null) { - clearTimeout(this.timer); - } - this.timer = setTimeout(async () => { - setItem(`temp-post-${document.id}`, { - ...document, - tempSaveDate: new Date(), - }); - - const updatedDocument = await updateDocument(document.id, document); - this.setState({ documentData: updatedDocument }); - this.$document.setState({ isVisible: true }); - }, 1000); - } - - get documentList() { - const { documentList } = this.state; - return documentList; - } - get documentData() { - const { documentData } = this.state; - return documentData; + const document = await getDocument(documentId); + this.$document.setState({ ...document, isVisible: true }); } } diff --git a/src/utils/api.js b/src/utils/api.js index e2b19044..4b5b56c5 100644 --- a/src/utils/api.js +++ b/src/utils/api.js @@ -1,7 +1,7 @@ const API_END_POINT = 'https://kdt-frontend.programmers.co.kr'; const getHeader = () => { - const username = 'kutta97'; + const username = 'roto'; return { 'Content-Type': 'application/json', 'x-username': username, diff --git a/webpack.config.js b/webpack.config.js index d6f95cc4..ca93b209 100644 --- a/webpack.config.js +++ b/webpack.config.js @@ -13,7 +13,6 @@ module.exports = (_, argv) => { '@api': path.resolve(__dirname, 'src/api'), '@components': path.resolve(__dirname, 'src/components'), '@consts': path.resolve(__dirname, 'src/consts'), - '@core': path.resolve(__dirname, 'src/core'), '@pages': path.resolve(__dirname, 'src/pages'), '@utils': path.resolve(__dirname, 'src/utils'), }, From d62584ae0868593f6eb6b0ce7a76014444d3e4ee Mon Sep 17 00:00:00 2001 From: kutta97 Date: Mon, 3 Jul 2023 07:14:03 +0900 Subject: [PATCH 37/87] =?UTF-8?q?chore:=20path=20alias=20=EC=97=90=20core?= =?UTF-8?q?=20=EB=94=94=EB=A0=89=ED=86=A0=EB=A6=AC=20=EC=B6=94=EA=B0=80?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .prettierrc.js | 1 + babel.config.js | 3 ++- jsconfig.json | 5 +++-- webpack.config.js | 1 + 4 files changed, 7 insertions(+), 3 deletions(-) diff --git a/.prettierrc.js b/.prettierrc.js index 073fdfec..caae71cd 100644 --- a/.prettierrc.js +++ b/.prettierrc.js @@ -8,6 +8,7 @@ module.exports = { '^@consts/(.*)$', '^@api/(.*)$', '^@utils/(.*)$', + '^@core/(.*)$', '^@components/(.*)$', '^@pages/(.*)$', '^[./]', diff --git a/babel.config.js b/babel.config.js index a4556599..9757ddce 100644 --- a/babel.config.js +++ b/babel.config.js @@ -8,8 +8,9 @@ module.exports = { alias: { '@api': './src/api', '@components': './src/components', - '@pages': './src/pages', '@consts': './src/consts', + '@core': './src/core', + '@pages': './src/pages', '@utils': './src/utils', }, }, diff --git a/jsconfig.json b/jsconfig.json index b4784344..ac9e5033 100644 --- a/jsconfig.json +++ b/jsconfig.json @@ -2,11 +2,12 @@ "compilerOptions": { "baseUrl": ".", "paths": { + "@consts/*": ["./src/consts/*"], "@api/*": ["./src/api/*"], "@utils/*": ["./src/utils/*"], + "@core/*": ["./src/core/*"], "@components/*": ["./src/components/*"], - "@pages/*": ["./src/pages/*"], - "@consts/*": ["./src/consts/*"] + "@pages/*": ["./src/pages/*"] } } } diff --git a/webpack.config.js b/webpack.config.js index ca93b209..d6f95cc4 100644 --- a/webpack.config.js +++ b/webpack.config.js @@ -13,6 +13,7 @@ module.exports = (_, argv) => { '@api': path.resolve(__dirname, 'src/api'), '@components': path.resolve(__dirname, 'src/components'), '@consts': path.resolve(__dirname, 'src/consts'), + '@core': path.resolve(__dirname, 'src/core'), '@pages': path.resolve(__dirname, 'src/pages'), '@utils': path.resolve(__dirname, 'src/utils'), }, From 034b0e9f31f6c4f39e2f41b053b89c3c23186f6e Mon Sep 17 00:00:00 2001 From: kutta97 Date: Mon, 3 Jul 2023 16:24:07 +0900 Subject: [PATCH 38/87] =?UTF-8?q?refactor:=20=EB=AA=A8=EB=93=A0=20?= =?UTF-8?q?=EC=BB=B4=ED=8F=AC=EB=84=8C=ED=8A=B8=EA=B0=80=20coreComponent?= =?UTF-8?q?=20=EB=A5=BC=20=EC=83=81=EC=86=8D=ED=95=98=EB=8F=84=EB=A1=9D=20?= =?UTF-8?q?=EC=88=98=EC=A0=95=20=20=20-=20=EB=AA=A8=EB=93=A0=20=EC=BB=B4?= =?UTF-8?q?=ED=8F=AC=EB=84=8C=ED=8A=B8=EA=B0=80=20=EB=8F=99=EC=9D=BC?= =?UTF-8?q?=ED=95=9C=20=EC=83=9D=EB=AA=85=EC=A3=BC=EA=B8=B0=EB=A5=BC=20?= =?UTF-8?q?=EA=B0=80=EC=A7=80=EB=8F=84=EB=A1=9D=20=EB=A6=AC=ED=8C=A9?= =?UTF-8?q?=ED=86=A0=EB=A7=81?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .eslintrc.js | 1 + src/App.js | 31 ++++---- src/components/Button/Button.js | 16 +++-- src/components/DocumentList/DocumentList.js | 25 +++---- .../DocumentList/Item/DocumentListItem.js | 15 ++-- .../Editor/Content/NotionEditorContent.js | 19 ++--- src/components/Editor/NotionEditor.js | 42 ++++++----- .../Editor/Title/NotionEditorTitle.js | 19 ++--- .../NotionDocument/NotionDocument.js | 66 ++++++++--------- src/components/NotionSidebar/NotionSidebar.js | 24 +++++-- src/core/Component.js | 31 ++++++++ src/main.js | 2 +- src/model/NotionModel.js | 53 ++++++++++++++ src/pages/NotionPage.js | 72 +++++++++++++++---- 14 files changed, 278 insertions(+), 138 deletions(-) create mode 100644 src/core/Component.js create mode 100644 src/model/NotionModel.js diff --git a/.eslintrc.js b/.eslintrc.js index 1525140e..da3bfb07 100644 --- a/.eslintrc.js +++ b/.eslintrc.js @@ -12,5 +12,6 @@ module.exports = { rules: { 'import/no-unresolved': 'off', 'class-methods-use-this': 'off', + 'no-new': 'off', }, }; diff --git a/src/App.js b/src/App.js index 48011fdd..0e5c1bd0 100644 --- a/src/App.js +++ b/src/App.js @@ -1,30 +1,33 @@ -import Pathname from '@utils/pathname'; import { initRouter } from '@utils/router'; +import Component from '@core/Component'; + import NotionPage from '@pages/NotionPage'; import './App.css'; -export default class App { - constructor({ $target }) { - this.$notionPage = new NotionPage({ $target }); +export default class App extends Component { + constructor($target) { + super($target); initRouter(this.route.bind(this)); this.route(); } - route() { - const { pathname } = window.location; + initChildComponents() { + this.$notionPage = new NotionPage(this.$target, { + getDocumentId: this.getDocumentId.bind(this), + }); + } - if (Pathname.isRoot(pathname)) { - return this.$notionPage.setState({ documentId: null }); - } + getDocumentId() { + const { pathname } = window.location; + const [, , documentId] = pathname.split('/'); - if (Pathname.isDocument(pathname)) { - const [, , documentId] = pathname.split('/'); - return this.$notionPage.setState({ documentId }); - } + return documentId ?? null; + } - return null; + route() { + this.$notionPage.setState(); } } diff --git a/src/components/Button/Button.js b/src/components/Button/Button.js index 18cfa50b..924c417d 100644 --- a/src/components/Button/Button.js +++ b/src/components/Button/Button.js @@ -1,18 +1,20 @@ -export default class Button { - constructor({ $target, className, textContent, onClick }) { +import Component from '@core/Component'; + +export default class Button extends Component { + initComponent() { + const { className, textContent } = this.props; this.$createButton = document.createElement('button'); this.$createButton.className = className; this.$createButton.textContent = textContent; - $target.appendChild(this.$createButton); - - this.setEvent(onClick); + this.$target.appendChild(this.$createButton); } - setEvent(callback) { + setEvent() { + const { onClick } = this.props; this.$createButton.addEventListener('click', () => { - callback?.(); + onClick(); }); } } diff --git a/src/components/DocumentList/DocumentList.js b/src/components/DocumentList/DocumentList.js index dfc0e8c6..9ea94201 100644 --- a/src/components/DocumentList/DocumentList.js +++ b/src/components/DocumentList/DocumentList.js @@ -2,18 +2,24 @@ import { createDocument, deleteDocument } from '@api/document'; import { history } from '@utils/router'; +import Component from '@core/Component'; + import DocumentListItem from '@components/DocumentList/Item/DocumentListItem'; import './DocumentList.css'; -export default class DocumentList { - constructor({ $target }) { +export default class DocumentList extends Component { + setup() { + this.state = { + documentList: [], + }; + } + + initComponent() { this.$documentList = document.createElement('ul'); this.$documentList.className = 'document-list'; - $target.appendChild(this.$documentList); - - this.setEvent(); + this.$target.appendChild(this.$documentList); } handleCreateIndsideButtonClick = async (id) => { @@ -47,16 +53,10 @@ export default class DocumentList { }); } - setState(nextState) { - this.state = nextState; - this.render(); - } - createDocumentItem(item, depth = 1) { const $div = document.createElement('div'); - const $documentItem = new DocumentListItem({ $target: $div }); - $documentItem.setState({ + new DocumentListItem($div, { documentItem: item, depth, }); @@ -76,6 +76,7 @@ export default class DocumentList { render() { const { documentList } = this.state; + this.$documentList.innerHTML = ''; documentList.forEach((document) => { const $documentItem = this.createDocumentItem(document); diff --git a/src/components/DocumentList/Item/DocumentListItem.js b/src/components/DocumentList/Item/DocumentListItem.js index a1c37ed4..5c51b517 100644 --- a/src/components/DocumentList/Item/DocumentListItem.js +++ b/src/components/DocumentList/Item/DocumentListItem.js @@ -1,19 +1,16 @@ +import Component from '@core/Component'; + import './DocumentListItem.css'; -export default class DocumentListItem { - constructor({ $target }) { +export default class DocumentListItem extends Component { + initComponent() { this.$documentListItem = document.createElement('li'); - $target.appendChild(this.$documentListItem); - } - - setState(nextState) { - this.state = nextState; - this.render(); + this.$target.appendChild(this.$documentListItem); } render() { - const { documentItem, depth } = this.state; + const { documentItem, depth } = this.props; this.$documentListItem.dataset.id = documentItem.id; this.$documentListItem.innerHTML = ` diff --git a/src/components/Editor/Content/NotionEditorContent.js b/src/components/Editor/Content/NotionEditorContent.js index c69ed29b..8ff26e66 100644 --- a/src/components/Editor/Content/NotionEditorContent.js +++ b/src/components/Editor/Content/NotionEditorContent.js @@ -1,18 +1,21 @@ +import Component from '@core/Component'; + import './NotionEditorContent.css'; -export default class NotionEditorContent { - constructor({ $target }) { +export default class NotionEditorContent extends Component { + setup() { + this.state = { + content: '', + }; + } + + initComponent() { this.$content = document.createElement('textarea'); this.$content.className = 'notion-editor-content'; this.$content.name = 'content'; - $target.appendChild(this.$content); - } - - setState(nextState) { - this.state = nextState; - this.render(); + this.$target.appendChild(this.$content); } render() { diff --git a/src/components/Editor/NotionEditor.js b/src/components/Editor/NotionEditor.js index aca7aec5..88f466e1 100644 --- a/src/components/Editor/NotionEditor.js +++ b/src/components/Editor/NotionEditor.js @@ -1,39 +1,37 @@ +import Component from '@core/Component'; + import NotionEditorTitle from '@components//Editor/Title/NotionEditorTitle'; import NotionEditorContent from '@components/Editor/Content/NotionEditorContent'; import './NotionEditor.css'; -export default class NotionEditor { - constructor({ $target, onEdit }) { +export default class NotionEditor extends Component { + initComponent() { this.$editor = document.createElement('section'); this.$editor.className = 'notion-editor'; - $target.appendChild(this.$editor); - - this.$title = new NotionEditorTitle({ $target: this.$editor }); - this.$content = new NotionEditorContent({ $target: this.$editor }); - - this.setEvent(onEdit); + this.$target.appendChild(this.$editor); } - getNextStateWithTarget(target) { - const name = target.getAttribute('name'); - const { value } = target; - - return { - ...this.state, - [name]: value, - }; + initChildComponents() { + this.$title = new NotionEditorTitle(this.$editor); + this.$content = new NotionEditorContent(this.$editor); } - setEvent(callback) { - this.$editor.addEventListener('input', (e) => { - const nextState = this.getNextStateWithTarget(e.target); - callback?.(nextState); + setEvent() { + const { onEdit } = this.props; + this.$editor.addEventListener('input', ({ target }) => { + const name = target.getAttribute('name'); + const { value } = target; + + onEdit(name, { + ...this.state, + [name]: value, + }); }); } - setState(nextState) { - this.state = nextState; + setState(newstate) { + super.setState(newstate); const { title, content } = this.state; this.$title.setState({ title }); diff --git a/src/components/Editor/Title/NotionEditorTitle.js b/src/components/Editor/Title/NotionEditorTitle.js index f350824f..0613facc 100644 --- a/src/components/Editor/Title/NotionEditorTitle.js +++ b/src/components/Editor/Title/NotionEditorTitle.js @@ -1,7 +1,15 @@ +import Component from '@core/Component'; + import './NotionEditorTitle.css'; -export default class NotionEditorTitle { - constructor({ $target }) { +export default class NotionEditorTitle extends Component { + setup() { + this.state = { + title: '', + }; + } + + initComponent() { this.$title = document.createElement('input'); this.$title.className = 'notion-editor-title'; @@ -9,12 +17,7 @@ export default class NotionEditorTitle { this.$title.name = 'title'; this.$title.placeholder = 'undefined'; - $target.appendChild(this.$title); - } - - setState(nextState) { - this.state = nextState; - this.render(); + this.$target.appendChild(this.$title); } render() { diff --git a/src/components/NotionDocument/NotionDocument.js b/src/components/NotionDocument/NotionDocument.js index 9da2bbb9..5b2b9001 100644 --- a/src/components/NotionDocument/NotionDocument.js +++ b/src/components/NotionDocument/NotionDocument.js @@ -1,58 +1,50 @@ -import { updateDocument } from '@api/document'; - -import { setItem } from '@utils/localStorage'; +import Component from '@core/Component'; import NotionEditor from '@components/Editor/NotionEditor'; import './NotionDocument.css'; -export default class NotionDocument { - constructor({ $target }) { +export default class NotionDocument extends Component { + setup() { + this.state = { + documentData: { + id: this.props.getDocumentId(), + title: '', + createdAt: '', + updatedAt: '', + content: null, + documents: [], + }, + }; + } + + initComponent() { this.$document = document.createElement('div'); this.$document.className = 'notion-document'; - $target.appendChild(this.$document); - - this.$editor = new NotionEditor({ - $target: this.$document, - onEdit: this.onEdit.bind(this), - }); - - this.timer = null; + this.$target.appendChild(this.$document); } - onEdit(document) { - if (this.timer !== null) { - clearTimeout(this.timer); - } - this.timer = setTimeout(async () => { - setItem(`temp-post-${document.id}`, { - ...document, - tempSaveDate: new Date(), - }); - - const updatedDocument = await updateDocument(document.id, document); - this.setState({ - ...this.state, - ...updatedDocument, - }); - }, 1000); + initChildComponents() { + const { onEdit } = this.props; + + this.$editor = new NotionEditor(this.$document, { onEdit }); } setState(nextState) { - this.state = { - ...this.state, - ...nextState, - }; + super.setState(nextState); - const { id, title, content } = this.state; - this.$editor.setState({ id, title, content }); + const { documentData } = this.state; + + if (documentData.id === null) return; - this.render(); + const { id, title, content } = documentData; + this.$editor.setState({ id, title, content }); } render() { - const { isVisible } = this.state; + const { id: documentId } = this.state.documentData; + const isVisible = documentId !== null; this.$document.style.visibility = isVisible ? 'visible' : 'hidden'; } diff --git a/src/components/NotionSidebar/NotionSidebar.js b/src/components/NotionSidebar/NotionSidebar.js index cca1b03d..1fb8b7ee 100644 --- a/src/components/NotionSidebar/NotionSidebar.js +++ b/src/components/NotionSidebar/NotionSidebar.js @@ -2,27 +2,37 @@ import { createDocument } from '@api/document'; import { history } from '@utils/router'; +import Component from '@core/Component'; + import Button from '@components/Button/Button'; import DocumentList from '@components/DocumentList/DocumentList'; import './NotionSidebar.css'; -export default class NotionSidebar { - constructor({ $target }) { +export default class NotionSidebar extends Component { + setup() { + this.state = { + documentList: [], + }; + } + + initComponent() { this.$sidebar = document.createElement('nav'); this.$sidebar.className = 'notion-sidebar'; - $target.appendChild(this.$sidebar); + this.$target.appendChild(this.$sidebar); + } - this.$createButton = new Button({ - $target: this.$sidebar, + initChildComponents() { + this.$createButton = new Button(this.$sidebar, { className: 'document-create-button', textContent: 'add a document', onClick: () => { this.handleCreateButtonClick(); }, }); - this.$documentList = new DocumentList({ $target: this.$sidebar }); + + this.$documentList = new DocumentList(this.$sidebar); } handleCreateButtonClick = async () => { @@ -31,7 +41,7 @@ export default class NotionSidebar { }; setState(nextState) { - this.state = nextState; + super.setState(nextState); const { documentList } = this.state; this.$documentList.setState({ documentList }); diff --git a/src/core/Component.js b/src/core/Component.js new file mode 100644 index 00000000..bb2308b8 --- /dev/null +++ b/src/core/Component.js @@ -0,0 +1,31 @@ +export default class Component { + constructor($target, props) { + this.$target = $target; + this.props = props; + + this.setup(); + this.initComponent(); + this.setEvent(); + this.render(); + + this.initChildComponents(); + } + + setup() {} + + initComponent() {} + + initChildComponents() {} + + setEvent() {} + + render() {} + + setState(newState) { + this.state = { + ...this.state, + ...newState, + }; + this.render(); + } +} diff --git a/src/main.js b/src/main.js index 1c7db75c..f60ab540 100644 --- a/src/main.js +++ b/src/main.js @@ -1,7 +1,7 @@ import App from './App'; import './styleReset.css'; -const initApp = ($target) => new App({ $target }); +const initApp = ($target) => new App($target); const $app = document.querySelector(`#app`); diff --git a/src/model/NotionModel.js b/src/model/NotionModel.js new file mode 100644 index 00000000..93b0da71 --- /dev/null +++ b/src/model/NotionModel.js @@ -0,0 +1,53 @@ +import { createDocument, getDocument, getDocumentList } from '@api/document'; + +export default class NotionModel { + constructor(documentId) { + this.state = { + documentId, + documentList: [], + documentData: {}, + }; + + this.init(); + } + + init() { + this.fetchDocumentList(); + this.fetchDocumentData(); + } + + async createDocument(parent, callback) { + const newDocument = await createDocument({ title: 'Untitled', parent }); + callback?.(newDocument.id); + } + + async fetchDocumentList() { + const documentList = await getDocumentList(); + this.setState({ documentList }); + } + + async fetchDocumentData() { + const { documentId } = this.state; + if (documentId === null) return; + + const documentData = await getDocument(); + this.setState({ documentData }); + } + + get documentList() { + const { documentList } = this.state; + return documentList; + } + + get documentData() { + const { documentData } = this.state; + return documentData; + } + + setState(nextState) { + this.state = { + ...this.state, + ...nextState, + }; + } +} diff --git a/src/pages/NotionPage.js b/src/pages/NotionPage.js index bc11347f..8241d531 100644 --- a/src/pages/NotionPage.js +++ b/src/pages/NotionPage.js @@ -1,33 +1,79 @@ -import { getDocument, getDocumentList } from '@api/document'; +import { getDocument, getDocumentList, updateDocument } from '@api/document'; + +import { setItem } from '@utils/localStorage'; + +import Component from '@core/Component'; import NotionDocument from '@components/NotionDocument/NotionDocument'; import NotionSidebar from '@components/NotionSidebar/NotionSidebar'; import './NotionPage.css'; -export default class NotionPage { - constructor({ $target }) { +export default class NotionPage extends Component { + timer = null; + + initComponent() { this.$page = document.createElement('div'); this.$page.className = 'notion-page'; - $target.appendChild(this.$page); + this.$target.appendChild(this.$page); + } - this.$sidebar = new NotionSidebar({ $target: this.$page }); - this.$document = new NotionDocument({ $target: this.$page }); + initChildComponents() { + const { getDocumentId } = this.props; + this.$sidebar = new NotionSidebar(this.$page, { getDocumentId }); + this.$document = new NotionDocument(this.$page, { + getDocumentId, + onEdit: this.onEdit.bind(this), + }); } - async setState(nextState) { - this.state = nextState; + get documentId() { + const { getDocumentId } = this.props; + return getDocumentId(); + } + + async fetchDocumentList() { + const { documentId } = this; const documentList = await getDocumentList(); - this.$sidebar.setState({ documentList }); + this.$sidebar.setState({ documentId, documentList }); + } + + async fetchDocumentData() { + const { documentId } = this; - const { documentId } = this.state; if (documentId === null) { - this.$document.setState({ isVisible: false }); + this.$document.setState({ documentData: { id: documentId } }); return; } - const document = await getDocument(documentId); - this.$document.setState({ ...document, isVisible: true }); + const documentData = await getDocument(documentId); + this.$document.setState({ documentData }); + } + + onEdit(name, document) { + if (this.timer !== null) { + clearTimeout(this.timer); + } + this.timer = setTimeout(async () => { + setItem(`temp-post-${document.id}`, { + ...document, + tempSaveDate: new Date(), + }); + + const updatedDocument = await updateDocument(document.id, document); + if (name === 'title') { + this.fetchDocumentList(); + } + + this.$document.setState({ documentData: updatedDocument }); + }, 1000); + } + + setState(newState) { + super.setState(newState); + + this.fetchDocumentList(); + this.fetchDocumentData(); } } From 4f6b0ade890e538445abc40f59e7145f15434e41 Mon Sep 17 00:00:00 2001 From: kutta97 Date: Tue, 4 Jul 2023 20:27:14 +0900 Subject: [PATCH 39/87] =?UTF-8?q?feat:=20onEdit=20=ED=95=B8=EB=93=A4?= =?UTF-8?q?=EB=9F=AC=EC=9D=98=20Debounce=20=EC=9E=91=EC=97=85=EC=9D=84=20?= =?UTF-8?q?=EA=B0=81=20input=20=EC=BB=B4=ED=8F=AC=EB=84=8C=ED=8A=B8?= =?UTF-8?q?=EC=97=90=EC=84=9C=20=EC=88=98=ED=96=89=ED=95=98=EB=8F=84?= =?UTF-8?q?=EB=A1=9D=20=EB=B3=80=EA=B2=BD?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../Editor/Content/NotionEditorContent.js | 29 +++++++++++++++---- src/components/Editor/NotionEditor.js | 22 +++++++------- .../Editor/Title/NotionEditorTitle.js | 29 ++++++++++++++----- src/core/Component.js | 4 +-- src/pages/NotionPage.js | 29 +++++++------------ 5 files changed, 70 insertions(+), 43 deletions(-) diff --git a/src/components/Editor/Content/NotionEditorContent.js b/src/components/Editor/Content/NotionEditorContent.js index 8ff26e66..51b6a94d 100644 --- a/src/components/Editor/Content/NotionEditorContent.js +++ b/src/components/Editor/Content/NotionEditorContent.js @@ -1,8 +1,12 @@ +import { Document } from '@utils/editor'; + import Component from '@core/Component'; import './NotionEditorContent.css'; export default class NotionEditorContent extends Component { + timer = null; + setup() { this.state = { content: '', @@ -10,17 +14,32 @@ export default class NotionEditorContent extends Component { } initComponent() { - this.$content = document.createElement('textarea'); + this.$contentEditor = document.createElement('div'); + + this.$contentEditor.contentEditable = true; + this.$contentEditor.className = 'notion-editor-content'; + this.$contentEditor.dataset.name = 'content'; + + this.$target.appendChild(this.$contentEditor); + } - this.$content.className = 'notion-editor-content'; - this.$content.name = 'content'; + setEvent() { + const { onEdit } = this.props; - this.$target.appendChild(this.$content); + this.$contentEditor.addEventListener('input', ({ target }) => { + if (this.timer !== null) { + clearTimeout(this.timer); + } + this.timer = setTimeout(async () => { + const value = Document.stringify(target.innerHTML); + onEdit('content', value); + }, 1000); + }); } render() { const { content } = this.state; - this.$content.value = content; + this.$contentEditor.innerHTML = Document.parse(content ?? ''); } } diff --git a/src/components/Editor/NotionEditor.js b/src/components/Editor/NotionEditor.js index 88f466e1..397690b0 100644 --- a/src/components/Editor/NotionEditor.js +++ b/src/components/Editor/NotionEditor.js @@ -13,20 +13,20 @@ export default class NotionEditor extends Component { } initChildComponents() { - this.$title = new NotionEditorTitle(this.$editor); - this.$content = new NotionEditorContent(this.$editor); + this.$title = new NotionEditorTitle(this.$editor, { + onEdit: this.handleEditorInputChange.bind(this), + }); + this.$content = new NotionEditorContent(this.$editor, { + onEdit: this.handleEditorInputChange.bind(this), + }); } - setEvent() { + handleEditorInputChange(name, value) { const { onEdit } = this.props; - this.$editor.addEventListener('input', ({ target }) => { - const name = target.getAttribute('name'); - const { value } = target; - - onEdit(name, { - ...this.state, - [name]: value, - }); + + onEdit(name, { + ...this.state, + [name]: value, }); } diff --git a/src/components/Editor/Title/NotionEditorTitle.js b/src/components/Editor/Title/NotionEditorTitle.js index 0613facc..dac74340 100644 --- a/src/components/Editor/Title/NotionEditorTitle.js +++ b/src/components/Editor/Title/NotionEditorTitle.js @@ -3,6 +3,8 @@ import Component from '@core/Component'; import './NotionEditorTitle.css'; export default class NotionEditorTitle extends Component { + timer = null; + setup() { this.state = { title: '', @@ -10,19 +12,32 @@ export default class NotionEditorTitle extends Component { } initComponent() { - this.$title = document.createElement('input'); + this.$titleEditor = document.createElement('input'); + + this.$titleEditor.className = 'notion-editor-title'; + this.$titleEditor.type = 'text'; + this.$titleEditor.dataset.name = 'title'; + this.$titleEditor.placeholder = 'undefined'; + + this.$target.appendChild(this.$titleEditor); + } - this.$title.className = 'notion-editor-title'; - this.$title.type = 'text'; - this.$title.name = 'title'; - this.$title.placeholder = 'undefined'; + setEvent() { + const { onEdit } = this.props; - this.$target.appendChild(this.$title); + this.$titleEditor.addEventListener('input', ({ target }) => { + if (this.timer !== null) { + clearTimeout(this.timer); + } + this.timer = setTimeout(async () => { + onEdit('title', target.value); + }, 1000); + }); } render() { const { title } = this.state; - this.$title.value = title ?? ''; + this.$titleEditor.value = title ?? ''; } } diff --git a/src/core/Component.js b/src/core/Component.js index bb2308b8..f40de1b6 100644 --- a/src/core/Component.js +++ b/src/core/Component.js @@ -5,10 +5,10 @@ export default class Component { this.setup(); this.initComponent(); + this.initChildComponents(); + this.setEvent(); this.render(); - - this.initChildComponents(); } setup() {} diff --git a/src/pages/NotionPage.js b/src/pages/NotionPage.js index 8241d531..bd737a9a 100644 --- a/src/pages/NotionPage.js +++ b/src/pages/NotionPage.js @@ -10,8 +10,6 @@ import NotionSidebar from '@components/NotionSidebar/NotionSidebar'; import './NotionPage.css'; export default class NotionPage extends Component { - timer = null; - initComponent() { this.$page = document.createElement('div'); this.$page.className = 'notion-page'; @@ -51,23 +49,18 @@ export default class NotionPage extends Component { this.$document.setState({ documentData }); } - onEdit(name, document) { - if (this.timer !== null) { - clearTimeout(this.timer); + async onEdit(name, document) { + setItem(`temp-post-${document.id}`, { + ...document, + tempSaveDate: new Date(), + }); + + const updatedDocument = await updateDocument(document.id, document); + if (name === 'title') { + this.fetchDocumentList(); } - this.timer = setTimeout(async () => { - setItem(`temp-post-${document.id}`, { - ...document, - tempSaveDate: new Date(), - }); - - const updatedDocument = await updateDocument(document.id, document); - if (name === 'title') { - this.fetchDocumentList(); - } - - this.$document.setState({ documentData: updatedDocument }); - }, 1000); + + this.$document.setState({ documentData: updatedDocument }); } setState(newState) { From 644325d0be4c07d94f947f3997964d987ac4d040 Mon Sep 17 00:00:00 2001 From: kutta97 Date: Wed, 5 Jul 2023 00:11:03 +0900 Subject: [PATCH 40/87] =?UTF-8?q?feat:=20HTMLString=20class=20=EC=B6=94?= =?UTF-8?q?=EA=B0=80=20=EB=B0=8F=20Editor=20=EC=9D=98=20innerHTML=20?= =?UTF-8?q?=EC=9D=84=20string=20=EC=9C=BC=EB=A1=9C=20=ED=8C=8C=EC=8B=B1=20?= =?UTF-8?q?=EA=B8=B0=EB=8A=A5=20=EC=B6=94=EA=B0=80?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../Editor/Content/NotionEditorContent.js | 7 +-- src/consts/html.js | 10 ++++ src/utils/editor.js | 52 +++++++++++++++++++ src/utils/html.js | 19 +++++++ src/utils/regex.js | 7 +++ 5 files changed, 92 insertions(+), 3 deletions(-) create mode 100644 src/consts/html.js create mode 100644 src/utils/editor.js create mode 100644 src/utils/html.js create mode 100644 src/utils/regex.js diff --git a/src/components/Editor/Content/NotionEditorContent.js b/src/components/Editor/Content/NotionEditorContent.js index 51b6a94d..4fe57fac 100644 --- a/src/components/Editor/Content/NotionEditorContent.js +++ b/src/components/Editor/Content/NotionEditorContent.js @@ -1,4 +1,4 @@ -import { Document } from '@utils/editor'; +import { Editor } from '@utils/editor'; import Component from '@core/Component'; @@ -31,7 +31,8 @@ export default class NotionEditorContent extends Component { clearTimeout(this.timer); } this.timer = setTimeout(async () => { - const value = Document.stringify(target.innerHTML); + const html = target.innerHTML; + const value = Editor.stringify(html); onEdit('content', value); }, 1000); }); @@ -40,6 +41,6 @@ export default class NotionEditorContent extends Component { render() { const { content } = this.state; - this.$contentEditor.innerHTML = Document.parse(content ?? ''); + this.$contentEditor.innerHTML = Editor.parse(content ?? ''); } } diff --git a/src/consts/html.js b/src/consts/html.js new file mode 100644 index 00000000..32f65196 --- /dev/null +++ b/src/consts/html.js @@ -0,0 +1,10 @@ +const reservedCharacters = { + '&': '&', + '<': '<', + '>': '>', + '"': '"', + ' ': ' ', + '–': '-', +}; + +export default reservedCharacters; diff --git a/src/utils/editor.js b/src/utils/editor.js new file mode 100644 index 00000000..99b46893 --- /dev/null +++ b/src/utils/editor.js @@ -0,0 +1,52 @@ +import HTMLString from './html'; + +const parse = (string) => { + if (!string) return ''; + + const html = string + .split('\n') + .map((line) => { + if (line.indexOf('# ') === 0) { + return `

    ${line.substring(2)}

    `; + } + if (line.indexOf('## ') === 0) { + return `

    ${line.substring(3)}

    `; + } + if (line.indexOf('### ') === 0) { + return `

    ${line.substring(4)}

    `; + } + if (line === '') { + return '
    '; + } + return `${line}`; + }) + .map((line) => `
    ${line}
    `) + .join(''); + + return html; +}; + +const stringify = (html) => { + const htmlString = new HTMLString(html); + if (!html) return ''; + + const string = htmlString + .replaceHTMLEntities() + .splitWithTag('div') + .map((line) => { + // div 내부 글자 parsing + if (line === '
    ') return ''; + // TODO: h1, h2, h3 태그인지 확인 + // TODO: list 태그인지 확인 + return line.replaceAll('
    ', '\n'); + }) + .join('\n'); + + return string; +}; + +export const Editor = { parse, stringify }; + +export const setCaret = () => {}; + +export const getCaret = () => {}; diff --git a/src/utils/html.js b/src/utils/html.js new file mode 100644 index 00000000..8af2df14 --- /dev/null +++ b/src/utils/html.js @@ -0,0 +1,19 @@ +import reservedCharacters from '@consts/html'; + +import { getHTMLEntityRegex, getHTMLTagRegex } from './regex'; + +const htmlEntityRegex = getHTMLEntityRegex(); + +export default class HTMLString extends String { + replaceHTMLEntities() { + return new HTMLString( + this.replaceAll(htmlEntityRegex, (match) => reservedCharacters[match]) + ); + } + + splitWithTag(tagName) { + const tagRegex = getHTMLTagRegex(tagName); + + return [...this.matchAll(tagRegex)].map(([, line]) => line); + } +} diff --git a/src/utils/regex.js b/src/utils/regex.js new file mode 100644 index 00000000..6700727c --- /dev/null +++ b/src/utils/regex.js @@ -0,0 +1,7 @@ +import reservedCharacters from '@consts/html'; + +export const getHTMLTagRegex = (tagName) => + new RegExp(`<${tagName}>(.*?)`, 'g'); + +export const getHTMLEntityRegex = () => + new RegExp(Object.keys(reservedCharacters).join('|'), 'g'); From 3fd92308f42baa38b62215ec14f138345a1ee617 Mon Sep 17 00:00:00 2001 From: kutta97 Date: Wed, 5 Jul 2023 01:19:16 +0900 Subject: [PATCH 41/87] =?UTF-8?q?feat:=20HTMLString.splitWithTag()=20?= =?UTF-8?q?=ED=95=A8=EC=88=98=EC=97=90=EC=84=9C=20tag=20=EB=A1=9C=20?= =?UTF-8?q?=EA=B0=90=EC=8B=B8=EC=A7=80=EC=A7=80=20=EC=95=8A=EC=9D=80=20str?= =?UTF-8?q?ing=20=EB=8F=84=20=ED=8F=AC=ED=95=A8=ED=95=98=EB=8F=84=EB=A1=9D?= =?UTF-8?q?=20=EC=88=98=EC=A0=95?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- src/utils/html.js | 24 +++++++++++++++++++++++- 1 file changed, 23 insertions(+), 1 deletion(-) diff --git a/src/utils/html.js b/src/utils/html.js index 8af2df14..74b05eaf 100644 --- a/src/utils/html.js +++ b/src/utils/html.js @@ -12,8 +12,30 @@ export default class HTMLString extends String { } splitWithTag(tagName) { + const tags = []; const tagRegex = getHTMLTagRegex(tagName); + const htmlString = this.valueOf(); - return [...this.matchAll(tagRegex)].map(([, line]) => line); + let { lastIndex } = tagRegex; + + let match = tagRegex.exec(htmlString); + if (match === null) { + tags.push(htmlString); + return tags; + } + + while (match) { + if (lastIndex < match.index) { + tags.push(htmlString.substring(lastIndex, match.index)); + } + + const [, matchText] = match; + tags.push(matchText); + + lastIndex = tagRegex.lastIndex; + match = tagRegex.exec(htmlString); + } + + return tags; } } From 30d7b6c6c3f015d0446ac0e3373065960794455d Mon Sep 17 00:00:00 2001 From: kutta97 Date: Wed, 5 Jul 2023 01:50:00 +0900 Subject: [PATCH 42/87] =?UTF-8?q?feat:=20plain=20string=20=EC=9D=84=20HTML?= =?UTF-8?q?=20=EB=A1=9C=20=EB=B3=80=ED=99=98=ED=95=98=EA=B8=B0=20=EC=9C=84?= =?UTF-8?q?=ED=95=9C=20MarkDownString=20class=20=EC=B6=94=EA=B0=80?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- src/consts/html.js | 11 +++++++++-- src/utils/markdown.js | 17 +++++++++++++++++ src/utils/regex.js | 13 ++++++++----- 3 files changed, 34 insertions(+), 7 deletions(-) create mode 100644 src/utils/markdown.js diff --git a/src/consts/html.js b/src/consts/html.js index 32f65196..01366d9b 100644 --- a/src/consts/html.js +++ b/src/consts/html.js @@ -1,4 +1,4 @@ -const reservedCharacters = { +export const reservedCharacters = { '&': '&', '<': '<', '>': '>', @@ -7,4 +7,11 @@ const reservedCharacters = { '–': '-', }; -export default reservedCharacters; +export const htmlEntities = { + '&': '&', + '<': '<', + '>': '>', + '"': '"', + ' ': ' ', + '-': '–', +}; diff --git a/src/utils/markdown.js b/src/utils/markdown.js new file mode 100644 index 00000000..71f0a433 --- /dev/null +++ b/src/utils/markdown.js @@ -0,0 +1,17 @@ +import { htmlEntities } from '@consts/html'; + +import { getReservedCharacterRegex } from './regex'; + +const reservedCharacterRegex = getReservedCharacterRegex(); + +export default class MarkDownString extends String { + replaceReservedCharacters() { + return new MarkDownString( + this.replaceAll(reservedCharacterRegex, (match) => htmlEntities[match]) + ); + } + + splitWithNewLine() { + return this.split('\n').map((line) => new MarkDownString(line)); + } +} diff --git a/src/utils/regex.js b/src/utils/regex.js index 6700727c..3b6b0293 100644 --- a/src/utils/regex.js +++ b/src/utils/regex.js @@ -1,7 +1,10 @@ -import reservedCharacters from '@consts/html'; +import { htmlEntities, reservedCharacters } from '@consts/html'; -export const getHTMLTagRegex = (tagName) => - new RegExp(`<${tagName}>(.*?)`, 'g'); +export const getHTMLTagRegex = (tagName, options = 'g') => + new RegExp(`<${tagName}>(.*?)`, options); -export const getHTMLEntityRegex = () => - new RegExp(Object.keys(reservedCharacters).join('|'), 'g'); +export const getHTMLEntityRegex = (options = 'g') => + new RegExp(Object.keys(reservedCharacters).join('|'), options); + +export const getReservedCharacterRegex = (options = 'g') => + new RegExp(Object.keys(htmlEntities).join('|'), options); From 0650a107300985edf9fae5a94f32d5b2280df063 Mon Sep 17 00:00:00 2001 From: kutta97 Date: Wed, 5 Jul 2023 01:50:50 +0900 Subject: [PATCH 43/87] =?UTF-8?q?feat:=20MarkDownString=20=EC=9D=84=20?= =?UTF-8?q?=EC=82=AC=EC=9A=A9=ED=95=98=EC=97=AC=20reservedCharacter=20?= =?UTF-8?q?=EB=93=A4=EC=9D=84=20HTML=20entity=20=EB=A1=9C=20=EB=B3=80?= =?UTF-8?q?=EA=B2=BD=ED=95=98=EB=8A=94=20=EA=B8=B0=EB=8A=A5=20=EC=B6=94?= =?UTF-8?q?=EA=B0=80?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- src/utils/editor.js | 19 +++++++++++-------- 1 file changed, 11 insertions(+), 8 deletions(-) diff --git a/src/utils/editor.js b/src/utils/editor.js index 99b46893..518632a9 100644 --- a/src/utils/editor.js +++ b/src/utils/editor.js @@ -1,10 +1,14 @@ import HTMLString from './html'; +import MarkDownString from './markdown'; const parse = (string) => { if (!string) return ''; - const html = string - .split('\n') + const markDownString = new MarkDownString(string); + + const html = markDownString + .replaceReservedCharacters() + .splitWithNewLine() .map((line) => { if (line.indexOf('# ') === 0) { return `

    ${line.substring(2)}

    `; @@ -15,7 +19,7 @@ const parse = (string) => { if (line.indexOf('### ') === 0) { return `

    ${line.substring(4)}

    `; } - if (line === '') { + if (line.valueOf() === '') { return '
    '; } return `${line}`; @@ -27,21 +31,20 @@ const parse = (string) => { }; const stringify = (html) => { - const htmlString = new HTMLString(html); if (!html) return ''; + const htmlString = new HTMLString(html); + const string = htmlString - .replaceHTMLEntities() .splitWithTag('div') .map((line) => { // div 내부 글자 parsing - if (line === '
    ') return ''; + if (line.valueOf() === '
    ') return ''; // TODO: h1, h2, h3 태그인지 확인 // TODO: list 태그인지 확인 - return line.replaceAll('
    ', '\n'); + return line.replaceHTMLEntities().replaceAll('
    ', '\n'); }) .join('\n'); - return string; }; From f1986c6e7a287b0d980f5c57f6c6e946162738e2 Mon Sep 17 00:00:00 2001 From: kutta97 Date: Wed, 5 Jul 2023 01:51:50 +0900 Subject: [PATCH 44/87] =?UTF-8?q?feat:=20HTMLString.splitWithTag=20?= =?UTF-8?q?=EA=B0=80=20=EB=B0=98=ED=99=98=EA=B0=92=EC=9C=BC=EB=A1=9C=20Str?= =?UTF-8?q?ing=20=EB=B0=B0=EC=97=B4=20=EB=8C=80=EC=8B=A0=20HTMLString=20?= =?UTF-8?q?=EB=B0=B0=EC=97=B4=20=EB=B0=98=ED=99=98=ED=95=98=EB=8F=84?= =?UTF-8?q?=EB=A1=9D=20=EC=88=98=EC=A0=95?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- src/utils/html.js | 14 +++++++------- 1 file changed, 7 insertions(+), 7 deletions(-) diff --git a/src/utils/html.js b/src/utils/html.js index 74b05eaf..e125be2d 100644 --- a/src/utils/html.js +++ b/src/utils/html.js @@ -1,4 +1,4 @@ -import reservedCharacters from '@consts/html'; +import { reservedCharacters } from '@consts/html'; import { getHTMLEntityRegex, getHTMLTagRegex } from './regex'; @@ -14,26 +14,26 @@ export default class HTMLString extends String { splitWithTag(tagName) { const tags = []; const tagRegex = getHTMLTagRegex(tagName); - const htmlString = this.valueOf(); let { lastIndex } = tagRegex; - let match = tagRegex.exec(htmlString); + let match = tagRegex.exec(this); if (match === null) { - tags.push(htmlString); + tags.push(this); return tags; } while (match) { if (lastIndex < match.index) { - tags.push(htmlString.substring(lastIndex, match.index)); + const text = this.substring(lastIndex, match.index); + tags.push(new HTMLString(text)); } const [, matchText] = match; - tags.push(matchText); + tags.push(new HTMLString(matchText)); lastIndex = tagRegex.lastIndex; - match = tagRegex.exec(htmlString); + match = tagRegex.exec(this); } return tags; From 8204bc79733353d7abd273ea82e162683be7773d Mon Sep 17 00:00:00 2001 From: kutta97 Date: Wed, 5 Jul 2023 05:30:55 +0900 Subject: [PATCH 45/87] =?UTF-8?q?feat:=20contentEditable=20innerHTML=20?= =?UTF-8?q?=EC=97=85=EB=8D=B0=EC=9D=B4=ED=8A=B8=20=ED=9B=84=EC=97=90?= =?UTF-8?q?=EB=8F=84=20=EC=BB=A4=EC=84=9C(caret)=20=EC=9C=84=EC=B9=98=20?= =?UTF-8?q?=EC=9C=A0=EC=A7=80=20=EA=B8=B0=EB=8A=A5=20=EC=B6=94=EA=B0=80?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../Editor/Content/NotionEditorContent.js | 13 ++++-- src/consts/caret.js | 6 +++ src/utils/caret.js | 31 +++++++++++++ src/utils/editor.js | 45 ++++++++++++++++--- src/utils/regex.js | 3 ++ 5 files changed, 88 insertions(+), 10 deletions(-) create mode 100644 src/consts/caret.js create mode 100644 src/utils/caret.js diff --git a/src/components/Editor/Content/NotionEditorContent.js b/src/components/Editor/Content/NotionEditorContent.js index 4fe57fac..79c399cb 100644 --- a/src/components/Editor/Content/NotionEditorContent.js +++ b/src/components/Editor/Content/NotionEditorContent.js @@ -1,4 +1,4 @@ -import { Editor } from '@utils/editor'; +import Editor from '@utils/editor'; import Component from '@core/Component'; @@ -31,9 +31,12 @@ export default class NotionEditorContent extends Component { clearTimeout(this.timer); } this.timer = setTimeout(async () => { + Editor.saveCaretPosition(); + const html = target.innerHTML; - const value = Editor.stringify(html); - onEdit('content', value); + const string = Editor.stringify(html); + + onEdit('content', string); }, 1000); }); } @@ -41,6 +44,8 @@ export default class NotionEditorContent extends Component { render() { const { content } = this.state; - this.$contentEditor.innerHTML = Editor.parse(content ?? ''); + this.$contentEditor.innerHTML = Editor.parse(content, this.caret); + + Editor.restoreCaretPosition(); } } diff --git a/src/consts/caret.js b/src/consts/caret.js new file mode 100644 index 00000000..9aae2851 --- /dev/null +++ b/src/consts/caret.js @@ -0,0 +1,6 @@ +const CARET = { + ID: 'caret', + SPAN: (id) => ``, +}; + +export default CARET; diff --git a/src/utils/caret.js b/src/utils/caret.js new file mode 100644 index 00000000..3e90c39d --- /dev/null +++ b/src/utils/caret.js @@ -0,0 +1,31 @@ +import CARET from '@consts/caret'; + +const savePosition = () => { + const $caret = document.createElement('span'); + $caret.id = CARET.ID; + + const selection = window.getSelection(); + const range = selection.getRangeAt(0); + range.insertNode($caret); +}; + +const setPosition = () => { + const $caret = document.getElementById(CARET.ID); + + if (!$caret) return; + + const range = document.createRange(); + range.selectNode($caret); + + const selection = window.getSelection(); + selection.removeAllRanges(); + selection.addRange(range); + range.deleteContents(); +}; + +const Caret = { + savePosition, + setPosition, +}; + +export default Caret; diff --git a/src/utils/editor.js b/src/utils/editor.js index 518632a9..8969ba1c 100644 --- a/src/utils/editor.js +++ b/src/utils/editor.js @@ -1,13 +1,36 @@ +import CARET from '@consts/caret'; + +import Caret from './caret'; import HTMLString from './html'; import MarkDownString from './markdown'; +import { getCaretSpanTagRegex } from './regex'; + +let caretPosition = 0; + +const caretSpanTagRegex = getCaretSpanTagRegex(); + +const getCaretPositionInString = (string) => { + const pos = caretSpanTagRegex.exec(string); + return pos === null ? 0 : pos.index; +}; + +const removeCaretFromString = (string) => string.replace(caretSpanTagRegex, ''); const parse = (string) => { if (!string) return ''; - const markDownString = new MarkDownString(string); + const headString = new MarkDownString(string.substring(0, caretPosition)) + .replaceReservedCharacters() + .valueOf(); - const html = markDownString + const tailString = new MarkDownString(string.substring(caretPosition)) .replaceReservedCharacters() + .valueOf(); + + const stringWithCaret = `${headString}${CARET.SPAN(CARET.ID)}${tailString}`; + const markDownString = new MarkDownString(stringWithCaret); + + const html = markDownString .splitWithNewLine() .map((line) => { if (line.indexOf('# ') === 0) { @@ -45,11 +68,21 @@ const stringify = (html) => { return line.replaceHTMLEntities().replaceAll('
    ', '\n'); }) .join('\n'); - return string; + + caretPosition = getCaretPositionInString(string); + const stringWithoutCaret = removeCaretFromString(string); + + return stringWithoutCaret; +}; + +const saveCaretPosition = () => { + Caret.savePosition(); }; -export const Editor = { parse, stringify }; +const restoreCaretPosition = () => { + Caret.setPosition(); +}; -export const setCaret = () => {}; +const Editor = { parse, stringify, saveCaretPosition, restoreCaretPosition }; -export const getCaret = () => {}; +export default Editor; diff --git a/src/utils/regex.js b/src/utils/regex.js index 3b6b0293..777f84dd 100644 --- a/src/utils/regex.js +++ b/src/utils/regex.js @@ -1,3 +1,4 @@ +import CARET from '@consts/caret'; import { htmlEntities, reservedCharacters } from '@consts/html'; export const getHTMLTagRegex = (tagName, options = 'g') => @@ -8,3 +9,5 @@ export const getHTMLEntityRegex = (options = 'g') => export const getReservedCharacterRegex = (options = 'g') => new RegExp(Object.keys(htmlEntities).join('|'), options); + +export const getCaretSpanTagRegex = () => new RegExp(CARET.SPAN(CARET.ID)); From 54db35a2be53be80e0f780a052cf83c38e7094fd Mon Sep 17 00:00:00 2001 From: kutta97 Date: Wed, 5 Jul 2023 05:46:56 +0900 Subject: [PATCH 46/87] =?UTF-8?q?fix:=20=EB=B9=88=20=EB=AC=B8=EC=9E=A5?= =?UTF-8?q?=EC=97=90=20Caret=20tag=20=EA=B0=80=20=EC=9E=88=EB=8A=94=20?= =?UTF-8?q?=EA=B2=BD=EC=9A=B0,=20new=20line=20=EC=B2=98=EB=A6=AC=EA=B0=80?= =?UTF-8?q?=20=EC=A0=9C=EB=8C=80=EB=A1=9C=20=EB=90=98=EC=A7=80=20=EC=95=8A?= =?UTF-8?q?=EB=8A=94=20=EB=B2=84=EA=B7=B8=20=EC=88=98=EC=A0=95?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- src/utils/editor.js | 5 +++++ 1 file changed, 5 insertions(+) diff --git a/src/utils/editor.js b/src/utils/editor.js index 8969ba1c..51a74e77 100644 --- a/src/utils/editor.js +++ b/src/utils/editor.js @@ -42,6 +42,9 @@ const parse = (string) => { if (line.indexOf('### ') === 0) { return `

    ${line.substring(4)}

    `; } + if (line.valueOf() === CARET.SPAN(CARET.ID)) { + return `${CARET.SPAN(CARET.ID)}
    `; + } if (line.valueOf() === '') { return '
    '; } @@ -63,6 +66,8 @@ const stringify = (html) => { .map((line) => { // div 내부 글자 parsing if (line.valueOf() === '
    ') return ''; + if (line.valueOf() === `${CARET.SPAN(CARET.ID)}
    `) + return CARET.SPAN(CARET.ID); // TODO: h1, h2, h3 태그인지 확인 // TODO: list 태그인지 확인 return line.replaceHTMLEntities().replaceAll('
    ', '\n'); From 23a02dbe3dcd4d88b4a645c3eb1817912e2d152f Mon Sep 17 00:00:00 2001 From: kutta97 Date: Wed, 5 Jul 2023 09:34:36 +0900 Subject: [PATCH 47/87] =?UTF-8?q?feat:=20=ED=95=9C=EA=B5=AD=EC=96=B4=20?= =?UTF-8?q?=EC=A4=91=EB=B3=B5=EC=9E=85=EB=A0=A5=EC=9D=84=20=EB=A7=89?= =?UTF-8?q?=EB=8A=94=20=EA=B8=B0=EB=8A=A5=20=EC=B6=94=EA=B0=80=20-=20?= =?UTF-8?q?=EC=97=90=EB=94=94=ED=84=B0=EC=97=90=20blur()=20=ED=96=88?= =?UTF-8?q?=EB=8B=A4=EA=B0=80=20=EB=8B=A4=EC=8B=9C=20focus()=20=ED=95=B4?= =?UTF-8?q?=EB=B3=B4=EA=B8=B0=20-=20compositionend,=20compositionupdate=20?= =?UTF-8?q?=EC=97=90=EB=8F=84=20=EC=9D=B4=EB=B2=A4=ED=8A=B8=20=EB=A6=AC?= =?UTF-8?q?=EC=8A=A4=EB=84=88=20=EB=8B=AC=EC=95=84=EB=B3=B4=EA=B8=B0?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../Editor/Content/NotionEditorContent.js | 69 ++++++++++++++++--- src/utils/caret.js | 5 ++ src/utils/regex.js | 2 + 3 files changed, 66 insertions(+), 10 deletions(-) diff --git a/src/components/Editor/Content/NotionEditorContent.js b/src/components/Editor/Content/NotionEditorContent.js index 79c399cb..2f7f8fbb 100644 --- a/src/components/Editor/Content/NotionEditorContent.js +++ b/src/components/Editor/Content/NotionEditorContent.js @@ -1,4 +1,5 @@ import Editor from '@utils/editor'; +import { getKoreanRegex } from '@utils/regex'; import Component from '@core/Component'; @@ -23,22 +24,70 @@ export default class NotionEditorContent extends Component { this.$target.appendChild(this.$contentEditor); } + handleEdit({ target, callback }) { + Editor.saveCaretPosition(); + + const html = target.innerHTML; + const string = Editor.stringify(html); + + callback('content', string); + } + setEvent() { const { onEdit } = this.props; - this.$contentEditor.addEventListener('input', ({ target }) => { - if (this.timer !== null) { - clearTimeout(this.timer); + this.$contentEditor.addEventListener( + 'compositionupdate', + ({ target, isTrusted }) => { + if (isTrusted === false) return; + + if (this.timer !== null) clearTimeout(this.timer); + + this.timer = setTimeout(async () => { + this.handleEdit({ target, callback: onEdit }); + }, 1000); + } + ); + + this.$contentEditor.addEventListener( + 'compositionend', + ({ target, isTrusted }) => { + if (isTrusted === false) return; + + if (this.timer !== null) clearTimeout(this.timer); + + this.timer = setTimeout(async () => { + this.handleEdit({ target, callback: onEdit }); + }, 1000); } - this.timer = setTimeout(async () => { - Editor.saveCaretPosition(); + ); + + this.$contentEditor.addEventListener( + 'input', + ({ target, isTrusted, isComposing, data }) => { + if (isTrusted === false) return; + + if (isComposing) return; - const html = target.innerHTML; - const string = Editor.stringify(html); + const koreanRegex = getKoreanRegex(); + const isKorean = koreanRegex.exec(data) !== null; - onEdit('content', string); - }, 1000); - }); + if (isKorean) { + this.$contentEditor.blur(); + const { content } = this.state; + this.setState({ content }); + this.$contentEditor.focus(); + + return; + } + + if (this.timer !== null) clearTimeout(this.timer); + + this.timer = setTimeout(async () => { + this.handleEdit({ target, callback: onEdit }); + }, 1000); + } + ); } render() { diff --git a/src/utils/caret.js b/src/utils/caret.js index 3e90c39d..f82dc71b 100644 --- a/src/utils/caret.js +++ b/src/utils/caret.js @@ -7,6 +7,9 @@ const savePosition = () => { const selection = window.getSelection(); const range = selection.getRangeAt(0); range.insertNode($caret); + + const { focusNode } = selection; + focusNode.blur(); }; const setPosition = () => { @@ -14,6 +17,8 @@ const setPosition = () => { if (!$caret) return; + $caret.focus(); + const range = document.createRange(); range.selectNode($caret); diff --git a/src/utils/regex.js b/src/utils/regex.js index 777f84dd..d97f53ea 100644 --- a/src/utils/regex.js +++ b/src/utils/regex.js @@ -11,3 +11,5 @@ export const getReservedCharacterRegex = (options = 'g') => new RegExp(Object.keys(htmlEntities).join('|'), options); export const getCaretSpanTagRegex = () => new RegExp(CARET.SPAN(CARET.ID)); + +export const getKoreanRegex = () => /[\u3131-\uD79D]/; From 88fff62fd713e8f611f31625caa3c466aed7522c Mon Sep 17 00:00:00 2001 From: kutta97 Date: Wed, 5 Jul 2023 09:47:29 +0900 Subject: [PATCH 48/87] =?UTF-8?q?fix:=20=ED=95=9C=EA=B8=80=EC=9E=85?= =?UTF-8?q?=EB=A0=A5=20=EC=94=B9=ED=9E=98=20=EC=98=A4=EB=A5=98=EB=A5=BC=20?= =?UTF-8?q?=EB=B0=A9=EC=A7=80=20=EC=BD=94=EB=93=9C=20=EC=B6=94=EA=B0=80=20?= =?UTF-8?q?-=20compositionupdate=20=EB=B0=8F=20compositionend=20=EB=A6=AC?= =?UTF-8?q?=EC=8A=A4=EB=84=88=EC=97=90=EC=84=9C=20onEdit=20=EC=8B=A4?= =?UTF-8?q?=ED=96=89=EC=A0=84=EC=97=90=20focus=20=ED=92=80=EC=97=88?= =?UTF-8?q?=EB=8B=A4=EA=B0=80=20=EB=8B=A4=EC=8B=9C=20=EC=A3=BC=EB=8F=84?= =?UTF-8?q?=EB=A1=9D=20=EC=88=98=EC=A0=95?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- src/components/Editor/Content/NotionEditorContent.js | 4 ++++ src/utils/caret.js | 5 ----- 2 files changed, 4 insertions(+), 5 deletions(-) diff --git a/src/components/Editor/Content/NotionEditorContent.js b/src/components/Editor/Content/NotionEditorContent.js index 2f7f8fbb..562a3ff1 100644 --- a/src/components/Editor/Content/NotionEditorContent.js +++ b/src/components/Editor/Content/NotionEditorContent.js @@ -44,6 +44,8 @@ export default class NotionEditorContent extends Component { if (this.timer !== null) clearTimeout(this.timer); this.timer = setTimeout(async () => { + this.$contentEditor.blur(); + this.$contentEditor.focus(); this.handleEdit({ target, callback: onEdit }); }, 1000); } @@ -57,6 +59,8 @@ export default class NotionEditorContent extends Component { if (this.timer !== null) clearTimeout(this.timer); this.timer = setTimeout(async () => { + this.$contentEditor.blur(); + this.$contentEditor.focus(); this.handleEdit({ target, callback: onEdit }); }, 1000); } diff --git a/src/utils/caret.js b/src/utils/caret.js index f82dc71b..3e90c39d 100644 --- a/src/utils/caret.js +++ b/src/utils/caret.js @@ -7,9 +7,6 @@ const savePosition = () => { const selection = window.getSelection(); const range = selection.getRangeAt(0); range.insertNode($caret); - - const { focusNode } = selection; - focusNode.blur(); }; const setPosition = () => { @@ -17,8 +14,6 @@ const setPosition = () => { if (!$caret) return; - $caret.focus(); - const range = document.createRange(); range.selectNode($caret); From 5fbccd92f789a0ae363cc07c0caeec660c1da821 Mon Sep 17 00:00:00 2001 From: kutta97 Date: Wed, 5 Jul 2023 10:04:42 +0900 Subject: [PATCH 49/87] =?UTF-8?q?feat:=20=EC=97=90=EB=94=94=ED=84=B0?= =?UTF-8?q?=EC=97=90=20paste=20=ED=95=A0=20=EB=95=8C,=20html=20=ED=83=9C?= =?UTF-8?q?=EA=B7=B8=EC=9D=98=20style=20=EC=A0=9C=EA=B1=B0=ED=95=98?= =?UTF-8?q?=EA=B3=A0=20=EB=B6=99=EC=97=AC=EB=84=A3=EB=8F=84=EB=A1=9D=20?= =?UTF-8?q?=EC=88=98=EC=A0=95?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- src/components/Editor/Content/NotionEditorContent.js | 6 ++++++ 1 file changed, 6 insertions(+) diff --git a/src/components/Editor/Content/NotionEditorContent.js b/src/components/Editor/Content/NotionEditorContent.js index 562a3ff1..b754faf7 100644 --- a/src/components/Editor/Content/NotionEditorContent.js +++ b/src/components/Editor/Content/NotionEditorContent.js @@ -36,6 +36,12 @@ export default class NotionEditorContent extends Component { setEvent() { const { onEdit } = this.props; + this.$contentEditor.addEventListener('paste', (e) => { + e.preventDefault(); + const text = e.clipboardData.getData('text/plain'); + document.execCommand('insertHTML', false, text); + }); + this.$contentEditor.addEventListener( 'compositionupdate', ({ target, isTrusted }) => { From ec6440c7078acec5cedcf0cb454a22ea48dfeae0 Mon Sep 17 00:00:00 2001 From: kutta97 Date: Wed, 5 Jul 2023 14:48:50 +0900 Subject: [PATCH 50/87] =?UTF-8?q?feat:=20API=20username=20=EB=B3=80?= =?UTF-8?q?=EA=B2=BD?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- src/utils/api.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/utils/api.js b/src/utils/api.js index 4b5b56c5..e2b19044 100644 --- a/src/utils/api.js +++ b/src/utils/api.js @@ -1,7 +1,7 @@ const API_END_POINT = 'https://kdt-frontend.programmers.co.kr'; const getHeader = () => { - const username = 'roto'; + const username = 'kutta97'; return { 'Content-Type': 'application/json', 'x-username': username, From 218bc4025ebcdade1f26c154534c8427e37ce879 Mon Sep 17 00:00:00 2001 From: kutta97 Date: Wed, 5 Jul 2023 16:49:48 +0900 Subject: [PATCH 51/87] =?UTF-8?q?feat:=20=EC=97=90=EB=94=94=ED=84=B0?= =?UTF-8?q?=EC=97=90=20Markdown=20heading=20=EB=AC=B8=EB=B2=95=20=EC=B6=94?= =?UTF-8?q?=EA=B0=80?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- src/utils/caret.js | 13 ++++++++++++ src/utils/editor.js | 47 +++++++++++++++---------------------------- src/utils/html.js | 32 ++++++++++++++++++++++++++++- src/utils/markdown.js | 20 ++++++++++++++++++ src/utils/regex.js | 3 +++ 5 files changed, 83 insertions(+), 32 deletions(-) diff --git a/src/utils/caret.js b/src/utils/caret.js index 3e90c39d..4fdf48ab 100644 --- a/src/utils/caret.js +++ b/src/utils/caret.js @@ -1,5 +1,7 @@ import CARET from '@consts/caret'; +import { getCaretSpanTagRegex } from './regex'; + const savePosition = () => { const $caret = document.createElement('span'); $caret.id = CARET.ID; @@ -23,9 +25,20 @@ const setPosition = () => { range.deleteContents(); }; +const caretSpanTagRegex = getCaretSpanTagRegex(); + +const getStringWithoutCaret = (string) => string.replace(caretSpanTagRegex, ''); + +const getCaretPositionInString = (string) => { + const pos = caretSpanTagRegex.exec(string); + return pos === null ? 0 : pos.index; +}; + const Caret = { savePosition, setPosition, + getStringWithoutCaret, + getCaretPositionInString, }; export default Caret; diff --git a/src/utils/editor.js b/src/utils/editor.js index 51a74e77..db03f5d5 100644 --- a/src/utils/editor.js +++ b/src/utils/editor.js @@ -3,19 +3,9 @@ import CARET from '@consts/caret'; import Caret from './caret'; import HTMLString from './html'; import MarkDownString from './markdown'; -import { getCaretSpanTagRegex } from './regex'; let caretPosition = 0; -const caretSpanTagRegex = getCaretSpanTagRegex(); - -const getCaretPositionInString = (string) => { - const pos = caretSpanTagRegex.exec(string); - return pos === null ? 0 : pos.index; -}; - -const removeCaretFromString = (string) => string.replace(caretSpanTagRegex, ''); - const parse = (string) => { if (!string) return ''; @@ -33,21 +23,14 @@ const parse = (string) => { const html = markDownString .splitWithNewLine() .map((line) => { - if (line.indexOf('# ') === 0) { - return `

    ${line.substring(2)}

    `; - } - if (line.indexOf('## ') === 0) { - return `

    ${line.substring(3)}

    `; - } - if (line.indexOf('### ') === 0) { - return `

    ${line.substring(4)}

    `; - } - if (line.valueOf() === CARET.SPAN(CARET.ID)) { + if (line.isHeading(1)) return line.toHTMLHeading(1); + if (line.isHeading(2)) return line.toHTMLHeading(2); + if (line.isHeading(3)) return line.toHTMLHeading(3); + + if (line.valueOf() === CARET.SPAN(CARET.ID)) return `${CARET.SPAN(CARET.ID)}
    `; - } - if (line.valueOf() === '') { - return '
    '; - } + if (line.valueOf() === '') return '
    '; + return `${line}`; }) .map((line) => `
    ${line}
    `) @@ -64,18 +47,20 @@ const stringify = (html) => { const string = htmlString .splitWithTag('div') .map((line) => { - // div 내부 글자 parsing - if (line.valueOf() === '
    ') return ''; - if (line.valueOf() === `${CARET.SPAN(CARET.ID)}
    `) - return CARET.SPAN(CARET.ID); - // TODO: h1, h2, h3 태그인지 확인 + if (line.isEmptyLine()) return ''; + if (line.isEmptyLineWithCaret()) return CARET.SPAN(CARET.ID); + + if (line.isHeading(1)) return line.toMarkDownHeading(1); + if (line.isHeading(2)) return line.toMarkDownHeading(2); + if (line.isHeading(3)) return line.toMarkDownHeading(3); + // TODO: list 태그인지 확인 return line.replaceHTMLEntities().replaceAll('
    ', '\n'); }) .join('\n'); - caretPosition = getCaretPositionInString(string); - const stringWithoutCaret = removeCaretFromString(string); + caretPosition = Caret.getCaretPositionInString(string); + const stringWithoutCaret = Caret.getStringWithoutCaret(string); return stringWithoutCaret; }; diff --git a/src/utils/html.js b/src/utils/html.js index e125be2d..4d2b40ed 100644 --- a/src/utils/html.js +++ b/src/utils/html.js @@ -1,6 +1,12 @@ +import CARET from '@consts/caret'; import { reservedCharacters } from '@consts/html'; -import { getHTMLEntityRegex, getHTMLTagRegex } from './regex'; +import Caret from './caret'; +import { + getHTMLEntityRegex, + getHTMLTagFullMatchRegex, + getHTMLTagRegex, +} from './regex'; const htmlEntityRegex = getHTMLEntityRegex(); @@ -38,4 +44,28 @@ export default class HTMLString extends String { return tags; } + + isEmptyLine() { + return this.valueOf() === '
    '; + } + + isEmptyLineWithCaret() { + return this.valueOf() === `${CARET.SPAN(CARET.ID)}
    `; + } + + isHeading(level = 1) { + if (level < 1 || level > 6) return false; + const noneCaretString = new HTMLString( + Caret.getStringWithoutCaret(this.valueOf()) + ); + + const headingRegex = getHTMLTagFullMatchRegex(`h${level}`); + return headingRegex.exec(noneCaretString.replaceHTMLEntities().valueOf()); + } + + toMarkDownHeading(level = 1) { + const headingRegex = getHTMLTagFullMatchRegex(`h${level}`); + const [, heading] = headingRegex.exec(this.replaceHTMLEntities().valueOf()); + return `${'#'.repeat(level)} ${heading}`; + } } diff --git a/src/utils/markdown.js b/src/utils/markdown.js index 71f0a433..3aef7c0d 100644 --- a/src/utils/markdown.js +++ b/src/utils/markdown.js @@ -1,5 +1,6 @@ import { htmlEntities } from '@consts/html'; +import Caret from './caret'; import { getReservedCharacterRegex } from './regex'; const reservedCharacterRegex = getReservedCharacterRegex(); @@ -14,4 +15,23 @@ export default class MarkDownString extends String { splitWithNewLine() { return this.split('\n').map((line) => new MarkDownString(line)); } + + isHeading(level) { + if (level < 1 || level > 6) return false; + const nbsp = htmlEntities[' ']; + const noneCaretString = Caret.getStringWithoutCaret(this.valueOf()); + + return noneCaretString.indexOf(`${'#'.repeat(level)}${nbsp}`) === 0; + } + + removeHeadingMark(level) { + const nbsp = htmlEntities[' ']; + return this.replace(`${'#'.repeat(level)}${nbsp}`, ''); + } + + toHTMLHeading(level = 1) { + const string = this.removeHeadingMark(level); + + return `${string}`; + } } diff --git a/src/utils/regex.js b/src/utils/regex.js index d97f53ea..2560c5dc 100644 --- a/src/utils/regex.js +++ b/src/utils/regex.js @@ -4,6 +4,9 @@ import { htmlEntities, reservedCharacters } from '@consts/html'; export const getHTMLTagRegex = (tagName, options = 'g') => new RegExp(`<${tagName}>(.*?)`, options); +export const getHTMLTagFullMatchRegex = (tagName) => + new RegExp(`^<${tagName}>(.*?)$`); + export const getHTMLEntityRegex = (options = 'g') => new RegExp(Object.keys(reservedCharacters).join('|'), options); From 3ea607db21d3c6ed97fffbe62d0a4fa31928d9fe Mon Sep 17 00:00:00 2001 From: kutta97 Date: Wed, 5 Jul 2023 22:39:55 +0900 Subject: [PATCH 52/87] =?UTF-8?q?feat:=20=EC=82=AC=EC=9D=B4=EB=93=9C?= =?UTF-8?q?=EB=B0=94=20=EC=95=84=EC=9D=B4=EC=BD=98=20SVG=20=EC=B6=94?= =?UTF-8?q?=EA=B0=80=20=EB=B0=8F=20=EA=B0=84=EB=8B=A8=ED=95=9C=20Styling?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .prettierrc.js | 1 + babel.config.js | 1 + jsconfig.json | 3 +- public/add.svg | 3 ++ public/expand.svg | 3 ++ public/trash.svg | 3 ++ src/components/DocumentList/DocumentList.js | 6 ++- .../DocumentList/Item/DocumentListItem.css | 32 +++++++++++++++- .../DocumentList/Item/DocumentListItem.js | 37 ++++++++++++++----- webpack.config.js | 5 +++ 10 files changed, 81 insertions(+), 13 deletions(-) create mode 100644 public/add.svg create mode 100644 public/expand.svg create mode 100644 public/trash.svg diff --git a/.prettierrc.js b/.prettierrc.js index caae71cd..e1a5abc5 100644 --- a/.prettierrc.js +++ b/.prettierrc.js @@ -5,6 +5,7 @@ module.exports = { singleQuote: true, importOrder: [ + '^@public/(.*)$', '^@consts/(.*)$', '^@api/(.*)$', '^@utils/(.*)$', diff --git a/babel.config.js b/babel.config.js index 9757ddce..11387124 100644 --- a/babel.config.js +++ b/babel.config.js @@ -11,6 +11,7 @@ module.exports = { '@consts': './src/consts', '@core': './src/core', '@pages': './src/pages', + '@public': './public', '@utils': './src/utils', }, }, diff --git a/jsconfig.json b/jsconfig.json index ac9e5033..b386be12 100644 --- a/jsconfig.json +++ b/jsconfig.json @@ -7,7 +7,8 @@ "@utils/*": ["./src/utils/*"], "@core/*": ["./src/core/*"], "@components/*": ["./src/components/*"], - "@pages/*": ["./src/pages/*"] + "@pages/*": ["./src/pages/*"], + "@public/*": ["./public/*"] } } } diff --git a/public/add.svg b/public/add.svg new file mode 100644 index 00000000..33ffb18e --- /dev/null +++ b/public/add.svg @@ -0,0 +1,3 @@ + + + diff --git a/public/expand.svg b/public/expand.svg new file mode 100644 index 00000000..1793a17b --- /dev/null +++ b/public/expand.svg @@ -0,0 +1,3 @@ + + + diff --git a/public/trash.svg b/public/trash.svg new file mode 100644 index 00000000..277f743b --- /dev/null +++ b/public/trash.svg @@ -0,0 +1,3 @@ + + + diff --git a/src/components/DocumentList/DocumentList.js b/src/components/DocumentList/DocumentList.js index 9ea94201..a191a484 100644 --- a/src/components/DocumentList/DocumentList.js +++ b/src/components/DocumentList/DocumentList.js @@ -35,10 +35,12 @@ export default class DocumentList extends Component { setEvent() { this.$documentList.addEventListener('click', ({ target }) => { const $li = target.closest('li'); - if (!$li) return; + if (!$li) return; const { id } = $li.dataset; - const { className } = target; + + const $button = target.closest('button'); + const { className } = $button; if (className === 'document-delete-button') { this.hanldeDeleteButtonClick(id); diff --git a/src/components/DocumentList/Item/DocumentListItem.css b/src/components/DocumentList/Item/DocumentListItem.css index 37f98264..1f2db147 100644 --- a/src/components/DocumentList/Item/DocumentListItem.css +++ b/src/components/DocumentList/Item/DocumentListItem.css @@ -3,7 +3,8 @@ flex-direction: row; align-items: center; width: 100%; - padding: 2px 0; + margin: 1px 0; + padding: 2px 10px; border-radius: 3px; color: #19171199; @@ -22,3 +23,32 @@ text-overflow: ellipsis; white-space: nowrap; } + +.document-list li button { + display: flex; + width: 22px; + height: 22px; + align-items: center; + justify-content: center; + + background: none; + border: none; + border-radius: 5px; + cursor: pointer; + + &:hover { + background-color: #e0e0de; + } +} + +.document-list li:hover .document-list-button-container { + display: flex; + opacity: 1; +} + +.document-list-button-container { + display: none; + margin-left: 4px; + gap: 2px; + transition: opacity 0.5s ease; +} diff --git a/src/components/DocumentList/Item/DocumentListItem.js b/src/components/DocumentList/Item/DocumentListItem.js index 5c51b517..b9c714b4 100644 --- a/src/components/DocumentList/Item/DocumentListItem.js +++ b/src/components/DocumentList/Item/DocumentListItem.js @@ -1,23 +1,42 @@ +import AddIcon from '@public/add.svg'; +import ExpandIcon from '@public/expand.svg'; +import TrashIcon from '@public/trash.svg'; + import Component from '@core/Component'; import './DocumentListItem.css'; export default class DocumentListItem extends Component { initComponent() { - this.$documentListItem = document.createElement('li'); - - this.$target.appendChild(this.$documentListItem); - } - - render() { const { documentItem, depth } = this.props; + this.$documentListItem = document.createElement('li'); this.$documentListItem.dataset.id = documentItem.id; + this.$documentListItem.innerHTML = ` - ${documentItem.title} - - + + ${documentItem.title} +
    + + +
    `; this.$documentListItem.style.paddingLeft = `${depth * 10}px`; + + this.$target.appendChild(this.$documentListItem); + } + + render() { + const { documentItem } = this.props; + const $title = this.$documentListItem.querySelector( + '.document-list-item-title' + ); + $title.textContent = documentItem.title; } } diff --git a/webpack.config.js b/webpack.config.js index d6f95cc4..62cbaba1 100644 --- a/webpack.config.js +++ b/webpack.config.js @@ -15,6 +15,7 @@ module.exports = (_, argv) => { '@consts': path.resolve(__dirname, 'src/consts'), '@core': path.resolve(__dirname, 'src/core'), '@pages': path.resolve(__dirname, 'src/pages'), + '@public': path.resolve(__dirname, '/public'), '@utils': path.resolve(__dirname, 'src/utils'), }, }, @@ -43,6 +44,10 @@ module.exports = (_, argv) => { exclude: /node_modules/, use: 'babel-loader', }, + { + test: /\.svg$/i, + type: 'asset/resource', + }, { test: /\.css$/i, use: ['style-loader', 'css-loader'], From 8fdd62763c79ded32ee188883d1c1fe63b302cda Mon Sep 17 00:00:00 2001 From: kutta97 Date: Wed, 5 Jul 2023 23:11:23 +0900 Subject: [PATCH 53/87] =?UTF-8?q?refactor:=20dom=20elements=20id=20?= =?UTF-8?q?=EB=B0=8F=20classnames=20=EC=83=81=EC=88=98=EB=A1=9C=20?= =?UTF-8?q?=EA=B4=80=EB=A6=AC?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- src/components/DocumentList/DocumentList.js | 18 ++++++++++--- .../DocumentList/Item/DocumentListItem.js | 12 +++++---- .../NotionDocument/NotionDocument.js | 4 ++- src/components/NotionSidebar/NotionSidebar.js | 6 +++-- src/consts/target.js | 25 +++++++++++++++++++ src/main.js | 4 ++- src/pages/NotionPage.js | 4 ++- 7 files changed, 60 insertions(+), 13 deletions(-) create mode 100644 src/consts/target.js diff --git a/src/components/DocumentList/DocumentList.js b/src/components/DocumentList/DocumentList.js index a191a484..80e0281f 100644 --- a/src/components/DocumentList/DocumentList.js +++ b/src/components/DocumentList/DocumentList.js @@ -1,3 +1,5 @@ +import { SIDEBAR } from '@consts/target'; + import { createDocument, deleteDocument } from '@api/document'; import { history } from '@utils/router'; @@ -17,7 +19,7 @@ export default class DocumentList extends Component { initComponent() { this.$documentList = document.createElement('ul'); - this.$documentList.className = 'document-list'; + this.$documentList.className = SIDEBAR.DOCUMENT_LIST; this.$target.appendChild(this.$documentList); } @@ -40,17 +42,27 @@ export default class DocumentList extends Component { const { id } = $li.dataset; const $button = target.closest('button'); + if (!$button) { + history.push(`/documents/${id}`); + return; + } const { className } = $button; - if (className === 'document-delete-button') { + if ( + className === SIDEBAR.DOCUMENT_LIST_ITEM.BUTTON_CONTAINER.DELETE_BUTTON + ) { this.hanldeDeleteButtonClick(id); return; } - if (className === 'document-create-inside-button') { + if ( + className === + SIDEBAR.DOCUMENT_LIST_ITEM.BUTTON_CONTAINER.CREATE_INSIDE_BUTTON + ) { this.handleCreateIndsideButtonClick(id); return; } + history.push(`/documents/${id}`); }); } diff --git a/src/components/DocumentList/Item/DocumentListItem.js b/src/components/DocumentList/Item/DocumentListItem.js index b9c714b4..554385a8 100644 --- a/src/components/DocumentList/Item/DocumentListItem.js +++ b/src/components/DocumentList/Item/DocumentListItem.js @@ -2,6 +2,8 @@ import AddIcon from '@public/add.svg'; import ExpandIcon from '@public/expand.svg'; import TrashIcon from '@public/trash.svg'; +import { SIDEBAR } from '@consts/target'; + import Component from '@core/Component'; import './DocumentListItem.css'; @@ -14,15 +16,15 @@ export default class DocumentListItem extends Component { this.$documentListItem.dataset.id = documentItem.id; this.$documentListItem.innerHTML = ` - - ${documentItem.title} -
    - -
    diff --git a/src/components/NotionDocument/NotionDocument.js b/src/components/NotionDocument/NotionDocument.js index 5b2b9001..4bee1d72 100644 --- a/src/components/NotionDocument/NotionDocument.js +++ b/src/components/NotionDocument/NotionDocument.js @@ -1,3 +1,5 @@ +import { DOCUMENT } from '@consts/target'; + import Component from '@core/Component'; import NotionEditor from '@components/Editor/NotionEditor'; @@ -20,7 +22,7 @@ export default class NotionDocument extends Component { initComponent() { this.$document = document.createElement('div'); - this.$document.className = 'notion-document'; + this.$document.className = DOCUMENT.ROOT; this.$target.appendChild(this.$document); } diff --git a/src/components/NotionSidebar/NotionSidebar.js b/src/components/NotionSidebar/NotionSidebar.js index 1fb8b7ee..1064fe6b 100644 --- a/src/components/NotionSidebar/NotionSidebar.js +++ b/src/components/NotionSidebar/NotionSidebar.js @@ -1,3 +1,5 @@ +import { SIDEBAR } from '@consts/target'; + import { createDocument } from '@api/document'; import { history } from '@utils/router'; @@ -18,14 +20,14 @@ export default class NotionSidebar extends Component { initComponent() { this.$sidebar = document.createElement('nav'); - this.$sidebar.className = 'notion-sidebar'; + this.$sidebar.className = SIDEBAR.ROOT; this.$target.appendChild(this.$sidebar); } initChildComponents() { this.$createButton = new Button(this.$sidebar, { - className: 'document-create-button', + className: SIDEBAR.CREATE_BUTTON, textContent: 'add a document', onClick: () => { this.handleCreateButtonClick(); diff --git a/src/consts/target.js b/src/consts/target.js new file mode 100644 index 00000000..4754f6e9 --- /dev/null +++ b/src/consts/target.js @@ -0,0 +1,25 @@ +export const TARGET = { + ENTRY: 'app', + PAGE: { + NOTION: 'notion-page', + }, +}; + +export const SIDEBAR = { + ROOT: 'notion-sidebar', + CREATE_BUTTON: 'document-create-button', + DOCUMENT_LIST: 'document-list', + DOCUMENT_LIST_ITEM: { + TITLE: 'document-list-item-title', + EXPAND_BUTTON: 'document-list-item-expand-button', + BUTTON_CONTAINER: { + ROOT: 'document-list-button-container', + DELETE_BUTTON: 'document-delete-button', + CREATE_INSIDE_BUTTON: 'document-create-inside-button', + }, + }, +}; + +export const DOCUMENT = { + ROOT: 'notion-document', +}; diff --git a/src/main.js b/src/main.js index f60ab540..9b69501c 100644 --- a/src/main.js +++ b/src/main.js @@ -1,8 +1,10 @@ +import { TARGET } from '@consts/target'; + import App from './App'; import './styleReset.css'; const initApp = ($target) => new App($target); -const $app = document.querySelector(`#app`); +const $app = document.querySelector(`#${TARGET.ENTRY}`); initApp($app); diff --git a/src/pages/NotionPage.js b/src/pages/NotionPage.js index bd737a9a..0edcf118 100644 --- a/src/pages/NotionPage.js +++ b/src/pages/NotionPage.js @@ -1,3 +1,5 @@ +import { TARGET } from '@consts/target'; + import { getDocument, getDocumentList, updateDocument } from '@api/document'; import { setItem } from '@utils/localStorage'; @@ -12,7 +14,7 @@ import './NotionPage.css'; export default class NotionPage extends Component { initComponent() { this.$page = document.createElement('div'); - this.$page.className = 'notion-page'; + this.$page.className = TARGET.PAGE.NOTION; this.$target.appendChild(this.$page); } From ef9774e42ad25a7405cabde34eb477c2bcab8abe Mon Sep 17 00:00:00 2001 From: kutta97 Date: Thu, 6 Jul 2023 06:38:19 +0900 Subject: [PATCH 54/87] =?UTF-8?q?feat:=20=EC=82=AC=EC=9D=B4=EB=93=9C?= =?UTF-8?q?=EB=B0=94=20=ED=86=A0=EA=B8=80=20=EA=B8=B0=EB=8A=A5=20=EC=B6=94?= =?UTF-8?q?=EA=B0=80?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- public/{expand.svg => collapsed.svg} | 0 public/expanded.svg | 3 + src/App.js | 7 +- src/components/DocumentList/DocumentList.js | 62 ++++++++---- .../DocumentList/Item/DocumentListItem.js | 51 +++++++--- .../NotionDocument/NotionDocument.js | 6 +- src/components/NotionSidebar/NotionSidebar.js | 9 +- src/models/NotionPageModel.js | 97 +++++++++++++++++++ src/pages/NotionPage.js | 29 +++--- src/utils/localStorage/documentStorage.js | 30 ++++++ src/utils/{ => localStorage}/localStorage.js | 3 +- src/utils/localStorage/sidebarStorage.js | 22 +++++ 12 files changed, 259 insertions(+), 60 deletions(-) rename public/{expand.svg => collapsed.svg} (100%) create mode 100644 public/expanded.svg create mode 100644 src/models/NotionPageModel.js create mode 100644 src/utils/localStorage/documentStorage.js rename src/utils/{ => localStorage}/localStorage.js (85%) create mode 100644 src/utils/localStorage/sidebarStorage.js diff --git a/public/expand.svg b/public/collapsed.svg similarity index 100% rename from public/expand.svg rename to public/collapsed.svg diff --git a/public/expanded.svg b/public/expanded.svg new file mode 100644 index 00000000..d1ea5be3 --- /dev/null +++ b/public/expanded.svg @@ -0,0 +1,3 @@ + + + diff --git a/src/App.js b/src/App.js index 0e5c1bd0..5cece59c 100644 --- a/src/App.js +++ b/src/App.js @@ -15,9 +15,7 @@ export default class App extends Component { } initChildComponents() { - this.$notionPage = new NotionPage(this.$target, { - getDocumentId: this.getDocumentId.bind(this), - }); + this.$notionPage = new NotionPage(this.$target); } getDocumentId() { @@ -28,6 +26,7 @@ export default class App extends Component { } route() { - this.$notionPage.setState(); + const documentId = this.getDocumentId(); + this.$notionPage.setState({ documentId }); } } diff --git a/src/components/DocumentList/DocumentList.js b/src/components/DocumentList/DocumentList.js index 80e0281f..76bf690d 100644 --- a/src/components/DocumentList/DocumentList.js +++ b/src/components/DocumentList/DocumentList.js @@ -2,6 +2,7 @@ import { SIDEBAR } from '@consts/target'; import { createDocument, deleteDocument } from '@api/document'; +import sidebarStorage from '@utils/localStorage/sidebarStorage'; import { history } from '@utils/router'; import Component from '@core/Component'; @@ -14,6 +15,7 @@ export default class DocumentList extends Component { setup() { this.state = { documentList: [], + expanded: {}, }; } @@ -24,6 +26,13 @@ export default class DocumentList extends Component { this.$target.appendChild(this.$documentList); } + handleExpandButtonClick = (id) => { + const { expanded } = this.state; + expanded[id] = !expanded[id]; + + this.setState({ expanded }); + }; + handleCreateIndsideButtonClick = async (id) => { const newDocument = await createDocument({ title: 'Untitled', parent: id }); history.push(`/documents/${newDocument.id}`); @@ -48,6 +57,11 @@ export default class DocumentList extends Component { } const { className } = $button; + if (className === SIDEBAR.DOCUMENT_LIST_ITEM.EXPAND_BUTTON) { + this.handleExpandButtonClick(id); + return; + } + if ( className === SIDEBAR.DOCUMENT_LIST_ITEM.BUTTON_CONTAINER.DELETE_BUTTON ) { @@ -67,34 +81,46 @@ export default class DocumentList extends Component { }); } - createDocumentItem(item, depth = 1) { - const $div = document.createElement('div'); + setState(newState) { + const { documentList } = newState; - new DocumentListItem($div, { - documentItem: item, - depth, - }); + if (documentList) { + const { expandedState } = sidebarStorage.getSidebarDataItem(); + documentList.forEach(({ id }) => { + expandedState[id] = expandedState[id] ?? false; + }); - const { documents: childItems } = item; + this.state = { + ...this.state, + expanded: expandedState, + }; - if (childItems.length > 0) { - const $ul = document.createElement('ul'); - childItems.forEach((childItem) => { - $ul.appendChild(this.createDocumentItem(childItem, depth + 1)); - }); - $div.appendChild($ul); + sidebarStorage.setSidebarDataItem({ expandedState }); + } + + const { expanded } = newState; + + if (expanded) { + sidebarStorage.setSidebarDataItem({ expandedState: expanded }); } - return $div; + super.setState(newState); } render() { - const { documentList } = this.state; + const { documentList, expanded } = this.state; this.$documentList.innerHTML = ''; - documentList.forEach((document) => { - const $documentItem = this.createDocumentItem(document); - this.$documentList.appendChild($documentItem); + documentList.forEach((doc) => { + const $div = document.createElement('div'); + + new DocumentListItem($div, { + documentItem: doc, + parents: [], + expanded, + }); + + this.$documentList.appendChild($div); }); } } diff --git a/src/components/DocumentList/Item/DocumentListItem.js b/src/components/DocumentList/Item/DocumentListItem.js index 554385a8..5fc93065 100644 --- a/src/components/DocumentList/Item/DocumentListItem.js +++ b/src/components/DocumentList/Item/DocumentListItem.js @@ -1,5 +1,6 @@ import AddIcon from '@public/add.svg'; -import ExpandIcon from '@public/expand.svg'; +import CollapsedIcon from '@public/collapsed.svg'; +import ExpandedIcon from '@public/expanded.svg'; import TrashIcon from '@public/trash.svg'; import { SIDEBAR } from '@consts/target'; @@ -10,35 +11,63 @@ import './DocumentListItem.css'; export default class DocumentListItem extends Component { initComponent() { - const { documentItem, depth } = this.props; + const { documentItem, parents, expanded } = this.props; this.$documentListItem = document.createElement('li'); this.$documentListItem.dataset.id = documentItem.id; this.$documentListItem.innerHTML = ` ${documentItem.title}
    - -
    `; - this.$documentListItem.style.paddingLeft = `${depth * 10}px`; + this.$documentListItem.style.paddingLeft = `${(parents.length + 1) * 10}px`; this.$target.appendChild(this.$documentListItem); } + createDocumentItem(item, parents, expanded) { + const $div = document.createElement('div'); + + new DocumentListItem($div, { + documentItem: item, + parents, + expanded, + }); + + return $div; + } + render() { - const { documentItem } = this.props; - const $title = this.$documentListItem.querySelector( - '.document-list-item-title' - ); - $title.textContent = documentItem.title; + const { documentItem, parents, expanded } = this.props; + const { id, documents: childItems } = documentItem; + + const newParents = [...parents, id]; + + if (expanded[id] && childItems.length > 0) { + childItems.forEach((document) => { + const $documentItem = this.createDocumentItem( + document, + newParents, + expanded + ); + this.$target.appendChild($documentItem); + }); + } } } diff --git a/src/components/NotionDocument/NotionDocument.js b/src/components/NotionDocument/NotionDocument.js index 4bee1d72..6c671d17 100644 --- a/src/components/NotionDocument/NotionDocument.js +++ b/src/components/NotionDocument/NotionDocument.js @@ -9,8 +9,9 @@ import './NotionDocument.css'; export default class NotionDocument extends Component { setup() { this.state = { + isVisible: false, documentData: { - id: this.props.getDocumentId(), + id: null, title: '', createdAt: '', updatedAt: '', @@ -45,8 +46,7 @@ export default class NotionDocument extends Component { } render() { - const { id: documentId } = this.state.documentData; - const isVisible = documentId !== null; + const { isVisible } = this.state; this.$document.style.visibility = isVisible ? 'visible' : 'hidden'; } diff --git a/src/components/NotionSidebar/NotionSidebar.js b/src/components/NotionSidebar/NotionSidebar.js index 1064fe6b..791d2b7c 100644 --- a/src/components/NotionSidebar/NotionSidebar.js +++ b/src/components/NotionSidebar/NotionSidebar.js @@ -29,23 +29,22 @@ export default class NotionSidebar extends Component { this.$createButton = new Button(this.$sidebar, { className: SIDEBAR.CREATE_BUTTON, textContent: 'add a document', - onClick: () => { - this.handleCreateButtonClick(); - }, + onClick: this.handleCreateButtonClick.bind(this), }); this.$documentList = new DocumentList(this.$sidebar); } - handleCreateButtonClick = async () => { + async handleCreateButtonClick() { const newDocument = await createDocument({ title: 'Untitled' }); history.push(`/documents/${newDocument.id}`); - }; + } setState(nextState) { super.setState(nextState); const { documentList } = this.state; + this.$documentList.setState({ documentList }); } } diff --git a/src/models/NotionPageModel.js b/src/models/NotionPageModel.js new file mode 100644 index 00000000..cd4a8c50 --- /dev/null +++ b/src/models/NotionPageModel.js @@ -0,0 +1,97 @@ +import { + createDocument, + deleteDocument, + getDocument, + getDocumentList, + updateDocument, +} from '@api/document'; + +import { getItem, setItem } from '@utils/localStorage/localStorage'; + +export default class NotionPageModel { + constructor() { + this.documentId = null; + this.documentList = []; + this.documentData = { + id: this.documentId, + title: '', + createdAt: '', + updatedAt: '', + content: null, + documents: [], + }; + } + + static instance = null; + + static getInstance() { + if (!NotionPageModel.instance) { + NotionPageModel.instance = new NotionPageModel(); + } + return NotionPageModel.instance; + } + + setDocumentId(id) { + this.documentId = id; + } + + async getDocumentList() { + try { + const documentList = await getDocumentList(); + this.documentList = documentList; + } catch (error) { + console.error(error); + } + } + + async getDocumentData() { + if (this.documentId === null) return; + + try { + const { documentId } = this; + const documentData = await getDocument(documentId); + this.documentData = documentData; + } catch (error) { + console.error(error); + } + } + + async createDocument(title, parent) { + try { + const newDocument = await createDocument({ title, parent }); + this.documentData = newDocument; + } catch (error) { + console.error(error); + } + } + + async updateDocument(id, document) { + this.saveDocumentToLocalStorage(document); + + try { + const newDocument = await updateDocument(id, document); + this.documentData = newDocument; + } catch (error) { + console.error(error); + } + } + + async deleteDocument(id) { + try { + await deleteDocument(id); + } catch (error) { + console.error(error); + } + } + + getDocumentFromLocalStorage(id) { + return getItem(`temp-post-${id}`); + } + + saveDocumentToLocalStorage(document) { + setItem(`temp-post-${document.id}`, { + ...document, + tempSaveDate: new Date(), + }); + } +} diff --git a/src/pages/NotionPage.js b/src/pages/NotionPage.js index 0edcf118..ffb71d6c 100644 --- a/src/pages/NotionPage.js +++ b/src/pages/NotionPage.js @@ -2,7 +2,7 @@ import { TARGET } from '@consts/target'; import { getDocument, getDocumentList, updateDocument } from '@api/document'; -import { setItem } from '@utils/localStorage'; +import documentStorage from '@utils/localStorage/documentStorage'; import Component from '@core/Component'; @@ -19,50 +19,43 @@ export default class NotionPage extends Component { } initChildComponents() { - const { getDocumentId } = this.props; - this.$sidebar = new NotionSidebar(this.$page, { getDocumentId }); + this.$sidebar = new NotionSidebar(this.$page); this.$document = new NotionDocument(this.$page, { - getDocumentId, onEdit: this.onEdit.bind(this), }); } - get documentId() { - const { getDocumentId } = this.props; - return getDocumentId(); - } - async fetchDocumentList() { - const { documentId } = this; + const { documentId } = this.state; const documentList = await getDocumentList(); this.$sidebar.setState({ documentId, documentList }); } async fetchDocumentData() { - const { documentId } = this; + const { documentId } = this.state; if (documentId === null) { - this.$document.setState({ documentData: { id: documentId } }); + this.$document.setState({ + isVisible: false, + documentData: { id: documentId }, + }); return; } const documentData = await getDocument(documentId); - this.$document.setState({ documentData }); + this.$document.setState({ isVisible: true, documentData }); } async onEdit(name, document) { - setItem(`temp-post-${document.id}`, { - ...document, - tempSaveDate: new Date(), - }); + documentStorage.setDocumentItem(document); const updatedDocument = await updateDocument(document.id, document); if (name === 'title') { this.fetchDocumentList(); } - this.$document.setState({ documentData: updatedDocument }); + this.$document.setState({ isVisible: true, documentData: updatedDocument }); } setState(newState) { diff --git a/src/utils/localStorage/documentStorage.js b/src/utils/localStorage/documentStorage.js new file mode 100644 index 00000000..4dff7d49 --- /dev/null +++ b/src/utils/localStorage/documentStorage.js @@ -0,0 +1,30 @@ +import { getItem, removeItem, setItem } from './localStorage'; + +const DOCUMENT_KEY = (id) => `temp-document-${id}`; + +const setDocumentItem = (document) => { + setItem(DOCUMENT_KEY(document.id), { + ...document, + tempSaveDate: new Date(), + }); +}; + +const getDocumentItem = (id) => + getItem(DOCUMENT_KEY(id), { + id: null, + title: '', + createdAt: '', + updatedAt: '', + content: null, + documents: [], + }); + +const removeDocumentItem = (id) => removeItem(DOCUMENT_KEY(id)); + +const documentStorage = { + setDocumentItem, + getDocumentItem, + removeDocumentItem, +}; + +export default documentStorage; diff --git a/src/utils/localStorage.js b/src/utils/localStorage/localStorage.js similarity index 85% rename from src/utils/localStorage.js rename to src/utils/localStorage/localStorage.js index 32bf8c51..a50fc72d 100644 --- a/src/utils/localStorage.js +++ b/src/utils/localStorage/localStorage.js @@ -11,7 +11,8 @@ export const setItem = (key, value) => { export const getItem = (key, defaultValue) => { try { const storedValue = storage.getItem(key); - return storedValue ? JSON.parse(storedValue) : defaultValue; + if (!storedValue) return defaultValue; + return JSON.parse(storedValue); } catch (e) { console.error(e); return defaultValue; diff --git a/src/utils/localStorage/sidebarStorage.js b/src/utils/localStorage/sidebarStorage.js new file mode 100644 index 00000000..0692f01d --- /dev/null +++ b/src/utils/localStorage/sidebarStorage.js @@ -0,0 +1,22 @@ +import { getItem, removeItem, setItem } from './localStorage'; + +const SIDEBAR_KEY = 'side-bar-state'; + +const setSidebarDataItem = (state) => { + setItem(SIDEBAR_KEY, state); +}; + +const getSidebarDataItem = () => + getItem(SIDEBAR_KEY, { + expandedState: {}, + }); + +const removeSidebarDataItem = () => removeItem(SIDEBAR_KEY); + +const sidebarStorage = { + setSidebarDataItem, + getSidebarDataItem, + removeSidebarDataItem, +}; + +export default sidebarStorage; From 073b2b4a92109dc27cf34ab0db12ff3040d0bfaf Mon Sep 17 00:00:00 2001 From: kutta97 Date: Thu, 6 Jul 2023 07:44:34 +0900 Subject: [PATCH 55/87] =?UTF-8?q?feat:=20=EC=8B=A4=EC=A0=9C=20=EB=85=B8?= =?UTF-8?q?=EC=85=98=20=EC=82=AC=EC=9D=B4=EB=93=9C=EB=B0=94=EC=99=80=20?= =?UTF-8?q?=EB=B9=84=EC=8A=B7=ED=95=98=EA=B2=8C=20=EB=94=94=EC=9E=90?= =?UTF-8?q?=EC=9D=B8=20=EC=88=98=EC=A0=95?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- public/add_circle.svg | 3 ++ src/components/Button/Button.js | 10 +++---- .../CreateButton/SidebarCreateButton.css | 26 +++++++++++++++++ .../CreateButton/SidebarCreateButton.js | 28 +++++++++++++++++++ .../NotionSidebar/NotionSidebar.css | 17 +++++++++-- src/components/NotionSidebar/NotionSidebar.js | 17 +++++++---- src/consts/target.js | 4 +++ 7 files changed, 92 insertions(+), 13 deletions(-) create mode 100644 public/add_circle.svg create mode 100644 src/components/NotionSidebar/CreateButton/SidebarCreateButton.css create mode 100644 src/components/NotionSidebar/CreateButton/SidebarCreateButton.js diff --git a/public/add_circle.svg b/public/add_circle.svg new file mode 100644 index 00000000..f4056d61 --- /dev/null +++ b/public/add_circle.svg @@ -0,0 +1,3 @@ + + + diff --git a/src/components/Button/Button.js b/src/components/Button/Button.js index 924c417d..197c5f09 100644 --- a/src/components/Button/Button.js +++ b/src/components/Button/Button.js @@ -3,17 +3,17 @@ import Component from '@core/Component'; export default class Button extends Component { initComponent() { const { className, textContent } = this.props; - this.$createButton = document.createElement('button'); + this.$button = document.createElement('button'); - this.$createButton.className = className; - this.$createButton.textContent = textContent; + this.$button.className = className; + this.$button.textContent = textContent; - this.$target.appendChild(this.$createButton); + this.$target.appendChild(this.$button); } setEvent() { const { onClick } = this.props; - this.$createButton.addEventListener('click', () => { + this.$button.addEventListener('click', () => { onClick(); }); } diff --git a/src/components/NotionSidebar/CreateButton/SidebarCreateButton.css b/src/components/NotionSidebar/CreateButton/SidebarCreateButton.css new file mode 100644 index 00000000..0371e5ab --- /dev/null +++ b/src/components/NotionSidebar/CreateButton/SidebarCreateButton.css @@ -0,0 +1,26 @@ +.document-create-button { + display: flex; + align-items: center; + margin: 1px 4px; + padding: 4px 10px; + gap: 8px; + + color: #19171199; + font-size: 14px; + font-weight: 600; + + background: none; + border: none; + border-radius: 5px; + cursor: pointer; + + &:hover { + background-color: #ededeb; + cursor: pointer; + } +} + +.document-create-button span { + width: 100%; + text-align: start; +} diff --git a/src/components/NotionSidebar/CreateButton/SidebarCreateButton.js b/src/components/NotionSidebar/CreateButton/SidebarCreateButton.js new file mode 100644 index 00000000..0e9706fd --- /dev/null +++ b/src/components/NotionSidebar/CreateButton/SidebarCreateButton.js @@ -0,0 +1,28 @@ +import AddCircleIcon from '@public/add_circle.svg'; + +import { SIDEBAR } from '@consts/target'; + +import Component from '@core/Component'; + +import './SidebarCreateButton.css'; + +export default class SidebarCreateButton extends Component { + initComponent() { + const { textContent } = this.props; + this.$button = document.createElement('button'); + this.$button.className = SIDEBAR.CREATE_BUTTON; + this.$button.innerHTML = ` + Image + ${textContent} + `; + + this.$target.appendChild(this.$button); + } + + setEvent() { + const { onClick } = this.props; + this.$button.addEventListener('click', () => { + onClick(); + }); + } +} diff --git a/src/components/NotionSidebar/NotionSidebar.css b/src/components/NotionSidebar/NotionSidebar.css index ff55e9ed..2ef21955 100644 --- a/src/components/NotionSidebar/NotionSidebar.css +++ b/src/components/NotionSidebar/NotionSidebar.css @@ -1,6 +1,8 @@ .notion-sidebar { display: flex; flex-direction: column; + gap: 12px; + min-width: 250px; width: 250px; height: 100%; padding: 20px 0; @@ -9,7 +11,16 @@ background-color: #fbfbfa; } -.document-create-button { - height: 30px; - margin: 5px 10px; +.sidebar-document-list-container { + display: flex; + flex-direction: column; + margin-top: 7px; +} + +.sidebar-document-list-container p { + padding: 0 14px; + margin: 0 6px; + color: #19171199; + font-size: 13px; + font-weight: 600; } diff --git a/src/components/NotionSidebar/NotionSidebar.js b/src/components/NotionSidebar/NotionSidebar.js index 791d2b7c..ec5daadf 100644 --- a/src/components/NotionSidebar/NotionSidebar.js +++ b/src/components/NotionSidebar/NotionSidebar.js @@ -6,9 +6,9 @@ import { history } from '@utils/router'; import Component from '@core/Component'; -import Button from '@components/Button/Button'; import DocumentList from '@components/DocumentList/DocumentList'; +import SidebarCreateButton from './CreateButton/SidebarCreateButton'; import './NotionSidebar.css'; export default class NotionSidebar extends Component { @@ -26,13 +26,20 @@ export default class NotionSidebar extends Component { } initChildComponents() { - this.$createButton = new Button(this.$sidebar, { - className: SIDEBAR.CREATE_BUTTON, - textContent: 'add a document', + this.$createButton = new SidebarCreateButton(this.$sidebar, { + textContent: 'New document', onClick: this.handleCreateButtonClick.bind(this), }); - this.$documentList = new DocumentList(this.$sidebar); + const $listContainer = document.createElement('div'); + $listContainer.className = SIDEBAR.CONTAINER.LIST; + $listContainer.innerHTML = ` +

    documents

    + `; + + this.$sidebar.appendChild($listContainer); + + this.$documentList = new DocumentList($listContainer); } async handleCreateButtonClick() { diff --git a/src/consts/target.js b/src/consts/target.js index 4754f6e9..1b6b7baf 100644 --- a/src/consts/target.js +++ b/src/consts/target.js @@ -7,6 +7,10 @@ export const TARGET = { export const SIDEBAR = { ROOT: 'notion-sidebar', + CONTAINER: { + BUTTON: 'sidebar-button-container', + LIST: 'sidebar-document-list-container', + }, CREATE_BUTTON: 'document-create-button', DOCUMENT_LIST: 'document-list', DOCUMENT_LIST_ITEM: { From c0f0ac70fa487c66505e824df723d52519975033 Mon Sep 17 00:00:00 2001 From: kutta97 Date: Thu, 6 Jul 2023 08:13:29 +0900 Subject: [PATCH 56/87] =?UTF-8?q?feat:=20Editor=20=EC=97=AC=EB=B0=B1=20?= =?UTF-8?q?=EB=B0=8F=20Heading=20=EC=8A=A4=ED=83=80=EC=9D=BC=20=EC=A0=81?= =?UTF-8?q?=EC=9A=A9?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../Editor/Content/NotionEditorContent.css | 27 ++++++++++++++++++- src/components/Editor/NotionEditor.css | 9 ++++++- .../Editor/Title/NotionEditorTitle.css | 7 ++++- 3 files changed, 40 insertions(+), 3 deletions(-) diff --git a/src/components/Editor/Content/NotionEditorContent.css b/src/components/Editor/Content/NotionEditorContent.css index a125b7ef..08eb7b47 100644 --- a/src/components/Editor/Content/NotionEditorContent.css +++ b/src/components/Editor/Content/NotionEditorContent.css @@ -3,7 +3,32 @@ flex-direction: column; width: 100%; height: 100%; - padding: 20px; + padding: 5px 10px; + font-size: 20px; + + border: none; + outline: none; + &:focus { + outline: none; + } +} + +.notion-editor-content div { + padding: 2px 3px; + font-size: 16px; +} + +.notion-editor-content h1 { + font-size: 30px; + margin-top: 10px; +} + +.notion-editor-content h2 { + font-size: 24px; + margin-top: 10px; +} +.notion-editor-content h3 { font-size: 20px; + margin-top: 10px; } diff --git a/src/components/Editor/NotionEditor.css b/src/components/Editor/NotionEditor.css index 7f0c744e..d4f45523 100644 --- a/src/components/Editor/NotionEditor.css +++ b/src/components/Editor/NotionEditor.css @@ -1,7 +1,14 @@ .notion-editor { display: flex; flex-direction: column; + gap: 20px; + width: 100%; + max-width: 900px; height: 100%; - padding: 20px 10px; + + margin: 70px auto 0 auto; + padding: 0px 96px; + + transition: 0.5s ease; } diff --git a/src/components/Editor/Title/NotionEditorTitle.css b/src/components/Editor/Title/NotionEditorTitle.css index 81dc0ad3..4f586a26 100644 --- a/src/components/Editor/Title/NotionEditorTitle.css +++ b/src/components/Editor/Title/NotionEditorTitle.css @@ -1,4 +1,9 @@ .notion-editor-title { - font-size: 25px; + font-size: 40px; padding: 5px 10px; + border: none; + outline: none; + &:focus { + outline: none; + } } From 67d010693239e0b9115e4a4b3622d8a45b78eaea Mon Sep 17 00:00:00 2001 From: kutta97 Date: Thu, 6 Jul 2023 08:18:12 +0900 Subject: [PATCH 57/87] =?UTF-8?q?feat:=20=ED=8E=B8=EC=A7=91=EA=B8=B0=20?= =?UTF-8?q?=ED=85=8D=EC=8A=A4=ED=8A=B8=20weight=20=EC=A0=81=EC=9A=A9?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- src/components/Editor/Content/NotionEditorContent.css | 3 +++ src/components/Editor/Title/NotionEditorTitle.css | 1 + 2 files changed, 4 insertions(+) diff --git a/src/components/Editor/Content/NotionEditorContent.css b/src/components/Editor/Content/NotionEditorContent.css index 08eb7b47..da3b8965 100644 --- a/src/components/Editor/Content/NotionEditorContent.css +++ b/src/components/Editor/Content/NotionEditorContent.css @@ -20,15 +20,18 @@ .notion-editor-content h1 { font-size: 30px; + font-weight: 600; margin-top: 10px; } .notion-editor-content h2 { font-size: 24px; + font-weight: 600; margin-top: 10px; } .notion-editor-content h3 { font-size: 20px; + font-weight: 600; margin-top: 10px; } diff --git a/src/components/Editor/Title/NotionEditorTitle.css b/src/components/Editor/Title/NotionEditorTitle.css index 4f586a26..7d876583 100644 --- a/src/components/Editor/Title/NotionEditorTitle.css +++ b/src/components/Editor/Title/NotionEditorTitle.css @@ -1,5 +1,6 @@ .notion-editor-title { font-size: 40px; + font-weight: bold; padding: 5px 10px; border: none; outline: none; From 9c1ce568a14ee03dd77d200c12abb9af6a49d813 Mon Sep 17 00:00:00 2001 From: kutta97 Date: Thu, 6 Jul 2023 14:43:43 +0900 Subject: [PATCH 58/87] =?UTF-8?q?feat:=20=ED=98=84=EC=9E=AC=20=EC=84=A0?= =?UTF-8?q?=ED=83=9D=EC=A4=91=EC=9D=B8=20=ED=8E=98=EC=9D=B4=EC=A7=80=20?= =?UTF-8?q?=EC=82=AC=EC=9D=B4=EB=93=9C=EB=B0=94=EC=97=90=20=ED=91=9C?= =?UTF-8?q?=EC=8B=9C=ED=95=98=EB=8A=94=20=EA=B8=B0=EB=8A=A5=20=EC=B6=94?= =?UTF-8?q?=EA=B0=80?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- src/components/DocumentList/DocumentList.js | 4 +++- .../DocumentList/Item/DocumentListItem.css | 17 +++++++++++++---- .../DocumentList/Item/DocumentListItem.js | 14 +++++++++++--- src/components/NotionSidebar/NotionSidebar.js | 5 +++-- src/consts/target.js | 1 + 5 files changed, 31 insertions(+), 10 deletions(-) diff --git a/src/components/DocumentList/DocumentList.js b/src/components/DocumentList/DocumentList.js index 76bf690d..270a726c 100644 --- a/src/components/DocumentList/DocumentList.js +++ b/src/components/DocumentList/DocumentList.js @@ -14,6 +14,7 @@ import './DocumentList.css'; export default class DocumentList extends Component { setup() { this.state = { + documentId: null, documentList: [], expanded: {}, }; @@ -108,7 +109,7 @@ export default class DocumentList extends Component { } render() { - const { documentList, expanded } = this.state; + const { documentId, documentList, expanded } = this.state; this.$documentList.innerHTML = ''; documentList.forEach((doc) => { @@ -117,6 +118,7 @@ export default class DocumentList extends Component { new DocumentListItem($div, { documentItem: doc, parents: [], + selectedId: documentId, expanded, }); diff --git a/src/components/DocumentList/Item/DocumentListItem.css b/src/components/DocumentList/Item/DocumentListItem.css index 1f2db147..a63b1a10 100644 --- a/src/components/DocumentList/Item/DocumentListItem.css +++ b/src/components/DocumentList/Item/DocumentListItem.css @@ -1,4 +1,4 @@ -.document-list li { +.document-list-item { display: flex; flex-direction: row; align-items: center; @@ -17,14 +17,23 @@ } } -.document-list li a { +.document-list-item[selected='true'] { + background-color: #f2f2f1; + + &:hover { + background-color: #ededeb; + cursor: pointer; + } +} + +.document-list-item a { width: 100%; overflow: hidden; text-overflow: ellipsis; white-space: nowrap; } -.document-list li button { +.document-list-item button { display: flex; width: 22px; height: 22px; @@ -41,7 +50,7 @@ } } -.document-list li:hover .document-list-button-container { +.document-list-item:hover .document-list-button-container { display: flex; opacity: 1; } diff --git a/src/components/DocumentList/Item/DocumentListItem.js b/src/components/DocumentList/Item/DocumentListItem.js index 5fc93065..fa3eed03 100644 --- a/src/components/DocumentList/Item/DocumentListItem.js +++ b/src/components/DocumentList/Item/DocumentListItem.js @@ -11,9 +11,15 @@ import './DocumentListItem.css'; export default class DocumentListItem extends Component { initComponent() { - const { documentItem, parents, expanded } = this.props; + const { documentItem, parents, selectedId, expanded } = this.props; this.$documentListItem = document.createElement('li'); + this.$documentListItem.className = SIDEBAR.DOCUMENT_LIST_ITEM.ROOT; + + if (Number(documentItem.id) === Number(selectedId)) { + this.$documentListItem.setAttribute('selected', 'true'); + } + this.$documentListItem.dataset.id = documentItem.id; this.$documentListItem.innerHTML = ` @@ -41,12 +47,13 @@ export default class DocumentListItem extends Component { this.$target.appendChild(this.$documentListItem); } - createDocumentItem(item, parents, expanded) { + createDocumentItem(item, parents, selectedId, expanded) { const $div = document.createElement('div'); new DocumentListItem($div, { documentItem: item, parents, + selectedId, expanded, }); @@ -54,7 +61,7 @@ export default class DocumentListItem extends Component { } render() { - const { documentItem, parents, expanded } = this.props; + const { documentItem, parents, selectedId, expanded } = this.props; const { id, documents: childItems } = documentItem; const newParents = [...parents, id]; @@ -64,6 +71,7 @@ export default class DocumentListItem extends Component { const $documentItem = this.createDocumentItem( document, newParents, + selectedId, expanded ); this.$target.appendChild($documentItem); diff --git a/src/components/NotionSidebar/NotionSidebar.js b/src/components/NotionSidebar/NotionSidebar.js index ec5daadf..06c20c9b 100644 --- a/src/components/NotionSidebar/NotionSidebar.js +++ b/src/components/NotionSidebar/NotionSidebar.js @@ -14,6 +14,7 @@ import './NotionSidebar.css'; export default class NotionSidebar extends Component { setup() { this.state = { + documentId: null, documentList: [], }; } @@ -50,8 +51,8 @@ export default class NotionSidebar extends Component { setState(nextState) { super.setState(nextState); - const { documentList } = this.state; + const { documentId, documentList } = this.state; - this.$documentList.setState({ documentList }); + this.$documentList.setState({ documentId, documentList }); } } diff --git a/src/consts/target.js b/src/consts/target.js index 1b6b7baf..249db93f 100644 --- a/src/consts/target.js +++ b/src/consts/target.js @@ -14,6 +14,7 @@ export const SIDEBAR = { CREATE_BUTTON: 'document-create-button', DOCUMENT_LIST: 'document-list', DOCUMENT_LIST_ITEM: { + ROOT: 'document-list-item', TITLE: 'document-list-item-title', EXPAND_BUTTON: 'document-list-item-expand-button', BUTTON_CONTAINER: { From f2756a4e0fe512bcd78752981b93a220206155ca Mon Sep 17 00:00:00 2001 From: kutta97 Date: Thu, 6 Jul 2023 15:14:34 +0900 Subject: [PATCH 59/87] =?UTF-8?q?feat:=20=EC=82=AC=EC=9D=B4=EB=93=9C?= =?UTF-8?q?=EB=B0=94=20expand=20=EB=B2=84=ED=8A=BC=20=ED=8E=BC=EC=B3=A4?= =?UTF-8?q?=EC=9D=84=20=EB=95=8C,=20=EC=9E=90=EC=8B=9D=20=ED=8E=98?= =?UTF-8?q?=EC=9D=B4=EC=A7=80=20=EC=97=86=EB=8A=94=20=EA=B2=BD=EC=9A=B0=20?= =?UTF-8?q?=EB=B3=B4=EC=97=AC=EC=A3=BC=EB=8A=90=20=EA=B8=B0=EB=8A=A5=20?= =?UTF-8?q?=EC=B6=94=EA=B0=80?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- src/components/DocumentList/DocumentList.js | 2 ++ .../EmptyItem/DocumentListEmptyItem.css | 21 ++++++++++++++ .../EmptyItem/DocumentListEmptyItem.js | 23 +++++++++++++++ .../DocumentList/Item/DocumentListItem.js | 29 ++++++++++++------- src/consts/target.js | 1 + 5 files changed, 66 insertions(+), 10 deletions(-) create mode 100644 src/components/DocumentList/EmptyItem/DocumentListEmptyItem.css create mode 100644 src/components/DocumentList/EmptyItem/DocumentListEmptyItem.js diff --git a/src/components/DocumentList/DocumentList.js b/src/components/DocumentList/DocumentList.js index 270a726c..9d8a2b6e 100644 --- a/src/components/DocumentList/DocumentList.js +++ b/src/components/DocumentList/DocumentList.js @@ -49,6 +49,8 @@ export default class DocumentList extends Component { const $li = target.closest('li'); if (!$li) return; + if ($li.className === SIDEBAR.DOCUMENT_LIST_ITEM.EMPTY) return; + const { id } = $li.dataset; const $button = target.closest('button'); diff --git a/src/components/DocumentList/EmptyItem/DocumentListEmptyItem.css b/src/components/DocumentList/EmptyItem/DocumentListEmptyItem.css new file mode 100644 index 00000000..cd5ae8d6 --- /dev/null +++ b/src/components/DocumentList/EmptyItem/DocumentListEmptyItem.css @@ -0,0 +1,21 @@ +.document-list-empty-item { + display: flex; + flex-direction: row; + align-items: center; + width: 100%; + height: 27px; + margin: 1px 0; + padding: 2px 10px; + border-radius: 3px; + + color: #bcbcbb; + font-size: 14px; + font-weight: 400; +} + +.document-list-empty-item a { + width: 100%; + overflow: hidden; + text-overflow: ellipsis; + white-space: nowrap; +} diff --git a/src/components/DocumentList/EmptyItem/DocumentListEmptyItem.js b/src/components/DocumentList/EmptyItem/DocumentListEmptyItem.js new file mode 100644 index 00000000..78c23d6c --- /dev/null +++ b/src/components/DocumentList/EmptyItem/DocumentListEmptyItem.js @@ -0,0 +1,23 @@ +import { SIDEBAR } from '@consts/target'; + +import Component from '@core/Component'; + +import './DocumentListEmptyItem.css'; + +export default class DocumentListEmptyItem extends Component { + initComponent() { + const { parents } = this.props; + + this.$documentListEmptyItem = document.createElement('li'); + this.$documentListEmptyItem.className = SIDEBAR.DOCUMENT_LIST_ITEM.EMPTY; + + this.$documentListEmptyItem.innerHTML = ` + No documents inside + `; + this.$documentListEmptyItem.style.paddingLeft = `${ + parents.length * 10 + 22 + }px`; + + this.$target.appendChild(this.$documentListEmptyItem); + } +} diff --git a/src/components/DocumentList/Item/DocumentListItem.js b/src/components/DocumentList/Item/DocumentListItem.js index fa3eed03..be9cc0fc 100644 --- a/src/components/DocumentList/Item/DocumentListItem.js +++ b/src/components/DocumentList/Item/DocumentListItem.js @@ -7,6 +7,7 @@ import { SIDEBAR } from '@consts/target'; import Component from '@core/Component'; +import DocumentListEmptyItem from '../EmptyItem/DocumentListEmptyItem'; import './DocumentListItem.css'; export default class DocumentListItem extends Component { @@ -66,16 +67,24 @@ export default class DocumentListItem extends Component { const newParents = [...parents, id]; - if (expanded[id] && childItems.length > 0) { - childItems.forEach((document) => { - const $documentItem = this.createDocumentItem( - document, - newParents, - selectedId, - expanded - ); - this.$target.appendChild($documentItem); - }); + if (!expanded[id]) return; + + if (childItems.length === 0) { + const $documentEmptyItem = document.createElement('div'); + + new DocumentListEmptyItem($documentEmptyItem, { parents: newParents }); + this.$target.appendChild($documentEmptyItem); + return; } + + childItems.forEach((document) => { + const $documentItem = this.createDocumentItem( + document, + newParents, + selectedId, + expanded + ); + this.$target.appendChild($documentItem); + }); } } diff --git a/src/consts/target.js b/src/consts/target.js index 249db93f..34b4939a 100644 --- a/src/consts/target.js +++ b/src/consts/target.js @@ -15,6 +15,7 @@ export const SIDEBAR = { DOCUMENT_LIST: 'document-list', DOCUMENT_LIST_ITEM: { ROOT: 'document-list-item', + EMPTY: 'document-list-empty-item', TITLE: 'document-list-item-title', EXPAND_BUTTON: 'document-list-item-expand-button', BUTTON_CONTAINER: { From c5b65a82db9924d30bf83bd7d2916e9fe1d5b14c Mon Sep 17 00:00:00 2001 From: kutta97 Date: Thu, 6 Jul 2023 16:49:39 +0900 Subject: [PATCH 60/87] =?UTF-8?q?feat:=20=EC=82=AC=EC=9D=B4=EB=93=9C?= =?UTF-8?q?=EB=B0=94=EC=99=80=20=ED=8E=B8=EC=A7=91=EA=B8=B0=20=EA=B0=81?= =?UTF-8?q?=EA=B0=81=20overflow=20=EC=8A=A4=ED=81=AC=EB=A1=A4=20=EA=B8=B0?= =?UTF-8?q?=EB=8A=A5=20=EC=B6=94=EA=B0=80?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- src/App.css | 2 ++ src/components/DocumentList/DocumentList.css | 2 -- src/components/Editor/NotionEditor.css | 4 +--- src/components/NotionDocument/NotionDocument.css | 10 ++++++++++ src/components/NotionDocument/NotionDocument.js | 6 +++++- src/components/NotionSidebar/NotionSidebar.css | 2 ++ src/pages/NotionPage.css | 2 +- 7 files changed, 21 insertions(+), 7 deletions(-) diff --git a/src/App.css b/src/App.css index af715397..33e0b4af 100644 --- a/src/App.css +++ b/src/App.css @@ -13,4 +13,6 @@ body { background-color: #fff; color: #37352f; font-family: 'Pretendard'; + + overflow: hidden; } diff --git a/src/components/DocumentList/DocumentList.css b/src/components/DocumentList/DocumentList.css index b8d9ffd0..5a488edb 100644 --- a/src/components/DocumentList/DocumentList.css +++ b/src/components/DocumentList/DocumentList.css @@ -2,6 +2,4 @@ width: 100%; height: 100%; padding: 10px 5px; - - overflow-y: scroll; } diff --git a/src/components/Editor/NotionEditor.css b/src/components/Editor/NotionEditor.css index d4f45523..7d70db82 100644 --- a/src/components/Editor/NotionEditor.css +++ b/src/components/Editor/NotionEditor.css @@ -7,8 +7,6 @@ max-width: 900px; height: 100%; - margin: 70px auto 0 auto; + margin: 70px 20px 200px 20px; padding: 0px 96px; - - transition: 0.5s ease; } diff --git a/src/components/NotionDocument/NotionDocument.css b/src/components/NotionDocument/NotionDocument.css index 8060d48c..f9a55af1 100644 --- a/src/components/NotionDocument/NotionDocument.css +++ b/src/components/NotionDocument/NotionDocument.css @@ -1,3 +1,13 @@ .notion-document { width: 100%; + height: 100%; +} + +.notion-editor-container { + display: flex; + justify-content: center; + width: 100%; + height: 100%; + + overflow-y: scroll; } diff --git a/src/components/NotionDocument/NotionDocument.js b/src/components/NotionDocument/NotionDocument.js index 6c671d17..224d1580 100644 --- a/src/components/NotionDocument/NotionDocument.js +++ b/src/components/NotionDocument/NotionDocument.js @@ -25,13 +25,17 @@ export default class NotionDocument extends Component { this.$document = document.createElement('div'); this.$document.className = DOCUMENT.ROOT; + this.$editorContainer = document.createElement('div'); + this.$editorContainer.className = 'notion-editor-container'; + + this.$document.appendChild(this.$editorContainer); this.$target.appendChild(this.$document); } initChildComponents() { const { onEdit } = this.props; - this.$editor = new NotionEditor(this.$document, { onEdit }); + this.$editor = new NotionEditor(this.$editorContainer, { onEdit }); } setState(nextState) { diff --git a/src/components/NotionSidebar/NotionSidebar.css b/src/components/NotionSidebar/NotionSidebar.css index 2ef21955..a951c50d 100644 --- a/src/components/NotionSidebar/NotionSidebar.css +++ b/src/components/NotionSidebar/NotionSidebar.css @@ -15,6 +15,8 @@ display: flex; flex-direction: column; margin-top: 7px; + + overflow-y: scroll; } .sidebar-document-list-container p { diff --git a/src/pages/NotionPage.css b/src/pages/NotionPage.css index 4e8d6471..85ea2b3d 100644 --- a/src/pages/NotionPage.css +++ b/src/pages/NotionPage.css @@ -1,5 +1,5 @@ .notion-page { + display: flex; width: 100%; height: 100%; - display: flex; } From 364bf25ec8ee91f2ed07cd78e5ec5be8dfa9fd05 Mon Sep 17 00:00:00 2001 From: kutta97 Date: Thu, 6 Jul 2023 16:50:55 +0900 Subject: [PATCH 61/87] =?UTF-8?q?feat:=20title=20=EC=9E=91=EC=84=B1=20?= =?UTF-8?q?=EC=A4=91=20Enter=20=EC=9E=85=EB=A0=A5=ED=95=98=EB=A9=B4=20cont?= =?UTF-8?q?ent=20=EB=A1=9C=20focus=20=EB=84=98=EC=96=B4=EA=B0=80=EB=8A=94?= =?UTF-8?q?=20=EA=B8=B0=EB=8A=A5=20=EC=B6=94=EA=B0=80?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../Editor/Content/NotionEditorContent.js | 5 ++++ src/components/Editor/NotionEditor.js | 10 ++++++++ .../Editor/Title/NotionEditorTitle.css | 4 ++++ .../Editor/Title/NotionEditorTitle.js | 23 +++++++++++++++---- 4 files changed, 38 insertions(+), 4 deletions(-) diff --git a/src/components/Editor/Content/NotionEditorContent.js b/src/components/Editor/Content/NotionEditorContent.js index b754faf7..6d9bfa37 100644 --- a/src/components/Editor/Content/NotionEditorContent.js +++ b/src/components/Editor/Content/NotionEditorContent.js @@ -100,6 +100,11 @@ export default class NotionEditorContent extends Component { ); } + setState(newState) { + if (newState?.content === this.state?.content) return; + super.setState(newState); + } + render() { const { content } = this.state; diff --git a/src/components/Editor/NotionEditor.js b/src/components/Editor/NotionEditor.js index 397690b0..0d92cf4d 100644 --- a/src/components/Editor/NotionEditor.js +++ b/src/components/Editor/NotionEditor.js @@ -15,6 +15,7 @@ export default class NotionEditor extends Component { initChildComponents() { this.$title = new NotionEditorTitle(this.$editor, { onEdit: this.handleEditorInputChange.bind(this), + onPressEnterKey: this.handlePressEnterKey.bind(this), }); this.$content = new NotionEditorContent(this.$editor, { onEdit: this.handleEditorInputChange.bind(this), @@ -30,6 +31,15 @@ export default class NotionEditor extends Component { }); } + handlePressEnterKey(name) { + if (name === 'title') { + const $contentEditor = this.$target.querySelector( + '.notion-editor-content' + ); + $contentEditor.focus(); + } + } + setState(newstate) { super.setState(newstate); diff --git a/src/components/Editor/Title/NotionEditorTitle.css b/src/components/Editor/Title/NotionEditorTitle.css index 7d876583..65bdfbcb 100644 --- a/src/components/Editor/Title/NotionEditorTitle.css +++ b/src/components/Editor/Title/NotionEditorTitle.css @@ -1,4 +1,8 @@ .notion-editor-title { + resize: none; + overflow: hidden; + + font-family: 'Pretendard'; font-size: 40px; font-weight: bold; padding: 5px 10px; diff --git a/src/components/Editor/Title/NotionEditorTitle.js b/src/components/Editor/Title/NotionEditorTitle.js index dac74340..d7b62f00 100644 --- a/src/components/Editor/Title/NotionEditorTitle.js +++ b/src/components/Editor/Title/NotionEditorTitle.js @@ -12,20 +12,29 @@ export default class NotionEditorTitle extends Component { } initComponent() { - this.$titleEditor = document.createElement('input'); + this.$titleEditor = document.createElement('textarea'); this.$titleEditor.className = 'notion-editor-title'; - this.$titleEditor.type = 'text'; this.$titleEditor.dataset.name = 'title'; - this.$titleEditor.placeholder = 'undefined'; + this.$titleEditor.placeholder = 'Untitled'; this.$target.appendChild(this.$titleEditor); } setEvent() { - const { onEdit } = this.props; + const { onEdit, onPressEnterKey } = this.props; + + this.$titleEditor.addEventListener('keypress', (e) => { + const { key } = e; + if (key !== 'Enter') return; + + e.preventDefault(); + onPressEnterKey('title'); + }); this.$titleEditor.addEventListener('input', ({ target }) => { + this.$titleEditor.style.height = `${this.$titleEditor.scrollHeight}px`; + if (this.timer !== null) { clearTimeout(this.timer); } @@ -35,9 +44,15 @@ export default class NotionEditorTitle extends Component { }); } + setState(newState) { + if (newState?.title === this.state?.title) return; + super.setState(newState); + } + render() { const { title } = this.state; this.$titleEditor.value = title ?? ''; + this.$titleEditor.style.height = `${40 + this.$titleEditor.scrollHeight}px`; } } From e2ebc8ac08627e9f7e9659f448fbfdda32182912 Mon Sep 17 00:00:00 2001 From: kutta97 Date: Thu, 6 Jul 2023 17:09:29 +0900 Subject: [PATCH 62/87] =?UTF-8?q?feat:=20=EB=AC=B8=EC=84=9C=20=ED=83=80?= =?UTF-8?q?=EC=9D=B4=ED=8B=80=20=ED=8E=B8=EC=A7=91=EA=B8=B0=EB=8F=84=20con?= =?UTF-8?q?tenteditable=20=EB=A1=9C=20=EB=B3=80=EA=B2=BD?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../Editor/Content/NotionEditorContent.css | 2 +- .../Editor/Content/NotionEditorContent.js | 2 +- src/components/Editor/NotionEditor.css | 2 +- .../Editor/Title/NotionEditorTitle.css | 3 +-- .../Editor/Title/NotionEditorTitle.js | 19 ++++++++++++------- 5 files changed, 16 insertions(+), 12 deletions(-) diff --git a/src/components/Editor/Content/NotionEditorContent.css b/src/components/Editor/Content/NotionEditorContent.css index da3b8965..efb5527f 100644 --- a/src/components/Editor/Content/NotionEditorContent.css +++ b/src/components/Editor/Content/NotionEditorContent.css @@ -2,7 +2,7 @@ display: flex; flex-direction: column; width: 100%; - height: 100%; + height: fit-content; padding: 5px 10px; font-size: 20px; diff --git a/src/components/Editor/Content/NotionEditorContent.js b/src/components/Editor/Content/NotionEditorContent.js index 6d9bfa37..8db03101 100644 --- a/src/components/Editor/Content/NotionEditorContent.js +++ b/src/components/Editor/Content/NotionEditorContent.js @@ -108,7 +108,7 @@ export default class NotionEditorContent extends Component { render() { const { content } = this.state; - this.$contentEditor.innerHTML = Editor.parse(content, this.caret); + this.$contentEditor.innerHTML = Editor.parse(content); Editor.restoreCaretPosition(); } diff --git a/src/components/Editor/NotionEditor.css b/src/components/Editor/NotionEditor.css index 7d70db82..91013f71 100644 --- a/src/components/Editor/NotionEditor.css +++ b/src/components/Editor/NotionEditor.css @@ -5,7 +5,7 @@ width: 100%; max-width: 900px; - height: 100%; + height: fit-content; margin: 70px 20px 200px 20px; padding: 0px 96px; diff --git a/src/components/Editor/Title/NotionEditorTitle.css b/src/components/Editor/Title/NotionEditorTitle.css index 65bdfbcb..1a0c7e8c 100644 --- a/src/components/Editor/Title/NotionEditorTitle.css +++ b/src/components/Editor/Title/NotionEditorTitle.css @@ -1,6 +1,5 @@ .notion-editor-title { - resize: none; - overflow: hidden; + height: fit-content; font-family: 'Pretendard'; font-size: 40px; diff --git a/src/components/Editor/Title/NotionEditorTitle.js b/src/components/Editor/Title/NotionEditorTitle.js index d7b62f00..2af73ed8 100644 --- a/src/components/Editor/Title/NotionEditorTitle.js +++ b/src/components/Editor/Title/NotionEditorTitle.js @@ -12,11 +12,11 @@ export default class NotionEditorTitle extends Component { } initComponent() { - this.$titleEditor = document.createElement('textarea'); + this.$titleEditor = document.createElement('div'); + this.$titleEditor.contentEditable = true; this.$titleEditor.className = 'notion-editor-title'; this.$titleEditor.dataset.name = 'title'; - this.$titleEditor.placeholder = 'Untitled'; this.$target.appendChild(this.$titleEditor); } @@ -24,6 +24,12 @@ export default class NotionEditorTitle extends Component { setEvent() { const { onEdit, onPressEnterKey } = this.props; + this.$titleEditor.addEventListener('paste', (e) => { + e.preventDefault(); + const text = e.clipboardData.getData('text/plain'); + document.execCommand('insertHTML', false, text); + }); + this.$titleEditor.addEventListener('keypress', (e) => { const { key } = e; if (key !== 'Enter') return; @@ -33,13 +39,11 @@ export default class NotionEditorTitle extends Component { }); this.$titleEditor.addEventListener('input', ({ target }) => { - this.$titleEditor.style.height = `${this.$titleEditor.scrollHeight}px`; - if (this.timer !== null) { clearTimeout(this.timer); } this.timer = setTimeout(async () => { - onEdit('title', target.value); + onEdit('title', target.innerHTML); }, 1000); }); } @@ -52,7 +56,8 @@ export default class NotionEditorTitle extends Component { render() { const { title } = this.state; - this.$titleEditor.value = title ?? ''; - this.$titleEditor.style.height = `${40 + this.$titleEditor.scrollHeight}px`; + if (title === this.$titleEditor.innerHTML) return; + + this.$titleEditor.innerHTML = title; } } From 36e15f54cc8733c9587ce0c9e0bb1fce9572accc Mon Sep 17 00:00:00 2001 From: kutta97 Date: Thu, 6 Jul 2023 17:41:41 +0900 Subject: [PATCH 63/87] =?UTF-8?q?feat:=20=EB=B9=88=20=EC=A0=9C=EB=AA=A9?= =?UTF-8?q?=EC=9D=B8=20=EA=B2=BD=EC=9A=B0=20Untitled=20placeholder=20?= =?UTF-8?q?=EA=B8=B0=EB=8A=A5=20=EC=B6=94=EA=B0=80?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- src/components/DocumentList/DocumentList.js | 2 +- .../DocumentList/Item/DocumentListItem.js | 4 +++- src/components/Editor/NotionEditor.js | 21 +++++++++++++++---- .../Editor/Title/NotionEditorTitle.css | 6 ++++++ .../Editor/Title/NotionEditorTitle.js | 1 + src/components/NotionSidebar/NotionSidebar.js | 2 +- 6 files changed, 29 insertions(+), 7 deletions(-) diff --git a/src/components/DocumentList/DocumentList.js b/src/components/DocumentList/DocumentList.js index 9d8a2b6e..62133a39 100644 --- a/src/components/DocumentList/DocumentList.js +++ b/src/components/DocumentList/DocumentList.js @@ -35,7 +35,7 @@ export default class DocumentList extends Component { }; handleCreateIndsideButtonClick = async (id) => { - const newDocument = await createDocument({ title: 'Untitled', parent: id }); + const newDocument = await createDocument({ title: '', parent: id }); history.push(`/documents/${newDocument.id}`); }; diff --git a/src/components/DocumentList/Item/DocumentListItem.js b/src/components/DocumentList/Item/DocumentListItem.js index be9cc0fc..f84aa6a1 100644 --- a/src/components/DocumentList/Item/DocumentListItem.js +++ b/src/components/DocumentList/Item/DocumentListItem.js @@ -29,7 +29,9 @@ export default class DocumentListItem extends Component { expanded[documentItem.id] ? ExpandedIcon : CollapsedIcon }" alt="Image" heigth="12" width="12"> - ${documentItem.title} + ${ + documentItem.title || 'Untitled' + }