From ca08caaba868f82e17507805e28c94ced7cb7447 Mon Sep 17 00:00:00 2001 From: howooking Date: Fri, 11 Aug 2023 00:33:29 +0900 Subject: [PATCH 1/3] =?UTF-8?q?=EB=B0=B0=ED=8F=AC?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .eslintrc.cjs | 20 + .gitignore | 27 + .prettierrc.json | 7 + README.md | 38 +- index.html | 17 + package-lock.json | 5919 +++++++++++++++++++ package.json | 39 + public/vite.svg | 1 + src/App.tsx | 47 + src/api/auth/checkEmail.ts | 20 + src/api/auth/cloudinary.ts | 20 + src/api/auth/signin.ts | 11 + src/api/auth/signout.ts | 6 + src/api/auth/signup.ts | 21 + src/api/auth/verification.ts | 9 + src/api/customAxios.ts | 115 + src/api/home/getUserHeader.ts | 6 + src/api/home/pendingList.ts | 8 + src/api/home/scheduleList.ts | 6 + src/api/myAccount/admin.ts | 32 + src/api/myAccount/changeMyInfo.ts | 10 + src/api/myAccount/getMyAccount.ts | 7 + src/api/mySchedule.ts | 23 + src/assets/defaultProfile.png | Bin 0 -> 45192 bytes src/assets/favicon.png | Bin 0 -> 16292 bytes src/components/MyHeader.tsx | 175 + src/components/MyLayout.tsx | 15 + src/components/ProtectedManagerRoute.tsx | 12 + src/components/ProtectedRoute.tsx | 12 + src/components/RequesTag.tsx | 23 + src/components/UserInfo.tsx | 41 + src/data/constants.ts | 48 + src/data/dummyData.ts | 58 + src/index.css | 50 + src/main.tsx | 16 + src/page/home/calendar.tsx | 147 + src/page/home/home.tsx | 371 ++ src/page/home/mySchedule.tsx | 159 + src/page/home/signin.tsx | 146 + src/page/myAccount/admin/adminLayout.tsx | 32 + src/page/myAccount/admin/approve.tsx | 222 + src/page/myAccount/admin/promote.tsx | 131 + src/page/myAccount/admin/promotionModal.tsx | 145 + src/page/myAccount/myAccount.tsx | 348 ++ src/page/myAccount/myAccoutLayout.tsx | 97 + src/page/myAccount/passwordChangeModal.tsx | 180 + src/page/myAccount/vacation.tsx | 229 + src/page/notFound/notFountd.tsx | 12 + src/page/signup/signup.tsx | 549 ++ src/recoil/AccessTokkenAtom.ts | 7 + src/recoil/IsManagerAtom.ts | 6 + src/recoil/ReRenderStateAtom.ts | 6 + src/recoil/UserEmailAtom.ts | 6 + src/types/IMySchdule.ts | 8 + src/utils/cookies.ts | 27 + src/utils/formatPhonenumber.ts | 6 + src/utils/getPayloadFromJWT.ts | 13 + src/vite-env.d.ts | 1 + tsconfig.json | 29 + tsconfig.node.json | 10 + vite.config.ts | 13 + 61 files changed, 9740 insertions(+), 19 deletions(-) create mode 100644 .eslintrc.cjs create mode 100644 .gitignore create mode 100644 .prettierrc.json create mode 100644 index.html create mode 100644 package-lock.json create mode 100644 package.json create mode 100644 public/vite.svg create mode 100644 src/App.tsx create mode 100644 src/api/auth/checkEmail.ts create mode 100644 src/api/auth/cloudinary.ts create mode 100644 src/api/auth/signin.ts create mode 100644 src/api/auth/signout.ts create mode 100644 src/api/auth/signup.ts create mode 100644 src/api/auth/verification.ts create mode 100644 src/api/customAxios.ts create mode 100644 src/api/home/getUserHeader.ts create mode 100644 src/api/home/pendingList.ts create mode 100644 src/api/home/scheduleList.ts create mode 100644 src/api/myAccount/admin.ts create mode 100644 src/api/myAccount/changeMyInfo.ts create mode 100644 src/api/myAccount/getMyAccount.ts create mode 100644 src/api/mySchedule.ts create mode 100644 src/assets/defaultProfile.png create mode 100644 src/assets/favicon.png create mode 100644 src/components/MyHeader.tsx create mode 100644 src/components/MyLayout.tsx create mode 100644 src/components/ProtectedManagerRoute.tsx create mode 100644 src/components/ProtectedRoute.tsx create mode 100644 src/components/RequesTag.tsx create mode 100644 src/components/UserInfo.tsx create mode 100644 src/data/constants.ts create mode 100644 src/data/dummyData.ts create mode 100644 src/index.css create mode 100644 src/main.tsx create mode 100644 src/page/home/calendar.tsx create mode 100644 src/page/home/home.tsx create mode 100644 src/page/home/mySchedule.tsx create mode 100644 src/page/home/signin.tsx create mode 100644 src/page/myAccount/admin/adminLayout.tsx create mode 100644 src/page/myAccount/admin/approve.tsx create mode 100644 src/page/myAccount/admin/promote.tsx create mode 100644 src/page/myAccount/admin/promotionModal.tsx create mode 100644 src/page/myAccount/myAccount.tsx create mode 100644 src/page/myAccount/myAccoutLayout.tsx create mode 100644 src/page/myAccount/passwordChangeModal.tsx create mode 100644 src/page/myAccount/vacation.tsx create mode 100644 src/page/notFound/notFountd.tsx create mode 100644 src/page/signup/signup.tsx create mode 100644 src/recoil/AccessTokkenAtom.ts create mode 100644 src/recoil/IsManagerAtom.ts create mode 100644 src/recoil/ReRenderStateAtom.ts create mode 100644 src/recoil/UserEmailAtom.ts create mode 100644 src/types/IMySchdule.ts create mode 100644 src/utils/cookies.ts create mode 100644 src/utils/formatPhonenumber.ts create mode 100644 src/utils/getPayloadFromJWT.ts create mode 100644 src/vite-env.d.ts create mode 100644 tsconfig.json create mode 100644 tsconfig.node.json create mode 100644 vite.config.ts diff --git a/.eslintrc.cjs b/.eslintrc.cjs new file mode 100644 index 00000000..05123974 --- /dev/null +++ b/.eslintrc.cjs @@ -0,0 +1,20 @@ +module.exports = { + root: true, + env: { browser: true, es2020: true }, + extends: [ + 'eslint:recommended', + 'plugin:@typescript-eslint/recommended', + 'plugin:react-hooks/recommended', + ], + ignorePatterns: ['dist', '.eslintrc.cjs'], + parser: '@typescript-eslint/parser', + plugins: ['react-refresh'], + rules: { + 'react-refresh/only-export-components': [ + 'warn', + { allowConstantExport: true }, + ], + 'no-console': ['warn'], + 'no-unused-vars': ['warn'], + }, +}; diff --git a/.gitignore b/.gitignore new file mode 100644 index 00000000..f4acd26d --- /dev/null +++ b/.gitignore @@ -0,0 +1,27 @@ +# Logs +logs +*.log +npm-debug.log* +yarn-debug.log* +yarn-error.log* +pnpm-debug.log* +lerna-debug.log* + +node_modules +dist +dist-ssr +*.local + +# Editor directories and files +.vscode/* +!.vscode/extensions.json +.idea +.DS_Store +*.suo +*.ntvs* +*.njsproj +*.sln +*.sw? + +.env +.env.d.ts \ No newline at end of file diff --git a/.prettierrc.json b/.prettierrc.json new file mode 100644 index 00000000..fa9699b8 --- /dev/null +++ b/.prettierrc.json @@ -0,0 +1,7 @@ +{ + "semi": true, + "trailingComma": "all", + "singleQuote": true, + "printWidth": 80, + "tabWidth": 2 +} diff --git a/README.md b/README.md index 733a8622..1ebe379f 100644 --- a/README.md +++ b/README.md @@ -1,27 +1,27 @@ -# ✔️ 미니 프로젝트_연차/당직 프로그램 만들기 +# React + TypeScript + Vite -## 필수 요구 사항 +This template provides a minimal setup to get React working in Vite with HMR and some ESLint rules. -- 로그인 / 회원가입 페이지 -- 개인 정보 수정 페이지 -- 사용자간 공유 게시 페이지 캘린더 사용 +Currently, two official plugins are available: -## 선택 요구 사항 +- [@vitejs/plugin-react](https://github.com/vitejs/vite-plugin-react/blob/main/packages/plugin-react/README.md) uses [Babel](https://babeljs.io/) for Fast Refresh +- [@vitejs/plugin-react-swc](https://github.com/vitejs/vite-plugin-react-swc) uses [SWC](https://swc.rs/) for Fast Refresh -- `useCallback`, `useMemo `등을 통한 컴포넌트 렌더링 최적화 -- 내가 작성한 코드를 팀원 중 누가봐도 쉽게 알아볼 수 있도록 고민하면서 작성해주세요. +## Expanding the ESLint configuration -## 과제 수행 및 제출 방법 +If you are developing a production application, we recommend updating the configuration to enable type aware lint rules: -1. 현재 저장소를 로컬에 클론(Clone)합니다. -2. 팀별로 브랜치를 생성합니다.(`git branch KDT5_TEAM_ABC`) -3. 팀별 브랜치에서 과제를 수행합니다. -4. 과제 수행이 완료되면, 자신의 본명 브랜치를 원격 저장소에 푸시(Push)합니다.(`main` 브랜치에 푸시하지 않도록 꼭 주의하세요, `git push origin KDT5_TEAM_ABC`) -5. 저장소에서 `main` 브랜치를 대상으로 Pull Request 생성하면, 과제 제출이 완료됩니다!(E.g, `main` <== `KDT5_TEAM_ABC`) +- Configure the top-level `parserOptions` property like this: -### 주의사항! +```js + parserOptions: { + ecmaVersion: 'latest', + sourceType: 'module', + project: ['./tsconfig.json', './tsconfig.node.json'], + tsconfigRootDir: __dirname, + }, +``` -- `main` 혹은 다른 사람의 브랜치로 절대 병합하지 않도록 주의하세요! -- Pull Request에서 보이는 설명을 다른 사람들이 이해하기 쉽도록 꼼꼼하게 작성하세요! -- Pull Request에서 과제 제출 후 절대 병합(Merge)하지 않도록 주의하세요! -- 과제 수행 및 제출 과정에서 문제가 발생한 경우, 바로 담당 멘토나 강사에서 얘기하세요! +- Replace `plugin:@typescript-eslint/recommended` to `plugin:@typescript-eslint/recommended-type-checked` or `plugin:@typescript-eslint/strict-type-checked` +- Optionally add `plugin:@typescript-eslint/stylistic-type-checked` +- Install [eslint-plugin-react](https://github.com/jsx-eslint/eslint-plugin-react) and add `plugin:react/recommended` & `plugin:react/jsx-runtime` to the `extends` list diff --git a/index.html b/index.html new file mode 100644 index 00000000..f6f9c5ff --- /dev/null +++ b/index.html @@ -0,0 +1,17 @@ + + + + + + + + 연차당직관리 + + +
+ + + diff --git a/package-lock.json b/package-lock.json new file mode 100644 index 00000000..9f28e75e --- /dev/null +++ b/package-lock.json @@ -0,0 +1,5919 @@ +{ + "name": "kdt5-mini", + "version": "0.0.0", + "lockfileVersion": 2, + "requires": true, + "packages": { + "": { + "name": "kdt5-mini", + "version": "0.0.0", + "dependencies": { + "@fullcalendar/core": "^6.1.8", + "@fullcalendar/daygrid": "^6.1.8", + "@fullcalendar/interaction": "^6.1.8", + "@fullcalendar/react": "^6.1.8", + "antd": "^5.7.2", + "axios": "^1.4.0", + "dayjs": "^1.11.9", + "react": "^18.2.0", + "react-dom": "^18.2.0", + "react-router-dom": "^6.14.2", + "recoil": "^0.7.7" + }, + "devDependencies": { + "@types/node": "^20.4.4", + "@types/react": "^18.2.15", + "@types/react-dom": "^18.2.7", + "@typescript-eslint/eslint-plugin": "^6.0.0", + "@typescript-eslint/parser": "^6.0.0", + "@vitejs/plugin-react-swc": "^3.3.2", + "dotenv": "^16.3.1", + "eslint": "^8.45.0", + "eslint-plugin-react-hooks": "^4.6.0", + "eslint-plugin-react-refresh": "^0.4.3", + "typescript": "^5.0.2", + "vite": "^4.4.5" + } + }, + "node_modules/@aashutoshrathi/word-wrap": { + "version": "1.2.6", + "resolved": "https://registry.npmjs.org/@aashutoshrathi/word-wrap/-/word-wrap-1.2.6.tgz", + "integrity": "sha512-1Yjs2SvM8TflER/OD3cOjhWWOZb58A2t7wpE2S9XfBYTiIl+XFhQG2bjy4Pu1I+EAlCNUzRDYDdFwFYUKvXcIA==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/@ant-design/colors": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/@ant-design/colors/-/colors-7.0.0.tgz", + "integrity": "sha512-iVm/9PfGCbC0dSMBrz7oiEXZaaGH7ceU40OJEfKmyuzR9R5CRimJYPlRiFtMQGQcbNMea/ePcoIebi4ASGYXtg==", + "dependencies": { + "@ctrl/tinycolor": "^3.4.0" + } + }, + "node_modules/@ant-design/cssinjs": { + "version": "1.13.2", + "resolved": "https://registry.npmjs.org/@ant-design/cssinjs/-/cssinjs-1.13.2.tgz", + "integrity": "sha512-II3QJx6V6boYkAIUiEd1/hquSeX1r67sUOtsvco35fTmV46JvkJz946/6K+ikiuODBSE1kLf7fJ5gHetQyUyww==", + "dependencies": { + "@babel/runtime": "^7.11.1", + "@emotion/hash": "^0.8.0", + "@emotion/unitless": "^0.7.5", + "classnames": "^2.3.1", + "csstype": "^3.0.10", + "rc-util": "^5.34.1", + "stylis": "^4.0.13" + }, + "peerDependencies": { + "react": ">=16.0.0", + "react-dom": ">=16.0.0" + } + }, + "node_modules/@ant-design/icons": { + "version": "5.1.4", + "resolved": "https://registry.npmjs.org/@ant-design/icons/-/icons-5.1.4.tgz", + "integrity": "sha512-YHKL7Jx3bM12OxvtiYDon04BsBT/6LGitYEqar3GljzWaAyMOAD8i/uF1Rsi5Us/YNdWWXBGSvZV2OZWMpJlcA==", + "dependencies": { + "@ant-design/colors": "^7.0.0", + "@ant-design/icons-svg": "^4.2.1", + "@babel/runtime": "^7.11.2", + "classnames": "^2.2.6", + "rc-util": "^5.31.1" + }, + "engines": { + "node": ">=8" + }, + "peerDependencies": { + "react": ">=16.0.0", + "react-dom": ">=16.0.0" + } + }, + "node_modules/@ant-design/icons-svg": { + "version": "4.2.1", + "resolved": "https://registry.npmjs.org/@ant-design/icons-svg/-/icons-svg-4.2.1.tgz", + "integrity": "sha512-EB0iwlKDGpG93hW8f85CTJTs4SvMX7tt5ceupvhALp1IF44SeUFOMhKUOYqpsoYWQKAOuTRDMqn75rEaKDp0Xw==" + }, + "node_modules/@ant-design/react-slick": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/@ant-design/react-slick/-/react-slick-1.0.1.tgz", + "integrity": "sha512-ARM0TmpGdDuUVE10NwUCENQlJSInNKo5NiBjL5szu5BxWNEHNwQMcDrlVCqFbkvFLy+2CvywW8Y59QJtC0YDag==", + "dependencies": { + "@babel/runtime": "^7.10.4", + "classnames": "^2.2.5", + "json2mq": "^0.2.0", + "resize-observer-polyfill": "^1.5.1", + "throttle-debounce": "^5.0.0" + }, + "peerDependencies": { + "react": ">=16.9.0" + } + }, + "node_modules/@babel/runtime": { + "version": "7.22.6", + "resolved": "https://registry.npmjs.org/@babel/runtime/-/runtime-7.22.6.tgz", + "integrity": "sha512-wDb5pWm4WDdF6LFUde3Jl8WzPA+3ZbxYqkC6xAXuD3irdEHN1k0NfTRrJD8ZD378SJ61miMLCqIOXYhd8x+AJQ==", + "dependencies": { + "regenerator-runtime": "^0.13.11" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@ctrl/tinycolor": { + "version": "3.6.0", + "resolved": "https://registry.npmjs.org/@ctrl/tinycolor/-/tinycolor-3.6.0.tgz", + "integrity": "sha512-/Z3l6pXthq0JvMYdUFyX9j0MaCltlIn6mfh9jLyQwg5aPKxkyNa0PTHtU1AlFXLNk55ZuAeJRcpvq+tmLfKmaQ==", + "engines": { + "node": ">=10" + } + }, + "node_modules/@emotion/hash": { + "version": "0.8.0", + "resolved": "https://registry.npmjs.org/@emotion/hash/-/hash-0.8.0.tgz", + "integrity": "sha512-kBJtf7PH6aWwZ6fka3zQ0p6SBYzx4fl1LoZXE2RrnYST9Xljm7WfKJrU4g/Xr3Beg72MLrp1AWNUmuYJTL7Cow==" + }, + "node_modules/@emotion/unitless": { + "version": "0.7.5", + "resolved": "https://registry.npmjs.org/@emotion/unitless/-/unitless-0.7.5.tgz", + "integrity": "sha512-OWORNpfjMsSSUBVrRBVGECkhWcULOAJz9ZW8uK9qgxD+87M7jHRcvh/A96XXNhXTLmKcoYSQtBEX7lHMO7YRwg==" + }, + "node_modules/@esbuild/android-arm": { + "version": "0.18.16", + "resolved": "https://registry.npmjs.org/@esbuild/android-arm/-/android-arm-0.18.16.tgz", + "integrity": "sha512-gCHjjQmA8L0soklKbLKA6pgsLk1byULuHe94lkZDzcO3/Ta+bbeewJioEn1Fr7kgy9NWNFy/C+MrBwC6I/WCug==", + "cpu": [ + "arm" + ], + "dev": true, + "optional": true, + "os": [ + "android" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/android-arm64": { + "version": "0.18.16", + "resolved": "https://registry.npmjs.org/@esbuild/android-arm64/-/android-arm64-0.18.16.tgz", + "integrity": "sha512-wsCqSPqLz+6Ov+OM4EthU43DyYVVyfn15S4j1bJzylDpc1r1jZFFfJQNfDuT8SlgwuqpmpJXK4uPlHGw6ve7eA==", + "cpu": [ + "arm64" + ], + "dev": true, + "optional": true, + "os": [ + "android" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/android-x64": { + "version": "0.18.16", + "resolved": "https://registry.npmjs.org/@esbuild/android-x64/-/android-x64-0.18.16.tgz", + "integrity": "sha512-ldsTXolyA3eTQ1//4DS+E15xl0H/3DTRJaRL0/0PgkqDsI0fV/FlOtD+h0u/AUJr+eOTlZv4aC9gvfppo3C4sw==", + "cpu": [ + "x64" + ], + "dev": true, + "optional": true, + "os": [ + "android" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/darwin-arm64": { + "version": "0.18.16", + "resolved": "https://registry.npmjs.org/@esbuild/darwin-arm64/-/darwin-arm64-0.18.16.tgz", + "integrity": "sha512-aBxruWCII+OtluORR/KvisEw0ALuw/qDQWvkoosA+c/ngC/Kwk0lLaZ+B++LLS481/VdydB2u6tYpWxUfnLAIw==", + "cpu": [ + "arm64" + ], + "dev": true, + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/darwin-x64": { + "version": "0.18.16", + "resolved": "https://registry.npmjs.org/@esbuild/darwin-x64/-/darwin-x64-0.18.16.tgz", + "integrity": "sha512-6w4Dbue280+rp3LnkgmriS1icOUZDyPuZo/9VsuMUTns7SYEiOaJ7Ca1cbhu9KVObAWfmdjUl4gwy9TIgiO5eA==", + "cpu": [ + "x64" + ], + "dev": true, + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/freebsd-arm64": { + "version": "0.18.16", + "resolved": "https://registry.npmjs.org/@esbuild/freebsd-arm64/-/freebsd-arm64-0.18.16.tgz", + "integrity": "sha512-x35fCebhe9s979DGKbVAwXUOcTmCIE32AIqB9CB1GralMIvxdnMLAw5CnID17ipEw9/3MvDsusj/cspYt2ZLNQ==", + "cpu": [ + "arm64" + ], + "dev": true, + "optional": true, + "os": [ + "freebsd" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/freebsd-x64": { + "version": "0.18.16", + "resolved": "https://registry.npmjs.org/@esbuild/freebsd-x64/-/freebsd-x64-0.18.16.tgz", + "integrity": "sha512-YM98f+PeNXF3GbxIJlUsj+McUWG1irguBHkszCIwfr3BXtXZsXo0vqybjUDFfu9a8Wr7uUD/YSmHib+EeGAFlg==", + "cpu": [ + "x64" + ], + "dev": true, + "optional": true, + "os": [ + "freebsd" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-arm": { + "version": "0.18.16", + "resolved": "https://registry.npmjs.org/@esbuild/linux-arm/-/linux-arm-0.18.16.tgz", + "integrity": "sha512-b5ABb+5Ha2C9JkeZXV+b+OruR1tJ33ePmv9ZwMeETSEKlmu/WJ45XTTG+l6a2KDsQtJJ66qo/hbSGBtk0XVLHw==", + "cpu": [ + "arm" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-arm64": { + "version": "0.18.16", + "resolved": "https://registry.npmjs.org/@esbuild/linux-arm64/-/linux-arm64-0.18.16.tgz", + "integrity": "sha512-XIqhNUxJiuy+zsR77+H5Z2f7s4YRlriSJKtvx99nJuG5ATuJPjmZ9n0ANgnGlPCpXGSReFpgcJ7O3SMtzIFeiQ==", + "cpu": [ + "arm64" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-ia32": { + "version": "0.18.16", + "resolved": "https://registry.npmjs.org/@esbuild/linux-ia32/-/linux-ia32-0.18.16.tgz", + "integrity": "sha512-no+pfEpwnRvIyH+txbBAWtjxPU9grslmTBfsmDndj7bnBmr55rOo/PfQmRfz7Qg9isswt1FP5hBbWb23fRWnow==", + "cpu": [ + "ia32" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-loong64": { + "version": "0.18.16", + "resolved": "https://registry.npmjs.org/@esbuild/linux-loong64/-/linux-loong64-0.18.16.tgz", + "integrity": "sha512-Zbnczs9ZXjmo0oZSS0zbNlJbcwKXa/fcNhYQjahDs4Xg18UumpXG/lwM2lcSvHS3mTrRyCYZvJbmzYc4laRI1g==", + "cpu": [ + "loong64" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-mips64el": { + "version": "0.18.16", + "resolved": "https://registry.npmjs.org/@esbuild/linux-mips64el/-/linux-mips64el-0.18.16.tgz", + "integrity": "sha512-YMF7hih1HVR/hQVa/ot4UVffc5ZlrzEb3k2ip0nZr1w6fnYypll9td2qcoMLvd3o8j3y6EbJM3MyIcXIVzXvQQ==", + "cpu": [ + "mips64el" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-ppc64": { + "version": "0.18.16", + "resolved": "https://registry.npmjs.org/@esbuild/linux-ppc64/-/linux-ppc64-0.18.16.tgz", + "integrity": "sha512-Wkz++LZ29lDwUyTSEnzDaaP5OveOgTU69q9IyIw9WqLRxM4BjTBjz9un4G6TOvehWpf/J3gYVFN96TjGHrbcNQ==", + "cpu": [ + "ppc64" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-riscv64": { + "version": "0.18.16", + "resolved": "https://registry.npmjs.org/@esbuild/linux-riscv64/-/linux-riscv64-0.18.16.tgz", + "integrity": "sha512-LFMKZ30tk78/mUv1ygvIP+568bwf4oN6reG/uczXnz6SvFn4e2QUFpUpZY9iSJT6Qpgstrhef/nMykIXZtZWGQ==", + "cpu": [ + "riscv64" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-s390x": { + "version": "0.18.16", + "resolved": "https://registry.npmjs.org/@esbuild/linux-s390x/-/linux-s390x-0.18.16.tgz", + "integrity": "sha512-3ZC0BgyYHYKfZo3AV2/66TD/I9tlSBaW7eWTEIkrQQKfJIifKMMttXl9FrAg+UT0SGYsCRLI35Gwdmm96vlOjg==", + "cpu": [ + "s390x" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-x64": { + "version": "0.18.16", + "resolved": "https://registry.npmjs.org/@esbuild/linux-x64/-/linux-x64-0.18.16.tgz", + "integrity": "sha512-xu86B3647DihHJHv/wx3NCz2Dg1gjQ8bbf9cVYZzWKY+gsvxYmn/lnVlqDRazObc3UMwoHpUhNYaZset4X8IPA==", + "cpu": [ + "x64" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/netbsd-x64": { + "version": "0.18.16", + "resolved": "https://registry.npmjs.org/@esbuild/netbsd-x64/-/netbsd-x64-0.18.16.tgz", + "integrity": "sha512-uVAgpimx9Ffw3xowtg/7qQPwHFx94yCje+DoBx+LNm2ePDpQXHrzE+Sb0Si2VBObYz+LcRps15cq+95YM7gkUw==", + "cpu": [ + "x64" + ], + "dev": true, + "optional": true, + "os": [ + "netbsd" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/openbsd-x64": { + "version": "0.18.16", + "resolved": "https://registry.npmjs.org/@esbuild/openbsd-x64/-/openbsd-x64-0.18.16.tgz", + "integrity": "sha512-6OjCQM9wf7z8/MBi6BOWaTL2AS/SZudsZtBziXMtNI8r/U41AxS9x7jn0ATOwVy08OotwkPqGRMkpPR2wcTJXA==", + "cpu": [ + "x64" + ], + "dev": true, + "optional": true, + "os": [ + "openbsd" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/sunos-x64": { + "version": "0.18.16", + "resolved": "https://registry.npmjs.org/@esbuild/sunos-x64/-/sunos-x64-0.18.16.tgz", + "integrity": "sha512-ZoNkruFYJp9d1LbUYCh8awgQDvB9uOMZqlQ+gGEZR7v6C+N6u7vPr86c+Chih8niBR81Q/bHOSKGBK3brJyvkQ==", + "cpu": [ + "x64" + ], + "dev": true, + "optional": true, + "os": [ + "sunos" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/win32-arm64": { + "version": "0.18.16", + "resolved": "https://registry.npmjs.org/@esbuild/win32-arm64/-/win32-arm64-0.18.16.tgz", + "integrity": "sha512-+j4anzQ9hrs+iqO+/wa8UE6TVkKua1pXUb0XWFOx0FiAj6R9INJ+WE//1/Xo6FG1vB5EpH3ko+XcgwiDXTxcdw==", + "cpu": [ + "arm64" + ], + "dev": true, + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/win32-ia32": { + "version": "0.18.16", + "resolved": "https://registry.npmjs.org/@esbuild/win32-ia32/-/win32-ia32-0.18.16.tgz", + "integrity": "sha512-5PFPmq3sSKTp9cT9dzvI67WNfRZGvEVctcZa1KGjDDu4n3H8k59Inbk0du1fz0KrAbKKNpJbdFXQMDUz7BG4rQ==", + "cpu": [ + "ia32" + ], + "dev": true, + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/win32-x64": { + "version": "0.18.16", + "resolved": "https://registry.npmjs.org/@esbuild/win32-x64/-/win32-x64-0.18.16.tgz", + "integrity": "sha512-sCIVrrtcWN5Ua7jYXNG1xD199IalrbfV2+0k/2Zf2OyV2FtnQnMgdzgpRAbi4AWlKJj1jkX+M+fEGPQj6BQB4w==", + "cpu": [ + "x64" + ], + "dev": true, + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@eslint-community/eslint-utils": { + "version": "4.4.0", + "resolved": "https://registry.npmjs.org/@eslint-community/eslint-utils/-/eslint-utils-4.4.0.tgz", + "integrity": "sha512-1/sA4dwrzBAyeUoQ6oxahHKmrZvsnLCg4RfxW3ZFGGmQkSNQPFNLV9CUEFQP1x9EYXHTo5p6xdhZM1Ne9p/AfA==", + "dev": true, + "dependencies": { + "eslint-visitor-keys": "^3.3.0" + }, + "engines": { + "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + }, + "peerDependencies": { + "eslint": "^6.0.0 || ^7.0.0 || >=8.0.0" + } + }, + "node_modules/@eslint-community/regexpp": { + "version": "4.6.0", + "resolved": "https://registry.npmjs.org/@eslint-community/regexpp/-/regexpp-4.6.0.tgz", + "integrity": "sha512-uiPeRISaglZnaZk8vwrjQZ1CxogZeY/4IYft6gBOTqu1WhVXWmCmZMWxUv2Q/pxSvPdp1JPaO62kLOcOkMqWrw==", + "dev": true, + "engines": { + "node": "^12.0.0 || ^14.0.0 || >=16.0.0" + } + }, + "node_modules/@eslint/eslintrc": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/@eslint/eslintrc/-/eslintrc-2.1.0.tgz", + "integrity": "sha512-Lj7DECXqIVCqnqjjHMPna4vn6GJcMgul/wuS0je9OZ9gsL0zzDpKPVtcG1HaDVc+9y+qgXneTeUMbCqXJNpH1A==", + "dev": true, + "dependencies": { + "ajv": "^6.12.4", + "debug": "^4.3.2", + "espree": "^9.6.0", + "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" + }, + "engines": { + "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + }, + "funding": { + "url": "https://opencollective.com/eslint" + } + }, + "node_modules/@eslint/js": { + "version": "8.44.0", + "resolved": "https://registry.npmjs.org/@eslint/js/-/js-8.44.0.tgz", + "integrity": "sha512-Ag+9YM4ocKQx9AarydN0KY2j0ErMHNIocPDrVo8zAE44xLTjEtz81OdR68/cydGtk6m6jDb5Za3r2useMzYmSw==", + "dev": true, + "engines": { + "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + } + }, + "node_modules/@fullcalendar/core": { + "version": "6.1.8", + "resolved": "https://registry.npmjs.org/@fullcalendar/core/-/core-6.1.8.tgz", + "integrity": "sha512-i8JBIvZCWGO9dsMEDcx9bnsQZ9PtGSJdOXGgWbhLaGq2iq41OBdp9g9gM4b/Otv2oK8bL5Gl6CsMmb/HkDtA6Q==", + "dependencies": { + "preact": "~10.12.1" + } + }, + "node_modules/@fullcalendar/daygrid": { + "version": "6.1.8", + "resolved": "https://registry.npmjs.org/@fullcalendar/daygrid/-/daygrid-6.1.8.tgz", + "integrity": "sha512-kCZxQFKb9Vqa3CZRX0v7rMSJ2mlTt4gDpyLfiNJKxUAq7W51uKurPaFZWicaXy1ESHVBxKNlbx5uNjBpyu50JQ==", + "peerDependencies": { + "@fullcalendar/core": "~6.1.8" + } + }, + "node_modules/@fullcalendar/interaction": { + "version": "6.1.8", + "resolved": "https://registry.npmjs.org/@fullcalendar/interaction/-/interaction-6.1.8.tgz", + "integrity": "sha512-r6W4E9ohaA87M2uPSlmpE2WT7Fzu7LN0u2pE6D/tThruCEaAPbN8Pw5+sqclsuyTIL09mg0eSJm/ggJekTabSA==", + "peerDependencies": { + "@fullcalendar/core": "~6.1.8" + } + }, + "node_modules/@fullcalendar/react": { + "version": "6.1.8", + "resolved": "https://registry.npmjs.org/@fullcalendar/react/-/react-6.1.8.tgz", + "integrity": "sha512-E8GQSQyZHkjpwxQW5Vci7iZgN7f33ntuRcvfGii4Fn35t9VHGz2SEyKAWXpVf38elcKTZKVgajU9ipStd+1LEg==", + "peerDependencies": { + "@fullcalendar/core": "~6.1.8", + "react": "^16.7.0 || ^17 || ^18", + "react-dom": "^16.7.0 || ^17 || ^18" + } + }, + "node_modules/@humanwhocodes/config-array": { + "version": "0.11.10", + "resolved": "https://registry.npmjs.org/@humanwhocodes/config-array/-/config-array-0.11.10.tgz", + "integrity": "sha512-KVVjQmNUepDVGXNuoRRdmmEjruj0KfiGSbS8LVc12LMsWDQzRXJ0qdhN8L8uUigKpfEHRhlaQFY0ib1tnUbNeQ==", + "dev": true, + "dependencies": { + "@humanwhocodes/object-schema": "^1.2.1", + "debug": "^4.1.1", + "minimatch": "^3.0.5" + }, + "engines": { + "node": ">=10.10.0" + } + }, + "node_modules/@humanwhocodes/module-importer": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/@humanwhocodes/module-importer/-/module-importer-1.0.1.tgz", + "integrity": "sha512-bxveV4V8v5Yb4ncFTT3rPSgZBOpCkjfK0y4oVVVJwIuDVBRMDXrPyXRL988i5ap9m9bnyEEjWfm5WkBmtffLfA==", + "dev": true, + "engines": { + "node": ">=12.22" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/nzakas" + } + }, + "node_modules/@humanwhocodes/object-schema": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/@humanwhocodes/object-schema/-/object-schema-1.2.1.tgz", + "integrity": "sha512-ZnQMnLV4e7hDlUvw8H+U8ASL02SS2Gn6+9Ac3wGGLIe7+je2AeAOxPY+izIPJDfFDb7eDjev0Us8MO1iFRN8hA==", + "dev": true + }, + "node_modules/@nodelib/fs.scandir": { + "version": "2.1.5", + "resolved": "https://registry.npmjs.org/@nodelib/fs.scandir/-/fs.scandir-2.1.5.tgz", + "integrity": "sha512-vq24Bq3ym5HEQm2NKCr3yXDwjc7vTsEThRDnkp2DK9p1uqLR+DHurm/NOTo0KG7HYHU7eppKZj3MyqYuMBf62g==", + "dev": true, + "dependencies": { + "@nodelib/fs.stat": "2.0.5", + "run-parallel": "^1.1.9" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/@nodelib/fs.stat": { + "version": "2.0.5", + "resolved": "https://registry.npmjs.org/@nodelib/fs.stat/-/fs.stat-2.0.5.tgz", + "integrity": "sha512-RkhPPp2zrqDAQA/2jNhnztcPAlv64XdhIp7a7454A5ovI7Bukxgt7MX7udwAu3zg1DcpPU0rz3VV1SeaqvY4+A==", + "dev": true, + "engines": { + "node": ">= 8" + } + }, + "node_modules/@nodelib/fs.walk": { + "version": "1.2.8", + "resolved": "https://registry.npmjs.org/@nodelib/fs.walk/-/fs.walk-1.2.8.tgz", + "integrity": "sha512-oGB+UxlgWcgQkgwo8GcEGwemoTFt3FIO9ababBmaGwXIoBKZ+GTy0pP185beGg7Llih/NSHSV2XAs1lnznocSg==", + "dev": true, + "dependencies": { + "@nodelib/fs.scandir": "2.1.5", + "fastq": "^1.6.0" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/@rc-component/color-picker": { + "version": "1.4.1", + "resolved": "https://registry.npmjs.org/@rc-component/color-picker/-/color-picker-1.4.1.tgz", + "integrity": "sha512-vh5EWqnsayZa/JwUznqDaPJz39jznx/YDbyBuVJntv735tKXKwEUZZb2jYEldOg+NKWZwtALjGMrNeGBmqFoEw==", + "dependencies": { + "@babel/runtime": "^7.10.1", + "@ctrl/tinycolor": "^3.6.0", + "classnames": "^2.2.6", + "rc-util": "^5.30.0" + }, + "peerDependencies": { + "react": ">=16.9.0", + "react-dom": ">=16.9.0" + } + }, + "node_modules/@rc-component/context": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/@rc-component/context/-/context-1.3.0.tgz", + "integrity": "sha512-6QdaCJ7Wn5UZLJs15IEfqy4Ru3OaL5ctqpQYWd5rlfV9wwzrzdt6+kgAQZV/qdB0MUPN4nhyBfRembQCIvBf+w==", + "dependencies": { + "@babel/runtime": "^7.10.1", + "rc-util": "^5.27.0" + }, + "peerDependencies": { + "react": ">=16.9.0", + "react-dom": ">=16.9.0" + } + }, + "node_modules/@rc-component/mini-decimal": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/@rc-component/mini-decimal/-/mini-decimal-1.1.0.tgz", + "integrity": "sha512-jS4E7T9Li2GuYwI6PyiVXmxTiM6b07rlD9Ge8uGZSCz3WlzcG5ZK7g5bbuKNeZ9pgUuPK/5guV781ujdVpm4HQ==", + "dependencies": { + "@babel/runtime": "^7.18.0" + }, + "engines": { + "node": ">=8.x" + } + }, + "node_modules/@rc-component/mutate-observer": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/@rc-component/mutate-observer/-/mutate-observer-1.0.0.tgz", + "integrity": "sha512-okqRJSfNisXdI6CUeOLZC5ukBW/8kir2Ii4PJiKpUt+3+uS7dxwJUMxsUZquxA1rQuL8YcEmKVp/TCnR+yUdZA==", + "dependencies": { + "@babel/runtime": "^7.18.0", + "classnames": "^2.3.2", + "rc-util": "^5.24.4" + }, + "engines": { + "node": ">=8.x" + }, + "peerDependencies": { + "react": ">=16.9.0", + "react-dom": ">=16.9.0" + } + }, + "node_modules/@rc-component/portal": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/@rc-component/portal/-/portal-1.1.2.tgz", + "integrity": "sha512-6f813C0IsasTZms08kfA8kPAGxbbkYToa8ALaiDIGGECU4i9hj8Plgbx0sNJDrey3EtHO30hmdaxtT0138xZcg==", + "dependencies": { + "@babel/runtime": "^7.18.0", + "classnames": "^2.3.2", + "rc-util": "^5.24.4" + }, + "engines": { + "node": ">=8.x" + }, + "peerDependencies": { + "react": ">=16.9.0", + "react-dom": ">=16.9.0" + } + }, + "node_modules/@rc-component/tour": { + "version": "1.8.1", + "resolved": "https://registry.npmjs.org/@rc-component/tour/-/tour-1.8.1.tgz", + "integrity": "sha512-CsrQnfKgNArxx2j1RNHVLZgVA+rLrEj06lIsl4KSynMqADsqz8eKvVkr0F3p9PA10948M6WEEZt5a/FGAbGR2A==", + "dependencies": { + "@babel/runtime": "^7.18.0", + "@rc-component/portal": "^1.0.0-9", + "@rc-component/trigger": "^1.3.6", + "classnames": "^2.3.2", + "rc-util": "^5.24.4" + }, + "engines": { + "node": ">=8.x" + }, + "peerDependencies": { + "react": ">=16.9.0", + "react-dom": ">=16.9.0" + } + }, + "node_modules/@rc-component/trigger": { + "version": "1.14.3", + "resolved": "https://registry.npmjs.org/@rc-component/trigger/-/trigger-1.14.3.tgz", + "integrity": "sha512-kIPL8srsAb3RWGWicLZyTMS/6IM1bmjcaqXS3GfgIe6EoPBxzGRx6NAvpDJg198zDSZ/RXGFLrqguCItnaN/Aw==", + "dependencies": { + "@babel/runtime": "^7.18.3", + "@rc-component/portal": "^1.1.0", + "classnames": "^2.3.2", + "rc-align": "^4.0.0", + "rc-motion": "^2.0.0", + "rc-resize-observer": "^1.3.1", + "rc-util": "^5.33.0" + }, + "engines": { + "node": ">=8.x" + }, + "peerDependencies": { + "react": ">=16.9.0", + "react-dom": ">=16.9.0" + } + }, + "node_modules/@remix-run/router": { + "version": "1.7.2", + "resolved": "https://registry.npmjs.org/@remix-run/router/-/router-1.7.2.tgz", + "integrity": "sha512-7Lcn7IqGMV+vizMPoEl5F0XDshcdDYtMI6uJLQdQz5CfZAwy3vvGKYSUk789qndt5dEC4HfSjviSYlSoHGL2+A==", + "engines": { + "node": ">=14" + } + }, + "node_modules/@swc/core": { + "version": "1.3.70", + "resolved": "https://registry.npmjs.org/@swc/core/-/core-1.3.70.tgz", + "integrity": "sha512-LWVWlEDLlOD25PvA2NEz41UzdwXnlDyBiZbe69s3zM0DfCPwZXLUm79uSqH9ItsOjTrXSL5/1+XUL6C/BZwChA==", + "dev": true, + "hasInstallScript": true, + "engines": { + "node": ">=10" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/swc" + }, + "optionalDependencies": { + "@swc/core-darwin-arm64": "1.3.70", + "@swc/core-darwin-x64": "1.3.70", + "@swc/core-linux-arm-gnueabihf": "1.3.70", + "@swc/core-linux-arm64-gnu": "1.3.70", + "@swc/core-linux-arm64-musl": "1.3.70", + "@swc/core-linux-x64-gnu": "1.3.70", + "@swc/core-linux-x64-musl": "1.3.70", + "@swc/core-win32-arm64-msvc": "1.3.70", + "@swc/core-win32-ia32-msvc": "1.3.70", + "@swc/core-win32-x64-msvc": "1.3.70" + }, + "peerDependencies": { + "@swc/helpers": "^0.5.0" + }, + "peerDependenciesMeta": { + "@swc/helpers": { + "optional": true + } + } + }, + "node_modules/@swc/core-darwin-arm64": { + "version": "1.3.70", + "resolved": "https://registry.npmjs.org/@swc/core-darwin-arm64/-/core-darwin-arm64-1.3.70.tgz", + "integrity": "sha512-31+mcl0dgdRHvZRjhLOK9V6B+qJ7nxDZYINr9pBlqGWxknz37Vld5KK19Kpr79r0dXUZvaaelLjCnJk9dA2PcQ==", + "cpu": [ + "arm64" + ], + "dev": true, + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": ">=10" + } + }, + "node_modules/@swc/core-darwin-x64": { + "version": "1.3.70", + "resolved": "https://registry.npmjs.org/@swc/core-darwin-x64/-/core-darwin-x64-1.3.70.tgz", + "integrity": "sha512-GMFJ65E18zQC80t0os+TZvI+8lbRuitncWVge/RXmXbVLPRcdykP4EJ87cqzcG5Ah0z18/E0T+ixD6jHRisrYQ==", + "cpu": [ + "x64" + ], + "dev": true, + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": ">=10" + } + }, + "node_modules/@swc/core-linux-arm-gnueabihf": { + "version": "1.3.70", + "resolved": "https://registry.npmjs.org/@swc/core-linux-arm-gnueabihf/-/core-linux-arm-gnueabihf-1.3.70.tgz", + "integrity": "sha512-wjhCwS8LCiAq2VedF1b4Bryyw68xZnfMED4pLRazAl8BaUlDFANfRBORNunxlfHQj4V3x39IaiLgCZRHMdzXBg==", + "cpu": [ + "arm" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=10" + } + }, + "node_modules/@swc/core-linux-arm64-gnu": { + "version": "1.3.70", + "resolved": "https://registry.npmjs.org/@swc/core-linux-arm64-gnu/-/core-linux-arm64-gnu-1.3.70.tgz", + "integrity": "sha512-9D/Rx67cAOnMiexvCqARxvhj7coRajTp5HlJHuf+rfwMqI2hLhpO9/pBMQxBUAWxODO/ksQ/OF+GJRjmtWw/2A==", + "cpu": [ + "arm64" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=10" + } + }, + "node_modules/@swc/core-linux-arm64-musl": { + "version": "1.3.70", + "resolved": "https://registry.npmjs.org/@swc/core-linux-arm64-musl/-/core-linux-arm64-musl-1.3.70.tgz", + "integrity": "sha512-gkjxBio7XD+1GlQVVyPP/qeFkLu83VhRHXaUrkNYpr5UZG9zZurBERT9nkS6Y+ouYh+Q9xmw57aIyd2KvD2zqQ==", + "cpu": [ + "arm64" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=10" + } + }, + "node_modules/@swc/core-linux-x64-gnu": { + "version": "1.3.70", + "resolved": "https://registry.npmjs.org/@swc/core-linux-x64-gnu/-/core-linux-x64-gnu-1.3.70.tgz", + "integrity": "sha512-/nCly+V4xfMVwfEUoLLAukxUSot/RcSzsf6GdsGTjFcrp5sZIntAjokYRytm3VT1c2TK321AfBorsi9R5w8Y7Q==", + "cpu": [ + "x64" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=10" + } + }, + "node_modules/@swc/core-linux-x64-musl": { + "version": "1.3.70", + "resolved": "https://registry.npmjs.org/@swc/core-linux-x64-musl/-/core-linux-x64-musl-1.3.70.tgz", + "integrity": "sha512-HoOsPJbt361KGKaivAK0qIiYARkhzlxeAfvF5NlnKxkIMOZpQ46Lwj3tR0VWohKbrhS+cYKFlVuDi5XnDkx0XA==", + "cpu": [ + "x64" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=10" + } + }, + "node_modules/@swc/core-win32-arm64-msvc": { + "version": "1.3.70", + "resolved": "https://registry.npmjs.org/@swc/core-win32-arm64-msvc/-/core-win32-arm64-msvc-1.3.70.tgz", + "integrity": "sha512-hm4IBK/IaRil+aj1cWU6f0GyAdHpw/Jr5nyFYLM2c/tt7w2t5hgb8NjzM2iM84lOClrig1fG6edj2vCF1dFzNQ==", + "cpu": [ + "arm64" + ], + "dev": true, + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">=10" + } + }, + "node_modules/@swc/core-win32-ia32-msvc": { + "version": "1.3.70", + "resolved": "https://registry.npmjs.org/@swc/core-win32-ia32-msvc/-/core-win32-ia32-msvc-1.3.70.tgz", + "integrity": "sha512-5cgKUKIT/9Fp5fCA+zIjYCQ4dSvjFYOeWGZR3QiTXGkC4bGa1Ji9SEPyeIAX0iruUnKjYaZB9RvHK2tNn7RLrQ==", + "cpu": [ + "ia32" + ], + "dev": true, + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">=10" + } + }, + "node_modules/@swc/core-win32-x64-msvc": { + "version": "1.3.70", + "resolved": "https://registry.npmjs.org/@swc/core-win32-x64-msvc/-/core-win32-x64-msvc-1.3.70.tgz", + "integrity": "sha512-LE8lW46+TQBzVkn2mHBlk8DIElPIZ2dO5P8AbJiARNBAnlqQWu67l9gWM89UiZ2l33J2cI37pHzON3tKnT8f9g==", + "cpu": [ + "x64" + ], + "dev": true, + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">=10" + } + }, + "node_modules/@types/json-schema": { + "version": "7.0.12", + "resolved": "https://registry.npmjs.org/@types/json-schema/-/json-schema-7.0.12.tgz", + "integrity": "sha512-Hr5Jfhc9eYOQNPYO5WLDq/n4jqijdHNlDXjuAQkkt+mWdQR+XJToOHrsD4cPaMXpn6KO7y2+wM8AZEs8VpBLVA==", + "dev": true + }, + "node_modules/@types/node": { + "version": "20.4.4", + "resolved": "https://registry.npmjs.org/@types/node/-/node-20.4.4.tgz", + "integrity": "sha512-CukZhumInROvLq3+b5gLev+vgpsIqC2D0deQr/yS1WnxvmYLlJXZpaQrQiseMY+6xusl79E04UjWoqyr+t1/Ew==", + "dev": true + }, + "node_modules/@types/prop-types": { + "version": "15.7.5", + "resolved": "https://registry.npmjs.org/@types/prop-types/-/prop-types-15.7.5.tgz", + "integrity": "sha512-JCB8C6SnDoQf0cNycqd/35A7MjcnK+ZTqE7judS6o7utxUCg6imJg3QK2qzHKszlTjcj2cn+NwMB2i96ubpj7w==", + "dev": true + }, + "node_modules/@types/react": { + "version": "18.2.15", + "resolved": "https://registry.npmjs.org/@types/react/-/react-18.2.15.tgz", + "integrity": "sha512-oEjE7TQt1fFTFSbf8kkNuc798ahTUzn3Le67/PWjE8MAfYAD/qB7O8hSTcromLFqHCt9bcdOg5GXMokzTjJ5SA==", + "dev": true, + "dependencies": { + "@types/prop-types": "*", + "@types/scheduler": "*", + "csstype": "^3.0.2" + } + }, + "node_modules/@types/react-dom": { + "version": "18.2.7", + "resolved": "https://registry.npmjs.org/@types/react-dom/-/react-dom-18.2.7.tgz", + "integrity": "sha512-GRaAEriuT4zp9N4p1i8BDBYmEyfo+xQ3yHjJU4eiK5NDa1RmUZG+unZABUTK4/Ox/M+GaHwb6Ow8rUITrtjszA==", + "dev": true, + "dependencies": { + "@types/react": "*" + } + }, + "node_modules/@types/scheduler": { + "version": "0.16.3", + "resolved": "https://registry.npmjs.org/@types/scheduler/-/scheduler-0.16.3.tgz", + "integrity": "sha512-5cJ8CB4yAx7BH1oMvdU0Jh9lrEXyPkar6F9G/ERswkCuvP4KQZfZkSjcMbAICCpQTN4OuZn8tz0HiKv9TGZgrQ==", + "dev": true + }, + "node_modules/@types/semver": { + "version": "7.5.0", + "resolved": "https://registry.npmjs.org/@types/semver/-/semver-7.5.0.tgz", + "integrity": "sha512-G8hZ6XJiHnuhQKR7ZmysCeJWE08o8T0AXtk5darsCaTVsYZhhgUrq53jizaR2FvsoeCwJhlmwTjkXBY5Pn/ZHw==", + "dev": true + }, + "node_modules/@typescript-eslint/eslint-plugin": { + "version": "6.1.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/eslint-plugin/-/eslint-plugin-6.1.0.tgz", + "integrity": "sha512-qg7Bm5TyP/I7iilGyp6DRqqkt8na00lI6HbjWZObgk3FFSzH5ypRwAHXJhJkwiRtTcfn+xYQIMOR5kJgpo6upw==", + "dev": true, + "dependencies": { + "@eslint-community/regexpp": "^4.5.1", + "@typescript-eslint/scope-manager": "6.1.0", + "@typescript-eslint/type-utils": "6.1.0", + "@typescript-eslint/utils": "6.1.0", + "@typescript-eslint/visitor-keys": "6.1.0", + "debug": "^4.3.4", + "graphemer": "^1.4.0", + "ignore": "^5.2.4", + "natural-compare": "^1.4.0", + "natural-compare-lite": "^1.4.0", + "semver": "^7.5.4", + "ts-api-utils": "^1.0.1" + }, + "engines": { + "node": "^16.0.0 || >=18.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependencies": { + "@typescript-eslint/parser": "^6.0.0 || ^6.0.0-alpha", + "eslint": "^7.0.0 || ^8.0.0" + }, + "peerDependenciesMeta": { + "typescript": { + "optional": true + } + } + }, + "node_modules/@typescript-eslint/parser": { + "version": "6.1.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/parser/-/parser-6.1.0.tgz", + "integrity": "sha512-hIzCPvX4vDs4qL07SYzyomamcs2/tQYXg5DtdAfj35AyJ5PIUqhsLf4YrEIFzZcND7R2E8tpQIZKayxg8/6Wbw==", + "dev": true, + "dependencies": { + "@typescript-eslint/scope-manager": "6.1.0", + "@typescript-eslint/types": "6.1.0", + "@typescript-eslint/typescript-estree": "6.1.0", + "@typescript-eslint/visitor-keys": "6.1.0", + "debug": "^4.3.4" + }, + "engines": { + "node": "^16.0.0 || >=18.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependencies": { + "eslint": "^7.0.0 || ^8.0.0" + }, + "peerDependenciesMeta": { + "typescript": { + "optional": true + } + } + }, + "node_modules/@typescript-eslint/scope-manager": { + "version": "6.1.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/scope-manager/-/scope-manager-6.1.0.tgz", + "integrity": "sha512-AxjgxDn27hgPpe2rQe19k0tXw84YCOsjDJ2r61cIebq1t+AIxbgiXKvD4999Wk49GVaAcdJ/d49FYel+Pp3jjw==", + "dev": true, + "dependencies": { + "@typescript-eslint/types": "6.1.0", + "@typescript-eslint/visitor-keys": "6.1.0" + }, + "engines": { + "node": "^16.0.0 || >=18.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + } + }, + "node_modules/@typescript-eslint/type-utils": { + "version": "6.1.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/type-utils/-/type-utils-6.1.0.tgz", + "integrity": "sha512-kFXBx6QWS1ZZ5Ni89TyT1X9Ag6RXVIVhqDs0vZE/jUeWlBv/ixq2diua6G7ece6+fXw3TvNRxP77/5mOMusx2w==", + "dev": true, + "dependencies": { + "@typescript-eslint/typescript-estree": "6.1.0", + "@typescript-eslint/utils": "6.1.0", + "debug": "^4.3.4", + "ts-api-utils": "^1.0.1" + }, + "engines": { + "node": "^16.0.0 || >=18.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependencies": { + "eslint": "^7.0.0 || ^8.0.0" + }, + "peerDependenciesMeta": { + "typescript": { + "optional": true + } + } + }, + "node_modules/@typescript-eslint/types": { + "version": "6.1.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/types/-/types-6.1.0.tgz", + "integrity": "sha512-+Gfd5NHCpDoHDOaU/yIF3WWRI2PcBRKKpP91ZcVbL0t5tQpqYWBs3z/GGhvU+EV1D0262g9XCnyqQh19prU0JQ==", + "dev": true, + "engines": { + "node": "^16.0.0 || >=18.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + } + }, + "node_modules/@typescript-eslint/typescript-estree": { + "version": "6.1.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/typescript-estree/-/typescript-estree-6.1.0.tgz", + "integrity": "sha512-nUKAPWOaP/tQjU1IQw9sOPCDavs/iU5iYLiY/6u7gxS7oKQoi4aUxXS1nrrVGTyBBaGesjkcwwHkbkiD5eBvcg==", + "dev": true, + "dependencies": { + "@typescript-eslint/types": "6.1.0", + "@typescript-eslint/visitor-keys": "6.1.0", + "debug": "^4.3.4", + "globby": "^11.1.0", + "is-glob": "^4.0.3", + "semver": "^7.5.4", + "ts-api-utils": "^1.0.1" + }, + "engines": { + "node": "^16.0.0 || >=18.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependenciesMeta": { + "typescript": { + "optional": true + } + } + }, + "node_modules/@typescript-eslint/utils": { + "version": "6.1.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/utils/-/utils-6.1.0.tgz", + "integrity": "sha512-wp652EogZlKmQoMS5hAvWqRKplXvkuOnNzZSE0PVvsKjpexd/XznRVHAtrfHFYmqaJz0DFkjlDsGYC9OXw+OhQ==", + "dev": true, + "dependencies": { + "@eslint-community/eslint-utils": "^4.4.0", + "@types/json-schema": "^7.0.12", + "@types/semver": "^7.5.0", + "@typescript-eslint/scope-manager": "6.1.0", + "@typescript-eslint/types": "6.1.0", + "@typescript-eslint/typescript-estree": "6.1.0", + "semver": "^7.5.4" + }, + "engines": { + "node": "^16.0.0 || >=18.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependencies": { + "eslint": "^7.0.0 || ^8.0.0" + } + }, + "node_modules/@typescript-eslint/visitor-keys": { + "version": "6.1.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/visitor-keys/-/visitor-keys-6.1.0.tgz", + "integrity": "sha512-yQeh+EXhquh119Eis4k0kYhj9vmFzNpbhM3LftWQVwqVjipCkwHBQOZutcYW+JVkjtTG9k8nrZU1UoNedPDd1A==", + "dev": true, + "dependencies": { + "@typescript-eslint/types": "6.1.0", + "eslint-visitor-keys": "^3.4.1" + }, + "engines": { + "node": "^16.0.0 || >=18.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + } + }, + "node_modules/@vitejs/plugin-react-swc": { + "version": "3.3.2", + "resolved": "https://registry.npmjs.org/@vitejs/plugin-react-swc/-/plugin-react-swc-3.3.2.tgz", + "integrity": "sha512-VJFWY5sfoZerQRvJrh518h3AcQt6f/yTuWn4/TRB+dqmYU0NX1qz7qM5Wfd+gOQqUzQW4gxKqKN3KpE/P3+zrA==", + "dev": true, + "dependencies": { + "@swc/core": "^1.3.61" + }, + "peerDependencies": { + "vite": "^4" + } + }, + "node_modules/acorn": { + "version": "8.10.0", + "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.10.0.tgz", + "integrity": "sha512-F0SAmZ8iUtS//m8DmCTA0jlh6TDKkHQyK6xc6V4KDTyZKA9dnvX9/3sRTVQrWm79glUAZbnmmNcdYwUIHWVybw==", + "dev": true, + "bin": { + "acorn": "bin/acorn" + }, + "engines": { + "node": ">=0.4.0" + } + }, + "node_modules/acorn-jsx": { + "version": "5.3.2", + "resolved": "https://registry.npmjs.org/acorn-jsx/-/acorn-jsx-5.3.2.tgz", + "integrity": "sha512-rq9s+JNhf0IChjtDXxllJ7g41oZk5SlXtp0LHwyA5cejwn7vKmKp4pPri6YEePv2PU65sAsegbXtIinmDFDXgQ==", + "dev": true, + "peerDependencies": { + "acorn": "^6.0.0 || ^7.0.0 || ^8.0.0" + } + }, + "node_modules/ajv": { + "version": "6.12.6", + "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.12.6.tgz", + "integrity": "sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g==", + "dev": true, + "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" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/epoberezkin" + } + }, + "node_modules/ansi-regex": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz", + "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "dependencies": { + "color-convert": "^2.0.1" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/antd": { + "version": "5.7.3", + "resolved": "https://registry.npmjs.org/antd/-/antd-5.7.3.tgz", + "integrity": "sha512-7sQeE86XkUrYDIKGu/Qu7kl+NWYzkVSGbGqWGIbITHkFZorCyOvvqgF63fiWo/tp2lZWbEOO0Cm7IiYnoeWh9A==", + "dependencies": { + "@ant-design/colors": "^7.0.0", + "@ant-design/cssinjs": "^1.13.2", + "@ant-design/icons": "^5.1.0", + "@ant-design/react-slick": "~1.0.0", + "@babel/runtime": "^7.18.3", + "@ctrl/tinycolor": "^3.6.0", + "@rc-component/color-picker": "~1.4.0", + "@rc-component/mutate-observer": "^1.0.0", + "@rc-component/tour": "~1.8.1", + "@rc-component/trigger": "^1.13.0", + "classnames": "^2.2.6", + "copy-to-clipboard": "^3.2.0", + "dayjs": "^1.11.1", + "qrcode.react": "^3.1.0", + "rc-cascader": "~3.12.0", + "rc-checkbox": "~3.1.0", + "rc-collapse": "~3.7.0", + "rc-dialog": "~9.1.0", + "rc-drawer": "~6.2.0", + "rc-dropdown": "~4.1.0", + "rc-field-form": "~1.34.0", + "rc-image": "~7.0.0", + "rc-input": "~1.1.0", + "rc-input-number": "~8.0.2", + "rc-mentions": "~2.5.0", + "rc-menu": "~9.10.0", + "rc-motion": "^2.7.3", + "rc-notification": "~5.0.4", + "rc-pagination": "~3.5.0", + "rc-picker": "~3.10.0", + "rc-progress": "~3.4.1", + "rc-rate": "~2.12.0", + "rc-resize-observer": "^1.2.0", + "rc-segmented": "~2.2.0", + "rc-select": "~14.5.0", + "rc-slider": "~10.1.0", + "rc-steps": "~6.0.1", + "rc-switch": "~4.1.0", + "rc-table": "~7.32.1", + "rc-tabs": "~12.9.0", + "rc-textarea": "~1.3.3", + "rc-tooltip": "~6.0.0", + "rc-tree": "~5.7.6", + "rc-tree-select": "~5.9.0", + "rc-upload": "~4.3.0", + "rc-util": "^5.32.0", + "scroll-into-view-if-needed": "^3.0.3", + "throttle-debounce": "^5.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/ant-design" + }, + "peerDependencies": { + "react": ">=16.9.0", + "react-dom": ">=16.9.0" + } + }, + "node_modules/argparse": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/argparse/-/argparse-2.0.1.tgz", + "integrity": "sha512-8+9WqebbFzpX9OR+Wa6O29asIogeRMzcGtAINdpMHHyAg10f05aSFVBbcEqGf/PXw1EjAZ+q2/bEBg3DvurK3Q==", + "dev": true + }, + "node_modules/array-tree-filter": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/array-tree-filter/-/array-tree-filter-2.1.0.tgz", + "integrity": "sha512-4ROwICNlNw/Hqa9v+rk5h22KjmzB1JGTMVKP2AKJBOCgb0yL0ASf0+YvCcLNNwquOHNX48jkeZIJ3a+oOQqKcw==" + }, + "node_modules/array-union": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/array-union/-/array-union-2.1.0.tgz", + "integrity": "sha512-HGyxoOTYUyCM6stUe6EJgnd4EoewAI7zMdfqO+kGjnlZmBDz/cR5pf8r/cR4Wq60sL/p0IkcjUEEPwS3GFrIyw==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/async-validator": { + "version": "4.2.5", + "resolved": "https://registry.npmjs.org/async-validator/-/async-validator-4.2.5.tgz", + "integrity": "sha512-7HhHjtERjqlNbZtqNqy2rckN/SpOOlmDliet+lP7k+eKZEjPk3DgyeU9lIXLdeLz0uBbbVp+9Qdow9wJWgwwfg==" + }, + "node_modules/asynckit": { + "version": "0.4.0", + "resolved": "https://registry.npmjs.org/asynckit/-/asynckit-0.4.0.tgz", + "integrity": "sha512-Oei9OH4tRh0YqU3GxhX79dM/mwVgvbZJaSNaRk+bshkj0S5cfHcgYakreBjrHwatXKbz+IoIdYLxrKim2MjW0Q==" + }, + "node_modules/axios": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/axios/-/axios-1.4.0.tgz", + "integrity": "sha512-S4XCWMEmzvo64T9GfvQDOXgYRDJ/wsSZc7Jvdgx5u1sd0JwsuPLqb3SYmusag+edF6ziyMensPVqLTSc1PiSEA==", + "dependencies": { + "follow-redirects": "^1.15.0", + "form-data": "^4.0.0", + "proxy-from-env": "^1.1.0" + } + }, + "node_modules/balanced-match": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz", + "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==", + "dev": true + }, + "node_modules/brace-expansion": { + "version": "1.1.11", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz", + "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==", + "dev": true, + "dependencies": { + "balanced-match": "^1.0.0", + "concat-map": "0.0.1" + } + }, + "node_modules/braces": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.2.tgz", + "integrity": "sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==", + "dev": true, + "dependencies": { + "fill-range": "^7.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/callsites": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/callsites/-/callsites-3.1.0.tgz", + "integrity": "sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ==", + "dev": true, + "engines": { + "node": ">=6" + } + }, + "node_modules/chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "dependencies": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/chalk?sponsor=1" + } + }, + "node_modules/classnames": { + "version": "2.3.2", + "resolved": "https://registry.npmjs.org/classnames/-/classnames-2.3.2.tgz", + "integrity": "sha512-CSbhY4cFEJRe6/GQzIk5qXZ4Jeg5pcsP7b5peFSDpffpe1cqjASH/n9UTjBwOp6XpMSTwQ8Za2K5V02ueA7Tmw==" + }, + "node_modules/color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "dependencies": { + "color-name": "~1.1.4" + }, + "engines": { + "node": ">=7.0.0" + } + }, + "node_modules/color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true + }, + "node_modules/combined-stream": { + "version": "1.0.8", + "resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.8.tgz", + "integrity": "sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==", + "dependencies": { + "delayed-stream": "~1.0.0" + }, + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/compute-scroll-into-view": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/compute-scroll-into-view/-/compute-scroll-into-view-3.0.3.tgz", + "integrity": "sha512-nadqwNxghAGTamwIqQSG433W6OADZx2vCo3UXHNrzTRHK/htu+7+L0zhjEoaeaQVNAi3YgqWDv8+tzf0hRfR+A==" + }, + "node_modules/concat-map": { + "version": "0.0.1", + "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", + "integrity": "sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg==", + "dev": true + }, + "node_modules/copy-to-clipboard": { + "version": "3.3.3", + "resolved": "https://registry.npmjs.org/copy-to-clipboard/-/copy-to-clipboard-3.3.3.tgz", + "integrity": "sha512-2KV8NhB5JqC3ky0r9PMCAZKbUHSwtEo4CwCs0KXgruG43gX5PMqDEBbVU4OUzw2MuAWUfsuFmWvEKG5QRfSnJA==", + "dependencies": { + "toggle-selection": "^1.0.6" + } + }, + "node_modules/cross-spawn": { + "version": "7.0.3", + "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.3.tgz", + "integrity": "sha512-iRDPJKUPVEND7dHPO8rkbOnPpyDygcDFtWjpeWNCgy8WP2rXcxXL8TskReQl6OrB2G7+UJrags1q15Fudc7G6w==", + "dev": true, + "dependencies": { + "path-key": "^3.1.0", + "shebang-command": "^2.0.0", + "which": "^2.0.1" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/csstype": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/csstype/-/csstype-3.1.2.tgz", + "integrity": "sha512-I7K1Uu0MBPzaFKg4nI5Q7Vs2t+3gWWW648spaF+Rg7pI9ds18Ugn+lvg4SHczUdKlHI5LWBXyqfS8+DufyBsgQ==" + }, + "node_modules/dayjs": { + "version": "1.11.9", + "resolved": "https://registry.npmjs.org/dayjs/-/dayjs-1.11.9.tgz", + "integrity": "sha512-QvzAURSbQ0pKdIye2txOzNaHmxtUBXerpY0FJsFXUMKbIZeFm5ht1LS/jFsrncjnmtv8HsG0W2g6c0zUjZWmpA==" + }, + "node_modules/debug": { + "version": "4.3.4", + "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.4.tgz", + "integrity": "sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ==", + "dev": true, + "dependencies": { + "ms": "2.1.2" + }, + "engines": { + "node": ">=6.0" + }, + "peerDependenciesMeta": { + "supports-color": { + "optional": true + } + } + }, + "node_modules/deep-is": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/deep-is/-/deep-is-0.1.4.tgz", + "integrity": "sha512-oIPzksmTg4/MriiaYGO+okXDT7ztn/w3Eptv/+gSIdMdKsJo0u4CfYNFJPy+4SKMuCqGw2wxnA+URMg3t8a/bQ==", + "dev": true + }, + "node_modules/delayed-stream": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz", + "integrity": "sha512-ZySD7Nf91aLB0RxL4KGrKHBXl7Eds1DAmEdcoVawXnLD7SDhpNgtuII2aAkg7a7QS41jxPSZ17p4VdGnMHk3MQ==", + "engines": { + "node": ">=0.4.0" + } + }, + "node_modules/dir-glob": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/dir-glob/-/dir-glob-3.0.1.tgz", + "integrity": "sha512-WkrWp9GR4KXfKGYzOLmTuGVi1UWFfws377n9cc55/tb6DuqyF6pcQ5AbiHEshaDpY9v6oaSr2XCDidGmMwdzIA==", + "dev": true, + "dependencies": { + "path-type": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/doctrine": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/doctrine/-/doctrine-3.0.0.tgz", + "integrity": "sha512-yS+Q5i3hBf7GBkd4KG8a7eBNNWNGLTaEwwYWUijIYM7zrlYDM0BFXHjjPWlWZ1Rg7UaddZeIDmi9jF3HmqiQ2w==", + "dev": true, + "dependencies": { + "esutils": "^2.0.2" + }, + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/dom-align": { + "version": "1.12.4", + "resolved": "https://registry.npmjs.org/dom-align/-/dom-align-1.12.4.tgz", + "integrity": "sha512-R8LUSEay/68zE5c8/3BDxiTEvgb4xZTF0RKmAHfiEVN3klfIpXfi2/QCoiWPccVQ0J/ZGdz9OjzL4uJEP/MRAw==" + }, + "node_modules/dotenv": { + "version": "16.3.1", + "resolved": "https://registry.npmjs.org/dotenv/-/dotenv-16.3.1.tgz", + "integrity": "sha512-IPzF4w4/Rd94bA9imS68tZBaYyBWSCE47V1RGuMrB94iyTOIEwRmVL2x/4An+6mETpLrKJ5hQkB8W4kFAadeIQ==", + "dev": true, + "engines": { + "node": ">=12" + }, + "funding": { + "url": "https://github.com/motdotla/dotenv?sponsor=1" + } + }, + "node_modules/esbuild": { + "version": "0.18.16", + "resolved": "https://registry.npmjs.org/esbuild/-/esbuild-0.18.16.tgz", + "integrity": "sha512-1xLsOXrDqwdHxyXb/x/SOyg59jpf/SH7YMvU5RNSU7z3TInaASNJWNFJ6iRvLvLETZMasF3d1DdZLg7sgRimRQ==", + "dev": true, + "hasInstallScript": true, + "bin": { + "esbuild": "bin/esbuild" + }, + "engines": { + "node": ">=12" + }, + "optionalDependencies": { + "@esbuild/android-arm": "0.18.16", + "@esbuild/android-arm64": "0.18.16", + "@esbuild/android-x64": "0.18.16", + "@esbuild/darwin-arm64": "0.18.16", + "@esbuild/darwin-x64": "0.18.16", + "@esbuild/freebsd-arm64": "0.18.16", + "@esbuild/freebsd-x64": "0.18.16", + "@esbuild/linux-arm": "0.18.16", + "@esbuild/linux-arm64": "0.18.16", + "@esbuild/linux-ia32": "0.18.16", + "@esbuild/linux-loong64": "0.18.16", + "@esbuild/linux-mips64el": "0.18.16", + "@esbuild/linux-ppc64": "0.18.16", + "@esbuild/linux-riscv64": "0.18.16", + "@esbuild/linux-s390x": "0.18.16", + "@esbuild/linux-x64": "0.18.16", + "@esbuild/netbsd-x64": "0.18.16", + "@esbuild/openbsd-x64": "0.18.16", + "@esbuild/sunos-x64": "0.18.16", + "@esbuild/win32-arm64": "0.18.16", + "@esbuild/win32-ia32": "0.18.16", + "@esbuild/win32-x64": "0.18.16" + } + }, + "node_modules/escape-string-regexp": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-4.0.0.tgz", + "integrity": "sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA==", + "dev": true, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/eslint": { + "version": "8.45.0", + "resolved": "https://registry.npmjs.org/eslint/-/eslint-8.45.0.tgz", + "integrity": "sha512-pd8KSxiQpdYRfYa9Wufvdoct3ZPQQuVuU5O6scNgMuOMYuxvH0IGaYK0wUFjo4UYYQQCUndlXiMbnxopwvvTiw==", + "dev": true, + "dependencies": { + "@eslint-community/eslint-utils": "^4.2.0", + "@eslint-community/regexpp": "^4.4.0", + "@eslint/eslintrc": "^2.1.0", + "@eslint/js": "8.44.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.6.0", + "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", + "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.3", + "strip-ansi": "^6.0.1", + "text-table": "^0.2.0" + }, + "bin": { + "eslint": "bin/eslint.js" + }, + "engines": { + "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + }, + "funding": { + "url": "https://opencollective.com/eslint" + } + }, + "node_modules/eslint-plugin-react-hooks": { + "version": "4.6.0", + "resolved": "https://registry.npmjs.org/eslint-plugin-react-hooks/-/eslint-plugin-react-hooks-4.6.0.tgz", + "integrity": "sha512-oFc7Itz9Qxh2x4gNHStv3BqJq54ExXmfC+a1NjAta66IAN87Wu0R/QArgIS9qKzX3dXKPI9H5crl9QchNMY9+g==", + "dev": true, + "engines": { + "node": ">=10" + }, + "peerDependencies": { + "eslint": "^3.0.0 || ^4.0.0 || ^5.0.0 || ^6.0.0 || ^7.0.0 || ^8.0.0-0" + } + }, + "node_modules/eslint-plugin-react-refresh": { + "version": "0.4.3", + "resolved": "https://registry.npmjs.org/eslint-plugin-react-refresh/-/eslint-plugin-react-refresh-0.4.3.tgz", + "integrity": "sha512-Hh0wv8bUNY877+sI0BlCUlsS0TYYQqvzEwJsJJPM2WF4RnTStSnSR3zdJYa2nPOJgg3UghXi54lVyMSmpCalzA==", + "dev": true, + "peerDependencies": { + "eslint": ">=7" + } + }, + "node_modules/eslint-scope": { + "version": "7.2.1", + "resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-7.2.1.tgz", + "integrity": "sha512-CvefSOsDdaYYvxChovdrPo/ZGt8d5lrJWleAc1diXRKhHGiTYEI26cvo8Kle/wGnsizoCJjK73FMg1/IkIwiNA==", + "dev": true, + "dependencies": { + "esrecurse": "^4.3.0", + "estraverse": "^5.2.0" + }, + "engines": { + "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + }, + "funding": { + "url": "https://opencollective.com/eslint" + } + }, + "node_modules/eslint-visitor-keys": { + "version": "3.4.1", + "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-3.4.1.tgz", + "integrity": "sha512-pZnmmLwYzf+kWaM/Qgrvpen51upAktaaiI01nsJD/Yr3lMOdNtq0cxkrrg16w64VtisN6okbs7Q8AfGqj4c9fA==", + "dev": true, + "engines": { + "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + }, + "funding": { + "url": "https://opencollective.com/eslint" + } + }, + "node_modules/espree": { + "version": "9.6.1", + "resolved": "https://registry.npmjs.org/espree/-/espree-9.6.1.tgz", + "integrity": "sha512-oruZaFkjorTpF32kDSI5/75ViwGeZginGGy2NoOSg3Q9bnwlnmDm4HLnkl0RE3n+njDXR037aY1+x58Z/zFdwQ==", + "dev": true, + "dependencies": { + "acorn": "^8.9.0", + "acorn-jsx": "^5.3.2", + "eslint-visitor-keys": "^3.4.1" + }, + "engines": { + "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + }, + "funding": { + "url": "https://opencollective.com/eslint" + } + }, + "node_modules/esquery": { + "version": "1.5.0", + "resolved": "https://registry.npmjs.org/esquery/-/esquery-1.5.0.tgz", + "integrity": "sha512-YQLXUplAwJgCydQ78IMJywZCceoqk1oH01OERdSAJc/7U2AylwjhSCLDEtqwg811idIS/9fIU5GjG73IgjKMVg==", + "dev": true, + "dependencies": { + "estraverse": "^5.1.0" + }, + "engines": { + "node": ">=0.10" + } + }, + "node_modules/esrecurse": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/esrecurse/-/esrecurse-4.3.0.tgz", + "integrity": "sha512-KmfKL3b6G+RXvP8N1vr3Tq1kL/oCFgn2NYXEtqP8/L3pKapUA4G8cFVaoF3SU323CD4XypR/ffioHmkti6/Tag==", + "dev": true, + "dependencies": { + "estraverse": "^5.2.0" + }, + "engines": { + "node": ">=4.0" + } + }, + "node_modules/estraverse": { + "version": "5.3.0", + "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-5.3.0.tgz", + "integrity": "sha512-MMdARuVEQziNTeJD8DgMqmhwR11BRQ/cBP+pLtYdSTnf3MIO8fFeiINEbX36ZdNlfU/7A9f3gUw49B3oQsvwBA==", + "dev": true, + "engines": { + "node": ">=4.0" + } + }, + "node_modules/esutils": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/esutils/-/esutils-2.0.3.tgz", + "integrity": "sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/fast-deep-equal": { + "version": "3.1.3", + "resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-3.1.3.tgz", + "integrity": "sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q==", + "dev": true + }, + "node_modules/fast-glob": { + "version": "3.3.1", + "resolved": "https://registry.npmjs.org/fast-glob/-/fast-glob-3.3.1.tgz", + "integrity": "sha512-kNFPyjhh5cKjrUltxs+wFx+ZkbRaxxmZ+X0ZU31SOsxCEtP9VPgtq2teZw1DebupL5GmDaNQ6yKMMVcM41iqDg==", + "dev": true, + "dependencies": { + "@nodelib/fs.stat": "^2.0.2", + "@nodelib/fs.walk": "^1.2.3", + "glob-parent": "^5.1.2", + "merge2": "^1.3.0", + "micromatch": "^4.0.4" + }, + "engines": { + "node": ">=8.6.0" + } + }, + "node_modules/fast-glob/node_modules/glob-parent": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.2.tgz", + "integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==", + "dev": true, + "dependencies": { + "is-glob": "^4.0.1" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/fast-json-stable-stringify": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/fast-json-stable-stringify/-/fast-json-stable-stringify-2.1.0.tgz", + "integrity": "sha512-lhd/wF+Lk98HZoTCtlVraHtfh5XYijIjalXck7saUtuanSDyLMxnHhSXEDJqHxD7msR8D0uCmqlkwjCV8xvwHw==", + "dev": true + }, + "node_modules/fast-levenshtein": { + "version": "2.0.6", + "resolved": "https://registry.npmjs.org/fast-levenshtein/-/fast-levenshtein-2.0.6.tgz", + "integrity": "sha512-DCXu6Ifhqcks7TZKY3Hxp3y6qphY5SJZmrWMDrKcERSOXWQdMhU9Ig/PYrzyw/ul9jOIyh0N4M0tbC5hodg8dw==", + "dev": true + }, + "node_modules/fastq": { + "version": "1.15.0", + "resolved": "https://registry.npmjs.org/fastq/-/fastq-1.15.0.tgz", + "integrity": "sha512-wBrocU2LCXXa+lWBt8RoIRD89Fi8OdABODa/kEnyeyjS5aZO5/GNvI5sEINADqP/h8M29UHTHUb53sUu5Ihqdw==", + "dev": true, + "dependencies": { + "reusify": "^1.0.4" + } + }, + "node_modules/file-entry-cache": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/file-entry-cache/-/file-entry-cache-6.0.1.tgz", + "integrity": "sha512-7Gps/XWymbLk2QLYK4NzpMOrYjMhdIxXuIvy2QBsLE6ljuodKvdkWs/cpyJJ3CVIVpH0Oi1Hvg1ovbMzLdFBBg==", + "dev": true, + "dependencies": { + "flat-cache": "^3.0.4" + }, + "engines": { + "node": "^10.12.0 || >=12.0.0" + } + }, + "node_modules/fill-range": { + "version": "7.0.1", + "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.0.1.tgz", + "integrity": "sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ==", + "dev": true, + "dependencies": { + "to-regex-range": "^5.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/find-up": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/find-up/-/find-up-5.0.0.tgz", + "integrity": "sha512-78/PXT1wlLLDgTzDs7sjq9hzz0vXD+zn+7wypEe4fXQxCmdmqfGsEPQxmiCSQI3ajFV91bVSsvNtrJRiW6nGng==", + "dev": true, + "dependencies": { + "locate-path": "^6.0.0", + "path-exists": "^4.0.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/flat-cache": { + "version": "3.0.4", + "resolved": "https://registry.npmjs.org/flat-cache/-/flat-cache-3.0.4.tgz", + "integrity": "sha512-dm9s5Pw7Jc0GvMYbshN6zchCA9RgQlzzEZX3vylR9IqFfS8XciblUXOKfW6SiuJ0e13eDYZoZV5wdrev7P3Nwg==", + "dev": true, + "dependencies": { + "flatted": "^3.1.0", + "rimraf": "^3.0.2" + }, + "engines": { + "node": "^10.12.0 || >=12.0.0" + } + }, + "node_modules/flatted": { + "version": "3.2.7", + "resolved": "https://registry.npmjs.org/flatted/-/flatted-3.2.7.tgz", + "integrity": "sha512-5nqDSxl8nn5BSNxyR3n4I6eDmbolI6WT+QqR547RwxQapgjQBmtktdP+HTBb/a/zLsbzERTONyUB5pefh5TtjQ==", + "dev": true + }, + "node_modules/follow-redirects": { + "version": "1.15.2", + "resolved": "https://registry.npmjs.org/follow-redirects/-/follow-redirects-1.15.2.tgz", + "integrity": "sha512-VQLG33o04KaQ8uYi2tVNbdrWp1QWxNNea+nmIB4EVM28v0hmP17z7aG1+wAkNzVq4KeXTq3221ye5qTJP91JwA==", + "funding": [ + { + "type": "individual", + "url": "https://github.com/sponsors/RubenVerborgh" + } + ], + "engines": { + "node": ">=4.0" + }, + "peerDependenciesMeta": { + "debug": { + "optional": true + } + } + }, + "node_modules/form-data": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/form-data/-/form-data-4.0.0.tgz", + "integrity": "sha512-ETEklSGi5t0QMZuiXoA/Q6vcnxcLQP5vdugSpuAyi6SVGi2clPPp+xgEhuMaHC+zGgn31Kd235W35f7Hykkaww==", + "dependencies": { + "asynckit": "^0.4.0", + "combined-stream": "^1.0.8", + "mime-types": "^2.1.12" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/fs.realpath": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz", + "integrity": "sha512-OO0pH2lK6a0hZnAdau5ItzHPI6pUlvI7jMVnxUQRtw4owF2wk8lOSabtGDCTP4Ggrg2MbGnWO9X8K1t4+fGMDw==", + "dev": true + }, + "node_modules/fsevents": { + "version": "2.3.2", + "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.2.tgz", + "integrity": "sha512-xiqMQR4xAeHTuB9uWm+fFRcIOgKBMiOBP+eXiyT7jsgVCq1bkVygt00oASowB7EdtpOHaaPgKt812P9ab+DDKA==", + "dev": true, + "hasInstallScript": true, + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": "^8.16.0 || ^10.6.0 || >=11.0.0" + } + }, + "node_modules/glob": { + "version": "7.2.3", + "resolved": "https://registry.npmjs.org/glob/-/glob-7.2.3.tgz", + "integrity": "sha512-nFR0zLpU2YCaRxwoCJvL6UvCH2JFyFVIvwTLsIf21AuHlMskA1hhTdk+LlYJtOlYt9v6dvszD2BGRqBL+iQK9Q==", + "dev": true, + "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" + }, + "engines": { + "node": "*" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/glob-parent": { + "version": "6.0.2", + "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-6.0.2.tgz", + "integrity": "sha512-XxwI8EOhVQgWp6iDL+3b0r86f4d6AX6zSU55HfB4ydCEuXLXc5FcYeOu+nnGftS4TEju/11rt4KJPTMgbfmv4A==", + "dev": true, + "dependencies": { + "is-glob": "^4.0.3" + }, + "engines": { + "node": ">=10.13.0" + } + }, + "node_modules/globals": { + "version": "13.20.0", + "resolved": "https://registry.npmjs.org/globals/-/globals-13.20.0.tgz", + "integrity": "sha512-Qg5QtVkCy/kv3FUSlu4ukeZDVf9ee0iXLAUYX13gbR17bnejFTzr4iS9bY7kwCf1NztRNm1t91fjOiyx4CSwPQ==", + "dev": true, + "dependencies": { + "type-fest": "^0.20.2" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/globby": { + "version": "11.1.0", + "resolved": "https://registry.npmjs.org/globby/-/globby-11.1.0.tgz", + "integrity": "sha512-jhIXaOzy1sb8IyocaruWSn1TjmnBVs8Ayhcy83rmxNJ8q2uWKCAj3CnJY+KpGSXCueAPc0i05kVvVKtP1t9S3g==", + "dev": true, + "dependencies": { + "array-union": "^2.1.0", + "dir-glob": "^3.0.1", + "fast-glob": "^3.2.9", + "ignore": "^5.2.0", + "merge2": "^1.4.1", + "slash": "^3.0.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/graphemer": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/graphemer/-/graphemer-1.4.0.tgz", + "integrity": "sha512-EtKwoO6kxCL9WO5xipiHTZlSzBm7WLT627TqC/uVRd0HKmq8NXyebnNYxDoBi7wt8eTWrUrKXCOVaFq9x1kgag==", + "dev": true + }, + "node_modules/hamt_plus": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/hamt_plus/-/hamt_plus-1.0.2.tgz", + "integrity": "sha512-t2JXKaehnMb9paaYA7J0BX8QQAY8lwfQ9Gjf4pg/mk4krt+cmwmU652HOoWonf+7+EQV97ARPMhhVgU1ra2GhA==" + }, + "node_modules/has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/ignore": { + "version": "5.2.4", + "resolved": "https://registry.npmjs.org/ignore/-/ignore-5.2.4.tgz", + "integrity": "sha512-MAb38BcSbH0eHNBxn7ql2NH/kX33OkB3lZ1BNdh7ENeRChHTYsTvWrMubiIAMNS2llXEEgZ1MUOBtXChP3kaFQ==", + "dev": true, + "engines": { + "node": ">= 4" + } + }, + "node_modules/import-fresh": { + "version": "3.3.0", + "resolved": "https://registry.npmjs.org/import-fresh/-/import-fresh-3.3.0.tgz", + "integrity": "sha512-veYYhQa+D1QBKznvhUHxb8faxlrwUnxseDAbAp457E0wLNio2bOSKnjYDhMj+YiAq61xrMGhQk9iXVk5FzgQMw==", + "dev": true, + "dependencies": { + "parent-module": "^1.0.0", + "resolve-from": "^4.0.0" + }, + "engines": { + "node": ">=6" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/imurmurhash": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz", + "integrity": "sha512-JmXMZ6wuvDmLiHEml9ykzqO6lwFbof0GG4IkcGaENdCRDDmMVnny7s5HsIgHCbaq0w2MyPhDqkhTUgS2LU2PHA==", + "dev": true, + "engines": { + "node": ">=0.8.19" + } + }, + "node_modules/inflight": { + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz", + "integrity": "sha512-k92I/b08q4wvFscXCLvqfsHCrjrF7yiXsQuIVvVE7N82W3+aqpzuUdBbfhWcy/FZR3/4IgflMgKLOsvPDrGCJA==", + "dev": true, + "dependencies": { + "once": "^1.3.0", + "wrappy": "1" + } + }, + "node_modules/inherits": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz", + "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==", + "dev": true + }, + "node_modules/is-extglob": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", + "integrity": "sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/is-glob": { + "version": "4.0.3", + "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.3.tgz", + "integrity": "sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==", + "dev": true, + "dependencies": { + "is-extglob": "^2.1.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/is-number": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz", + "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==", + "dev": true, + "engines": { + "node": ">=0.12.0" + } + }, + "node_modules/is-path-inside": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/is-path-inside/-/is-path-inside-3.0.3.tgz", + "integrity": "sha512-Fd4gABb+ycGAmKou8eMftCupSir5lRxqf4aD/vd0cD2qc4HL07OjCeuHMr8Ro4CoMaeCKDB0/ECBOVWjTwUvPQ==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/isexe": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz", + "integrity": "sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw==", + "dev": true + }, + "node_modules/js-tokens": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-4.0.0.tgz", + "integrity": "sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==" + }, + "node_modules/js-yaml": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-4.1.0.tgz", + "integrity": "sha512-wpxZs9NoxZaJESJGIZTyDEaYpl0FKSA+FB9aJiyemKhMwkxQg63h4T1KJgUGHpTqPDNRcmmYLugrRjJlBtWvRA==", + "dev": true, + "dependencies": { + "argparse": "^2.0.1" + }, + "bin": { + "js-yaml": "bin/js-yaml.js" + } + }, + "node_modules/json-schema-traverse": { + "version": "0.4.1", + "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz", + "integrity": "sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg==", + "dev": true + }, + "node_modules/json-stable-stringify-without-jsonify": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/json-stable-stringify-without-jsonify/-/json-stable-stringify-without-jsonify-1.0.1.tgz", + "integrity": "sha512-Bdboy+l7tA3OGW6FjyFHWkP5LuByj1Tk33Ljyq0axyzdk9//JSi2u3fP1QSmd1KNwq6VOKYGlAu87CisVir6Pw==", + "dev": true + }, + "node_modules/json2mq": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/json2mq/-/json2mq-0.2.0.tgz", + "integrity": "sha512-SzoRg7ux5DWTII9J2qkrZrqV1gt+rTaoufMxEzXbS26Uid0NwaJd123HcoB80TgubEppxxIGdNxCx50fEoEWQA==", + "dependencies": { + "string-convert": "^0.2.0" + } + }, + "node_modules/levn": { + "version": "0.4.1", + "resolved": "https://registry.npmjs.org/levn/-/levn-0.4.1.tgz", + "integrity": "sha512-+bT2uH4E5LGE7h/n3evcS/sQlJXCpIp6ym8OWJ5eV6+67Dsql/LaaT7qJBAt2rzfoa/5QBGBhxDix1dMt2kQKQ==", + "dev": true, + "dependencies": { + "prelude-ls": "^1.2.1", + "type-check": "~0.4.0" + }, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/locate-path": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-6.0.0.tgz", + "integrity": "sha512-iPZK6eYjbxRu3uB4/WZ3EsEIMJFMqAoopl3R+zuq0UjcAm/MO6KCweDgPfP3elTztoKP3KtnVHxTn2NHBSDVUw==", + "dev": true, + "dependencies": { + "p-locate": "^5.0.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/lodash.merge": { + "version": "4.6.2", + "resolved": "https://registry.npmjs.org/lodash.merge/-/lodash.merge-4.6.2.tgz", + "integrity": "sha512-0KpjqXRVvrYyCsX1swR/XTK0va6VQkQM6MNo7PqW77ByjAhoARA8EfrP1N4+KlKj8YS0ZUCtRT/YUuhyYDujIQ==", + "dev": true + }, + "node_modules/loose-envify": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/loose-envify/-/loose-envify-1.4.0.tgz", + "integrity": "sha512-lyuxPGr/Wfhrlem2CL/UcnUc1zcqKAImBDzukY7Y5F/yQiNdko6+fRLevlw1HgMySw7f611UIY408EtxRSoK3Q==", + "dependencies": { + "js-tokens": "^3.0.0 || ^4.0.0" + }, + "bin": { + "loose-envify": "cli.js" + } + }, + "node_modules/lru-cache": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-6.0.0.tgz", + "integrity": "sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA==", + "dev": true, + "dependencies": { + "yallist": "^4.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/merge2": { + "version": "1.4.1", + "resolved": "https://registry.npmjs.org/merge2/-/merge2-1.4.1.tgz", + "integrity": "sha512-8q7VEgMJW4J8tcfVPy8g09NcQwZdbwFEqhe/WZkoIzjn/3TGDwtOCYtXGxA3O8tPzpczCCDgv+P2P5y00ZJOOg==", + "dev": true, + "engines": { + "node": ">= 8" + } + }, + "node_modules/micromatch": { + "version": "4.0.5", + "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-4.0.5.tgz", + "integrity": "sha512-DMy+ERcEW2q8Z2Po+WNXuw3c5YaUSFjAO5GsJqfEl7UjvtIuFKO6ZrKvcItdy98dwFI2N1tg3zNIdKaQT+aNdA==", + "dev": true, + "dependencies": { + "braces": "^3.0.2", + "picomatch": "^2.3.1" + }, + "engines": { + "node": ">=8.6" + } + }, + "node_modules/mime-db": { + "version": "1.52.0", + "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.52.0.tgz", + "integrity": "sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/mime-types": { + "version": "2.1.35", + "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.35.tgz", + "integrity": "sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==", + "dependencies": { + "mime-db": "1.52.0" + }, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/minimatch": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", + "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", + "dev": true, + "dependencies": { + "brace-expansion": "^1.1.7" + }, + "engines": { + "node": "*" + } + }, + "node_modules/moment": { + "version": "2.29.4", + "resolved": "https://registry.npmjs.org/moment/-/moment-2.29.4.tgz", + "integrity": "sha512-5LC9SOxjSc2HF6vO2CyuTDNivEdoz2IvyJJGj6X8DJ0eFyfszE0QiEd+iXmBvUP3WHxSjFH/vIsA0EN00cgr8w==", + "optional": true, + "peer": true, + "engines": { + "node": "*" + } + }, + "node_modules/ms": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", + "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==", + "dev": true + }, + "node_modules/nanoid": { + "version": "3.3.6", + "resolved": "https://registry.npmjs.org/nanoid/-/nanoid-3.3.6.tgz", + "integrity": "sha512-BGcqMMJuToF7i1rt+2PWSNVnWIkGCU78jBG3RxO/bZlnZPK2Cmi2QaffxGO/2RvWi9sL+FAiRiXMgsyxQ1DIDA==", + "dev": true, + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/ai" + } + ], + "bin": { + "nanoid": "bin/nanoid.cjs" + }, + "engines": { + "node": "^10 || ^12 || ^13.7 || ^14 || >=15.0.1" + } + }, + "node_modules/natural-compare": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/natural-compare/-/natural-compare-1.4.0.tgz", + "integrity": "sha512-OWND8ei3VtNC9h7V60qff3SVobHr996CTwgxubgyQYEpg290h9J0buyECNNJexkFm5sOajh5G116RYA1c8ZMSw==", + "dev": true + }, + "node_modules/natural-compare-lite": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/natural-compare-lite/-/natural-compare-lite-1.4.0.tgz", + "integrity": "sha512-Tj+HTDSJJKaZnfiuw+iaF9skdPpTo2GtEly5JHnWV/hfv2Qj/9RKsGISQtLh2ox3l5EAGw487hnBee0sIJ6v2g==", + "dev": true + }, + "node_modules/once": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz", + "integrity": "sha512-lNaJgI+2Q5URQBkccEKHTQOPaXdUxnZZElQTZY0MFUAuaEqe1E+Nyvgdz/aIyNi6Z9MzO5dv1H8n58/GELp3+w==", + "dev": true, + "dependencies": { + "wrappy": "1" + } + }, + "node_modules/optionator": { + "version": "0.9.3", + "resolved": "https://registry.npmjs.org/optionator/-/optionator-0.9.3.tgz", + "integrity": "sha512-JjCoypp+jKn1ttEFExxhetCKeJt9zhAgAve5FXHixTvFDW/5aEktX9bufBKLRRMdU7bNtpLfcGu94B3cdEJgjg==", + "dev": true, + "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" + }, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/p-limit": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-3.1.0.tgz", + "integrity": "sha512-TYOanM3wGwNGsZN2cVTYPArw454xnXj5qmWF1bEoAc4+cU/ol7GVh7odevjp1FNHduHc3KZMcFduxU5Xc6uJRQ==", + "dev": true, + "dependencies": { + "yocto-queue": "^0.1.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/p-locate": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-5.0.0.tgz", + "integrity": "sha512-LaNjtRWUBY++zB5nE/NwcaoMylSPk+S+ZHNB1TzdbMJMny6dynpAGt7X/tl/QYq3TIeE6nxHppbo2LGymrG5Pw==", + "dev": true, + "dependencies": { + "p-limit": "^3.0.2" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/parent-module": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/parent-module/-/parent-module-1.0.1.tgz", + "integrity": "sha512-GQ2EWRpQV8/o+Aw8YqtfZZPfNRWZYkbidE9k5rpl/hC3vtHHBfGm2Ifi6qWV+coDGkrUKZAxE3Lot5kcsRlh+g==", + "dev": true, + "dependencies": { + "callsites": "^3.0.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/path-exists": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-4.0.0.tgz", + "integrity": "sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/path-is-absolute": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/path-is-absolute/-/path-is-absolute-1.0.1.tgz", + "integrity": "sha512-AVbw3UJ2e9bq64vSaS9Am0fje1Pa8pbGqTTsmXfaIiMpnr5DlDhfJOuLj9Sf95ZPVDAUerDfEk88MPmPe7UCQg==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/path-key": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/path-key/-/path-key-3.1.1.tgz", + "integrity": "sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/path-type": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/path-type/-/path-type-4.0.0.tgz", + "integrity": "sha512-gDKb8aZMDeD/tZWs9P6+q0J9Mwkdl6xMV8TjnGP3qJVJ06bdMgkbBlLU8IdfOsIsFz2BW1rNVT3XuNEl8zPAvw==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/picocolors": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/picocolors/-/picocolors-1.0.0.tgz", + "integrity": "sha512-1fygroTLlHu66zi26VoTDv8yRgm0Fccecssto+MhsZ0D/DGW2sm8E8AjW7NU5VVTRt5GxbeZ5qBuJr+HyLYkjQ==", + "dev": true + }, + "node_modules/picomatch": { + "version": "2.3.1", + "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.1.tgz", + "integrity": "sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==", + "dev": true, + "engines": { + "node": ">=8.6" + }, + "funding": { + "url": "https://github.com/sponsors/jonschlinkert" + } + }, + "node_modules/postcss": { + "version": "8.4.27", + "resolved": "https://registry.npmjs.org/postcss/-/postcss-8.4.27.tgz", + "integrity": "sha512-gY/ACJtJPSmUFPDCHtX78+01fHa64FaU4zaaWfuh1MhGJISufJAH4cun6k/8fwsHYeK4UQmENQK+tRLCFJE8JQ==", + "dev": true, + "funding": [ + { + "type": "opencollective", + "url": "https://opencollective.com/postcss/" + }, + { + "type": "tidelift", + "url": "https://tidelift.com/funding/github/npm/postcss" + }, + { + "type": "github", + "url": "https://github.com/sponsors/ai" + } + ], + "dependencies": { + "nanoid": "^3.3.6", + "picocolors": "^1.0.0", + "source-map-js": "^1.0.2" + }, + "engines": { + "node": "^10 || ^12 || >=14" + } + }, + "node_modules/preact": { + "version": "10.12.1", + "resolved": "https://registry.npmjs.org/preact/-/preact-10.12.1.tgz", + "integrity": "sha512-l8386ixSsBdbreOAkqtrwqHwdvR35ID8c3rKPa8lCWuO86dBi32QWHV4vfsZK1utLLFMvw+Z5Ad4XLkZzchscg==", + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/preact" + } + }, + "node_modules/prelude-ls": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/prelude-ls/-/prelude-ls-1.2.1.tgz", + "integrity": "sha512-vkcDPrRZo1QZLbn5RLGPpg/WmIQ65qoWWhcGKf/b5eplkkarX0m9z8ppCat4mlOqUsWpyNuYgO3VRyrYHSzX5g==", + "dev": true, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/proxy-from-env": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/proxy-from-env/-/proxy-from-env-1.1.0.tgz", + "integrity": "sha512-D+zkORCbA9f1tdWRK0RaCR3GPv50cMxcrz4X8k5LTSUD1Dkw47mKJEZQNunItRTkWwgtaUSo1RVFRIG9ZXiFYg==" + }, + "node_modules/punycode": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/punycode/-/punycode-2.3.0.tgz", + "integrity": "sha512-rRV+zQD8tVFys26lAGR9WUuS4iUAngJScM+ZRSKtvl5tKeZ2t5bvdNFdNHBW9FWR4guGHlgmsZ1G7BSm2wTbuA==", + "dev": true, + "engines": { + "node": ">=6" + } + }, + "node_modules/qrcode.react": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/qrcode.react/-/qrcode.react-3.1.0.tgz", + "integrity": "sha512-oyF+Urr3oAMUG/OiOuONL3HXM+53wvuH3mtIWQrYmsXoAq0DkvZp2RYUWFSMFtbdOpuS++9v+WAkzNVkMlNW6Q==", + "peerDependencies": { + "react": "^16.8.0 || ^17.0.0 || ^18.0.0" + } + }, + "node_modules/queue-microtask": { + "version": "1.2.3", + "resolved": "https://registry.npmjs.org/queue-microtask/-/queue-microtask-1.2.3.tgz", + "integrity": "sha512-NuaNSa6flKT5JaSYQzJok04JzTL1CA6aGhv5rfLW3PgqA+M2ChpZQnAC8h8i4ZFkBS8X5RqkDBHA7r4hej3K9A==", + "dev": true, + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/feross" + }, + { + "type": "patreon", + "url": "https://www.patreon.com/feross" + }, + { + "type": "consulting", + "url": "https://feross.org/support" + } + ] + }, + "node_modules/rc-align": { + "version": "4.0.15", + "resolved": "https://registry.npmjs.org/rc-align/-/rc-align-4.0.15.tgz", + "integrity": "sha512-wqJtVH60pka/nOX7/IspElA8gjPNQKIx/ZqJ6heATCkXpe1Zg4cPVrMD2vC96wjsFFL8WsmhPbx9tdMo1qqlIA==", + "dependencies": { + "@babel/runtime": "^7.10.1", + "classnames": "2.x", + "dom-align": "^1.7.0", + "rc-util": "^5.26.0", + "resize-observer-polyfill": "^1.5.1" + }, + "peerDependencies": { + "react": ">=16.9.0", + "react-dom": ">=16.9.0" + } + }, + "node_modules/rc-cascader": { + "version": "3.12.1", + "resolved": "https://registry.npmjs.org/rc-cascader/-/rc-cascader-3.12.1.tgz", + "integrity": "sha512-g6In2y6eudHXS/Fs9dKFhp9acvHRUPqem/7xReR9ng8M1pNAE137uGBOt9WNpgsKT/cDGudXZQVehaBwAKg6hQ==", + "dependencies": { + "@babel/runtime": "^7.12.5", + "array-tree-filter": "^2.1.0", + "classnames": "^2.3.1", + "rc-select": "~14.5.0", + "rc-tree": "~5.7.0", + "rc-util": "^5.6.1" + }, + "peerDependencies": { + "react": ">=16.9.0", + "react-dom": ">=16.9.0" + } + }, + "node_modules/rc-checkbox": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/rc-checkbox/-/rc-checkbox-3.1.0.tgz", + "integrity": "sha512-PAwpJFnBa3Ei+5pyqMMXdcKYKNBMS+TvSDiLdDnARnMJHC8ESxwPfm4Ao1gJiKtWLdmGfigascnCpwrHFgoOBQ==", + "dependencies": { + "@babel/runtime": "^7.10.1", + "classnames": "^2.3.2", + "rc-util": "^5.25.2" + }, + "peerDependencies": { + "react": ">=16.9.0", + "react-dom": ">=16.9.0" + } + }, + "node_modules/rc-collapse": { + "version": "3.7.0", + "resolved": "https://registry.npmjs.org/rc-collapse/-/rc-collapse-3.7.0.tgz", + "integrity": "sha512-Cir1c89cENiK5wryd9ut+XltrIfx/+KH1/63uJIVjuXkgfrIvIy6W1fYGgEYtttbHW2fEfxg1s31W+Vm98fSRw==", + "dependencies": { + "@babel/runtime": "^7.10.1", + "classnames": "2.x", + "rc-motion": "^2.3.4", + "rc-util": "^5.27.0" + }, + "peerDependencies": { + "react": ">=16.9.0", + "react-dom": ">=16.9.0" + } + }, + "node_modules/rc-dialog": { + "version": "9.1.0", + "resolved": "https://registry.npmjs.org/rc-dialog/-/rc-dialog-9.1.0.tgz", + "integrity": "sha512-5ry+JABAWEbaKyYsmITtrJbZbJys8CtMyzV8Xn4LYuXMeUx5XVHNyJRoqLFE4AzBuXXzOWeaC49cg+XkxK6kHA==", + "dependencies": { + "@babel/runtime": "^7.10.1", + "@rc-component/portal": "^1.0.0-8", + "classnames": "^2.2.6", + "rc-motion": "^2.3.0", + "rc-util": "^5.21.0" + }, + "peerDependencies": { + "react": ">=16.9.0", + "react-dom": ">=16.9.0" + } + }, + "node_modules/rc-drawer": { + "version": "6.2.0", + "resolved": "https://registry.npmjs.org/rc-drawer/-/rc-drawer-6.2.0.tgz", + "integrity": "sha512-spPkZ3WvP0U0vy5dyzSwlUJ/+vLFtjP/cTwSwejhQRoDBaexSZHsBhELoCZcEggI7LQ7typmtG30lAue2HEhvA==", + "dependencies": { + "@babel/runtime": "^7.10.1", + "@rc-component/portal": "^1.1.1", + "classnames": "^2.2.6", + "rc-motion": "^2.6.1", + "rc-util": "^5.21.2" + }, + "peerDependencies": { + "react": ">=16.9.0", + "react-dom": ">=16.9.0" + } + }, + "node_modules/rc-dropdown": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/rc-dropdown/-/rc-dropdown-4.1.0.tgz", + "integrity": "sha512-VZjMunpBdlVzYpEdJSaV7WM7O0jf8uyDjirxXLZRNZ+tAC+NzD3PXPEtliFwGzVwBBdCmGuSqiS9DWcOLxQ9tw==", + "dependencies": { + "@babel/runtime": "^7.18.3", + "@rc-component/trigger": "^1.7.0", + "classnames": "^2.2.6", + "rc-util": "^5.17.0" + }, + "peerDependencies": { + "react": ">=16.11.0", + "react-dom": ">=16.11.0" + } + }, + "node_modules/rc-field-form": { + "version": "1.34.2", + "resolved": "https://registry.npmjs.org/rc-field-form/-/rc-field-form-1.34.2.tgz", + "integrity": "sha512-BdciU5C7dBO51/9ZKcMvK2f8zaaO12Lt1eBhlAo8nNv+6htlNcgY9DAkUlZ7gfyWjnCc1Oo4hHIXau1m6tLw1A==", + "dependencies": { + "@babel/runtime": "^7.18.0", + "async-validator": "^4.1.0", + "rc-util": "^5.32.2" + }, + "engines": { + "node": ">=8.x" + }, + "peerDependencies": { + "react": ">=16.9.0", + "react-dom": ">=16.9.0" + } + }, + "node_modules/rc-image": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/rc-image/-/rc-image-7.0.0.tgz", + "integrity": "sha512-pOr/LYthg5a+R2LDlFPv8u2ndX4aJQNghWCiWxflmLglC3p0uts/NIWLAituQOKvV1wO1aFI1CZtLMT7jrU3vA==", + "dependencies": { + "@babel/runtime": "^7.11.2", + "@rc-component/portal": "^1.0.2", + "classnames": "^2.2.6", + "rc-dialog": "~9.1.0", + "rc-motion": "^2.6.2", + "rc-util": "^5.34.1" + }, + "peerDependencies": { + "react": ">=16.9.0", + "react-dom": ">=16.9.0" + } + }, + "node_modules/rc-input": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/rc-input/-/rc-input-1.1.0.tgz", + "integrity": "sha512-izuNXPABQPh4KD7ANFcTrIGp9EZU0FkjTw6AvwCQ/rGPrdDsUTHLsp/Wju/kzGMLJFJWKNF3smbmXRNO23DtXA==", + "dependencies": { + "@babel/runtime": "^7.11.1", + "classnames": "^2.2.1", + "rc-util": "^5.18.1" + }, + "peerDependencies": { + "react": ">=16.0.0", + "react-dom": ">=16.0.0" + } + }, + "node_modules/rc-input-number": { + "version": "8.0.3", + "resolved": "https://registry.npmjs.org/rc-input-number/-/rc-input-number-8.0.3.tgz", + "integrity": "sha512-GHfWvufXEmwF/wtR8oPZNTuMdFb/rvx/+Sp2bZfaPftM+LFFdO8o3/PaeTk8DKt0Tv+u5Zuf68lqLdGCkmAXRg==", + "dependencies": { + "@babel/runtime": "^7.10.1", + "@rc-component/mini-decimal": "^1.0.1", + "classnames": "^2.2.5", + "rc-input": "~1.1.0", + "rc-util": "^5.28.0" + }, + "peerDependencies": { + "react": ">=16.9.0", + "react-dom": ">=16.9.0" + } + }, + "node_modules/rc-mentions": { + "version": "2.5.0", + "resolved": "https://registry.npmjs.org/rc-mentions/-/rc-mentions-2.5.0.tgz", + "integrity": "sha512-rERXsbUTNVrb5T/iDC0ki/SRGWJnOVraDy6O25Us3FSpuUZ3uq2TPZB4fRk0Hss5kyiEPzz2sprhkI4b+F4jUw==", + "dependencies": { + "@babel/runtime": "^7.22.5", + "@rc-component/trigger": "^1.5.0", + "classnames": "^2.2.6", + "rc-input": "~1.1.0", + "rc-menu": "~9.10.0", + "rc-textarea": "~1.3.0", + "rc-util": "^5.22.5" + }, + "peerDependencies": { + "react": ">=16.9.0", + "react-dom": ">=16.9.0" + } + }, + "node_modules/rc-menu": { + "version": "9.10.0", + "resolved": "https://registry.npmjs.org/rc-menu/-/rc-menu-9.10.0.tgz", + "integrity": "sha512-g27kpXaAoJh/fkPZF65/d4V+w4DhDeqomBdPcGnkFAcJnEM4o21TnVccrBUoDedLKzC7wJRw1Q7VTqEsfEufmw==", + "dependencies": { + "@babel/runtime": "^7.10.1", + "@rc-component/trigger": "^1.6.2", + "classnames": "2.x", + "rc-motion": "^2.4.3", + "rc-overflow": "^1.3.1", + "rc-util": "^5.27.0" + }, + "peerDependencies": { + "react": ">=16.9.0", + "react-dom": ">=16.9.0" + } + }, + "node_modules/rc-motion": { + "version": "2.7.3", + "resolved": "https://registry.npmjs.org/rc-motion/-/rc-motion-2.7.3.tgz", + "integrity": "sha512-2xUvo8yGHdOHeQbdI8BtBsCIrWKchEmFEIskf0nmHtJsou+meLd/JE+vnvSX2JxcBrJtXY2LuBpxAOxrbY/wMQ==", + "dependencies": { + "@babel/runtime": "^7.11.1", + "classnames": "^2.2.1", + "rc-util": "^5.21.0" + }, + "peerDependencies": { + "react": ">=16.9.0", + "react-dom": ">=16.9.0" + } + }, + "node_modules/rc-notification": { + "version": "5.0.5", + "resolved": "https://registry.npmjs.org/rc-notification/-/rc-notification-5.0.5.tgz", + "integrity": "sha512-uEz2jggourwv/rR0obe7RHEa63UchqX4k+e+Qt2c3LaY7U9Tc+L6ANhzgCKYSA/afm0ebjmNZHoB5Cv47xEOcA==", + "dependencies": { + "@babel/runtime": "^7.10.1", + "classnames": "2.x", + "rc-motion": "^2.6.0", + "rc-util": "^5.20.1" + }, + "engines": { + "node": ">=8.x" + }, + "peerDependencies": { + "react": ">=16.9.0", + "react-dom": ">=16.9.0" + } + }, + "node_modules/rc-overflow": { + "version": "1.3.1", + "resolved": "https://registry.npmjs.org/rc-overflow/-/rc-overflow-1.3.1.tgz", + "integrity": "sha512-RY0nVBlfP9CkxrpgaLlGzkSoh9JhjJLu6Icqs9E7CW6Ewh9s0peF9OHIex4OhfoPsR92LR0fN6BlCY9Z4VoUtA==", + "dependencies": { + "@babel/runtime": "^7.11.1", + "classnames": "^2.2.1", + "rc-resize-observer": "^1.0.0", + "rc-util": "^5.19.2" + }, + "peerDependencies": { + "react": ">=16.9.0", + "react-dom": ">=16.9.0" + } + }, + "node_modules/rc-pagination": { + "version": "3.5.0", + "resolved": "https://registry.npmjs.org/rc-pagination/-/rc-pagination-3.5.0.tgz", + "integrity": "sha512-lUBVtVVUn7gGsq4mTyVpcZQr+AMcljbMiL/HcCmSdFrcsK0iZVKwwbXDxhz2IV0JXUs9Hzepr5sQFaF+9ad/pQ==", + "dependencies": { + "@babel/runtime": "^7.10.1", + "classnames": "^2.2.1", + "rc-util": "^5.32.2" + }, + "peerDependencies": { + "react": ">=16.9.0", + "react-dom": ">=16.9.0" + } + }, + "node_modules/rc-picker": { + "version": "3.10.0", + "resolved": "https://registry.npmjs.org/rc-picker/-/rc-picker-3.10.0.tgz", + "integrity": "sha512-Euki50qtEct6ByOeYlnA4TLs/LcXz7BAYS4cmCTKJ3dWg2sNTVtredLdbS9aJ/9fhMacxGAYAlcQJpQx+av43A==", + "dependencies": { + "@babel/runtime": "^7.10.1", + "@rc-component/trigger": "^1.5.0", + "classnames": "^2.2.1", + "rc-util": "^5.30.0" + }, + "engines": { + "node": ">=8.x" + }, + "peerDependencies": { + "date-fns": ">= 2.x", + "dayjs": ">= 1.x", + "luxon": ">= 3.x", + "moment": ">= 2.x", + "react": ">=16.9.0", + "react-dom": ">=16.9.0" + }, + "peerDependenciesMeta": { + "date-fns": { + "optional": true + }, + "dayjs": { + "optional": true + }, + "luxon": { + "optional": true + }, + "moment": { + "optional": true + } + } + }, + "node_modules/rc-progress": { + "version": "3.4.2", + "resolved": "https://registry.npmjs.org/rc-progress/-/rc-progress-3.4.2.tgz", + "integrity": "sha512-iAGhwWU+tsayP+Jkl9T4+6rHeQTG9kDz8JAHZk4XtQOcYN5fj9H34NXNEdRdZx94VUDHMqCb1yOIvi8eJRh67w==", + "dependencies": { + "@babel/runtime": "^7.10.1", + "classnames": "^2.2.6", + "rc-util": "^5.16.1" + }, + "peerDependencies": { + "react": ">=16.9.0", + "react-dom": ">=16.9.0" + } + }, + "node_modules/rc-rate": { + "version": "2.12.0", + "resolved": "https://registry.npmjs.org/rc-rate/-/rc-rate-2.12.0.tgz", + "integrity": "sha512-g092v5iZCdVzbjdn28FzvWebK2IutoVoiTeqoLTj9WM7SjA/gOJIw5/JFZMRyJYYVe1jLAU2UhAfstIpCNRozg==", + "dependencies": { + "@babel/runtime": "^7.10.1", + "classnames": "^2.2.5", + "rc-util": "^5.0.1" + }, + "engines": { + "node": ">=8.x" + }, + "peerDependencies": { + "react": ">=16.9.0", + "react-dom": ">=16.9.0" + } + }, + "node_modules/rc-resize-observer": { + "version": "1.3.1", + "resolved": "https://registry.npmjs.org/rc-resize-observer/-/rc-resize-observer-1.3.1.tgz", + "integrity": "sha512-iFUdt3NNhflbY3mwySv5CA1TC06zdJ+pfo0oc27xpf4PIOvfZwZGtD9Kz41wGYqC4SLio93RVAirSSpYlV/uYg==", + "dependencies": { + "@babel/runtime": "^7.20.7", + "classnames": "^2.2.1", + "rc-util": "^5.27.0", + "resize-observer-polyfill": "^1.5.1" + }, + "peerDependencies": { + "react": ">=16.9.0", + "react-dom": ">=16.9.0" + } + }, + "node_modules/rc-segmented": { + "version": "2.2.2", + "resolved": "https://registry.npmjs.org/rc-segmented/-/rc-segmented-2.2.2.tgz", + "integrity": "sha512-Mq52M96QdHMsNdE/042ibT5vkcGcD5jxKp7HgPC2SRofpia99P5fkfHy1pEaajLMF/kj0+2Lkq1UZRvqzo9mSA==", + "dependencies": { + "@babel/runtime": "^7.11.1", + "classnames": "^2.2.1", + "rc-motion": "^2.4.4", + "rc-util": "^5.17.0" + }, + "peerDependencies": { + "react": ">=16.0.0", + "react-dom": ">=16.0.0" + } + }, + "node_modules/rc-select": { + "version": "14.5.2", + "resolved": "https://registry.npmjs.org/rc-select/-/rc-select-14.5.2.tgz", + "integrity": "sha512-Np/lDHvxCnVhVsheQjSV1I/OMJTWJf1n10wq8q1AGy3ytyYLfjNpi6uaz/pmjsbbiSddSWzJnNZCli9LmgBZsA==", + "dependencies": { + "@babel/runtime": "^7.10.1", + "@rc-component/trigger": "^1.5.0", + "classnames": "2.x", + "rc-motion": "^2.0.1", + "rc-overflow": "^1.0.0", + "rc-util": "^5.16.1", + "rc-virtual-list": "^3.5.2" + }, + "engines": { + "node": ">=8.x" + }, + "peerDependencies": { + "react": "*", + "react-dom": "*" + } + }, + "node_modules/rc-slider": { + "version": "10.1.1", + "resolved": "https://registry.npmjs.org/rc-slider/-/rc-slider-10.1.1.tgz", + "integrity": "sha512-gn8oXazZISEhnmRinI89Z/JD/joAaM35jp+gDtIVSTD/JJMCCBqThqLk1SVJmvtfeiEF/kKaFY0+qt4SDHFUDw==", + "dependencies": { + "@babel/runtime": "^7.10.1", + "classnames": "^2.2.5", + "rc-util": "^5.27.0" + }, + "engines": { + "node": ">=8.x" + }, + "peerDependencies": { + "react": ">=16.9.0", + "react-dom": ">=16.9.0" + } + }, + "node_modules/rc-steps": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/rc-steps/-/rc-steps-6.0.1.tgz", + "integrity": "sha512-lKHL+Sny0SeHkQKKDJlAjV5oZ8DwCdS2hFhAkIjuQt1/pB81M0cA0ErVFdHq9+jmPmFw1vJB2F5NBzFXLJxV+g==", + "dependencies": { + "@babel/runtime": "^7.16.7", + "classnames": "^2.2.3", + "rc-util": "^5.16.1" + }, + "engines": { + "node": ">=8.x" + }, + "peerDependencies": { + "react": ">=16.9.0", + "react-dom": ">=16.9.0" + } + }, + "node_modules/rc-switch": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/rc-switch/-/rc-switch-4.1.0.tgz", + "integrity": "sha512-TI8ufP2Az9oEbvyCeVE4+90PDSljGyuwix3fV58p7HV2o4wBnVToEyomJRVyTaZeqNPAp+vqeo4Wnj5u0ZZQBg==", + "dependencies": { + "@babel/runtime": "^7.21.0", + "classnames": "^2.2.1", + "rc-util": "^5.30.0" + }, + "peerDependencies": { + "react": ">=16.9.0", + "react-dom": ">=16.9.0" + } + }, + "node_modules/rc-table": { + "version": "7.32.1", + "resolved": "https://registry.npmjs.org/rc-table/-/rc-table-7.32.1.tgz", + "integrity": "sha512-fHMQteKMocUC9I9Vex3eBLH7QsiaMR/qtzh3B1Ty2PoNGwVTwVdDFyRL05zch+JU3KnNNczgQeVvtf/p//gdrQ==", + "dependencies": { + "@babel/runtime": "^7.10.1", + "@rc-component/context": "^1.3.0", + "classnames": "^2.2.5", + "rc-resize-observer": "^1.1.0", + "rc-util": "^5.27.1" + }, + "engines": { + "node": ">=8.x" + }, + "peerDependencies": { + "react": ">=16.9.0", + "react-dom": ">=16.9.0" + } + }, + "node_modules/rc-tabs": { + "version": "12.9.0", + "resolved": "https://registry.npmjs.org/rc-tabs/-/rc-tabs-12.9.0.tgz", + "integrity": "sha512-2HnVowgMVrq0DfQtyu4mCd9E6pXlWNdM6VaDvOOHMsLYqPmpY+7zBqUC6YrrQ9xYXHciTS0e7TtjOHIvpVCHLQ==", + "dependencies": { + "@babel/runtime": "^7.11.2", + "classnames": "2.x", + "rc-dropdown": "~4.1.0", + "rc-menu": "~9.10.0", + "rc-motion": "^2.6.2", + "rc-resize-observer": "^1.0.0", + "rc-util": "^5.16.0" + }, + "engines": { + "node": ">=8.x" + }, + "peerDependencies": { + "react": ">=16.9.0", + "react-dom": ">=16.9.0" + } + }, + "node_modules/rc-textarea": { + "version": "1.3.3", + "resolved": "https://registry.npmjs.org/rc-textarea/-/rc-textarea-1.3.3.tgz", + "integrity": "sha512-846kjD/RYZx/th32FW4T80IrRTt2dT7+kxdToI7pwzJPlsfmZyo8e2F2m0FLcvriv6rtAUMSqQRH1HC3i+sAbw==", + "dependencies": { + "@babel/runtime": "^7.10.1", + "classnames": "^2.2.1", + "rc-input": "~1.1.0", + "rc-resize-observer": "^1.0.0", + "rc-util": "^5.27.0" + }, + "peerDependencies": { + "react": ">=16.9.0", + "react-dom": ">=16.9.0" + } + }, + "node_modules/rc-tooltip": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/rc-tooltip/-/rc-tooltip-6.0.1.tgz", + "integrity": "sha512-MdvPlsD1fDSxKp9+HjXrc/CxLmA/s11QYIh1R7aExxfodKP7CZA++DG1AjrW80F8IUdHYcR43HAm0Y2BYPelHA==", + "dependencies": { + "@babel/runtime": "^7.11.2", + "@rc-component/trigger": "^1.0.4", + "classnames": "^2.3.1" + }, + "peerDependencies": { + "react": ">=16.9.0", + "react-dom": ">=16.9.0" + } + }, + "node_modules/rc-tree": { + "version": "5.7.9", + "resolved": "https://registry.npmjs.org/rc-tree/-/rc-tree-5.7.9.tgz", + "integrity": "sha512-1hKkToz/EVjJlMVwmZnpXeLXt/1iQMsaAq9m+GNkUbK746gkc7QpJXSN/TzjhTI5Hi+LOSlrMaXLMT0bHPqILQ==", + "dependencies": { + "@babel/runtime": "^7.10.1", + "classnames": "2.x", + "rc-motion": "^2.0.1", + "rc-util": "^5.16.1", + "rc-virtual-list": "^3.5.1" + }, + "engines": { + "node": ">=10.x" + }, + "peerDependencies": { + "react": "*", + "react-dom": "*" + } + }, + "node_modules/rc-tree-select": { + "version": "5.9.0", + "resolved": "https://registry.npmjs.org/rc-tree-select/-/rc-tree-select-5.9.0.tgz", + "integrity": "sha512-oh3blESzLfLCBPSiVDtZ2irzrWWZUMeHvnSwRvFo79br8Z+K/1OhXhXBZmROvfKwaH8YUugAQy8B2j5EGQbdyA==", + "dependencies": { + "@babel/runtime": "^7.10.1", + "classnames": "2.x", + "rc-select": "~14.5.0", + "rc-tree": "~5.7.0", + "rc-util": "^5.16.1" + }, + "peerDependencies": { + "react": "*", + "react-dom": "*" + } + }, + "node_modules/rc-upload": { + "version": "4.3.4", + "resolved": "https://registry.npmjs.org/rc-upload/-/rc-upload-4.3.4.tgz", + "integrity": "sha512-uVbtHFGNjHG/RyAfm9fluXB6pvArAGyAx8z7XzXXyorEgVIWj6mOlriuDm0XowDHYz4ycNK0nE0oP3cbFnzxiQ==", + "dependencies": { + "@babel/runtime": "^7.18.3", + "classnames": "^2.2.5", + "rc-util": "^5.2.0" + }, + "peerDependencies": { + "react": ">=16.9.0", + "react-dom": ">=16.9.0" + } + }, + "node_modules/rc-util": { + "version": "5.34.1", + "resolved": "https://registry.npmjs.org/rc-util/-/rc-util-5.34.1.tgz", + "integrity": "sha512-SqiUT8Ssgh5C+hu4y887xwCrMNcxLm6ScOo8AFlWYYF3z9uNNiPpwwSjvicqOlWd79rNw1g44rnP7tz9MrO1ZQ==", + "dependencies": { + "@babel/runtime": "^7.18.3", + "react-is": "^16.12.0" + }, + "peerDependencies": { + "react": ">=16.9.0", + "react-dom": ">=16.9.0" + } + }, + "node_modules/rc-virtual-list": { + "version": "3.5.3", + "resolved": "https://registry.npmjs.org/rc-virtual-list/-/rc-virtual-list-3.5.3.tgz", + "integrity": "sha512-rG6IuD4EYM8K6oZ8Shu2BC/CmcTdqng4yBWkc/5fjWhB20bl6QwR2Upyt7+MxvfscoVm8zOQY+tcpEO5cu4GaQ==", + "dependencies": { + "@babel/runtime": "^7.20.0", + "classnames": "^2.2.6", + "rc-resize-observer": "^1.0.0", + "rc-util": "^5.15.0" + }, + "engines": { + "node": ">=8.x" + }, + "peerDependencies": { + "react": "*", + "react-dom": "*" + } + }, + "node_modules/react": { + "version": "18.2.0", + "resolved": "https://registry.npmjs.org/react/-/react-18.2.0.tgz", + "integrity": "sha512-/3IjMdb2L9QbBdWiW5e3P2/npwMBaU9mHCSCUzNln0ZCYbcfTsGbTJrU/kGemdH2IWmB2ioZ+zkxtmq6g09fGQ==", + "dependencies": { + "loose-envify": "^1.1.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/react-dom": { + "version": "18.2.0", + "resolved": "https://registry.npmjs.org/react-dom/-/react-dom-18.2.0.tgz", + "integrity": "sha512-6IMTriUmvsjHUjNtEDudZfuDQUoWXVxKHhlEGSk81n4YFS+r/Kl99wXiwlVXtPBtJenozv2P+hxDsw9eA7Xo6g==", + "dependencies": { + "loose-envify": "^1.1.0", + "scheduler": "^0.23.0" + }, + "peerDependencies": { + "react": "^18.2.0" + } + }, + "node_modules/react-is": { + "version": "16.13.1", + "resolved": "https://registry.npmjs.org/react-is/-/react-is-16.13.1.tgz", + "integrity": "sha512-24e6ynE2H+OKt4kqsOvNd8kBpV65zoxbA4BVsEOB3ARVWQki/DHzaUoC5KuON/BiccDaCCTZBuOcfZs70kR8bQ==" + }, + "node_modules/react-router": { + "version": "6.14.2", + "resolved": "https://registry.npmjs.org/react-router/-/react-router-6.14.2.tgz", + "integrity": "sha512-09Zss2dE2z+T1D03IheqAFtK4UzQyX8nFPWx6jkwdYzGLXd5ie06A6ezS2fO6zJfEb/SpG6UocN2O1hfD+2urQ==", + "dependencies": { + "@remix-run/router": "1.7.2" + }, + "engines": { + "node": ">=14" + }, + "peerDependencies": { + "react": ">=16.8" + } + }, + "node_modules/react-router-dom": { + "version": "6.14.2", + "resolved": "https://registry.npmjs.org/react-router-dom/-/react-router-dom-6.14.2.tgz", + "integrity": "sha512-5pWX0jdKR48XFZBuJqHosX3AAHjRAzygouMTyimnBPOLdY3WjzUSKhus2FVMihUFWzeLebDgr4r8UeQFAct7Bg==", + "dependencies": { + "@remix-run/router": "1.7.2", + "react-router": "6.14.2" + }, + "engines": { + "node": ">=14" + }, + "peerDependencies": { + "react": ">=16.8", + "react-dom": ">=16.8" + } + }, + "node_modules/recoil": { + "version": "0.7.7", + "resolved": "https://registry.npmjs.org/recoil/-/recoil-0.7.7.tgz", + "integrity": "sha512-8Og5KPQW9LwC577Vc7Ug2P0vQshkv1y3zG3tSSkWMqkWSwHmE+by06L8JtnGocjW6gcCvfwB3YtrJG6/tWivNQ==", + "dependencies": { + "hamt_plus": "1.0.2" + }, + "peerDependencies": { + "react": ">=16.13.1" + }, + "peerDependenciesMeta": { + "react-dom": { + "optional": true + }, + "react-native": { + "optional": true + } + } + }, + "node_modules/regenerator-runtime": { + "version": "0.13.11", + "resolved": "https://registry.npmjs.org/regenerator-runtime/-/regenerator-runtime-0.13.11.tgz", + "integrity": "sha512-kY1AZVr2Ra+t+piVaJ4gxaFaReZVH40AKNo7UCX6W+dEwBo/2oZJzqfuN1qLq1oL45o56cPaTXELwrTh8Fpggg==" + }, + "node_modules/resize-observer-polyfill": { + "version": "1.5.1", + "resolved": "https://registry.npmjs.org/resize-observer-polyfill/-/resize-observer-polyfill-1.5.1.tgz", + "integrity": "sha512-LwZrotdHOo12nQuZlHEmtuXdqGoOD0OhaxopaNFxWzInpEgaLWoVuAMbTzixuosCx2nEG58ngzW3vxdWoxIgdg==" + }, + "node_modules/resolve-from": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-4.0.0.tgz", + "integrity": "sha512-pb/MYmXstAkysRFx8piNI1tGFNQIFA3vkE3Gq4EuA1dF6gHp/+vgZqsCGJapvy8N3Q+4o7FwvquPJcnZ7RYy4g==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/reusify": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/reusify/-/reusify-1.0.4.tgz", + "integrity": "sha512-U9nH88a3fc/ekCF1l0/UP1IosiuIjyTh7hBvXVMHYgVcfGvt897Xguj2UOLDeI5BG2m7/uwyaLVT6fbtCwTyzw==", + "dev": true, + "engines": { + "iojs": ">=1.0.0", + "node": ">=0.10.0" + } + }, + "node_modules/rimraf": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-3.0.2.tgz", + "integrity": "sha512-JZkJMZkAGFFPP2YqXZXPbMlMBgsxzE8ILs4lMIX/2o0L9UBw9O/Y3o6wFw/i9YLapcUJWwqbi3kdxIPdC62TIA==", + "dev": true, + "dependencies": { + "glob": "^7.1.3" + }, + "bin": { + "rimraf": "bin.js" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/rollup": { + "version": "3.26.3", + "resolved": "https://registry.npmjs.org/rollup/-/rollup-3.26.3.tgz", + "integrity": "sha512-7Tin0C8l86TkpcMtXvQu6saWH93nhG3dGQ1/+l5V2TDMceTxO7kDiK6GzbfLWNNxqJXm591PcEZUozZm51ogwQ==", + "dev": true, + "bin": { + "rollup": "dist/bin/rollup" + }, + "engines": { + "node": ">=14.18.0", + "npm": ">=8.0.0" + }, + "optionalDependencies": { + "fsevents": "~2.3.2" + } + }, + "node_modules/run-parallel": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/run-parallel/-/run-parallel-1.2.0.tgz", + "integrity": "sha512-5l4VyZR86LZ/lDxZTR6jqL8AFE2S0IFLMP26AbjsLVADxHdhB/c0GUsH+y39UfCi3dzz8OlQuPmnaJOMoDHQBA==", + "dev": true, + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/feross" + }, + { + "type": "patreon", + "url": "https://www.patreon.com/feross" + }, + { + "type": "consulting", + "url": "https://feross.org/support" + } + ], + "dependencies": { + "queue-microtask": "^1.2.2" + } + }, + "node_modules/scheduler": { + "version": "0.23.0", + "resolved": "https://registry.npmjs.org/scheduler/-/scheduler-0.23.0.tgz", + "integrity": "sha512-CtuThmgHNg7zIZWAXi3AsyIzA3n4xx7aNyjwC2VJldO2LMVDhFK+63xGqq6CsJH4rTAt6/M+N4GhZiDYPx9eUw==", + "dependencies": { + "loose-envify": "^1.1.0" + } + }, + "node_modules/scroll-into-view-if-needed": { + "version": "3.0.10", + "resolved": "https://registry.npmjs.org/scroll-into-view-if-needed/-/scroll-into-view-if-needed-3.0.10.tgz", + "integrity": "sha512-t44QCeDKAPf1mtQH3fYpWz8IM/DyvHLjs8wUvvwMYxk5moOqCzrMSxK6HQVD0QVmVjXFavoFIPRVrMuJPKAvtg==", + "dependencies": { + "compute-scroll-into-view": "^3.0.2" + } + }, + "node_modules/semver": { + "version": "7.5.4", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.5.4.tgz", + "integrity": "sha512-1bCSESV6Pv+i21Hvpxp3Dx+pSD8lIPt8uVjRrxAUt/nbswYc+tK6Y2btiULjd4+fnq15PX+nqQDC7Oft7WkwcA==", + "dev": true, + "dependencies": { + "lru-cache": "^6.0.0" + }, + "bin": { + "semver": "bin/semver.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/shebang-command": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-2.0.0.tgz", + "integrity": "sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA==", + "dev": true, + "dependencies": { + "shebang-regex": "^3.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/shebang-regex": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-3.0.0.tgz", + "integrity": "sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/slash": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/slash/-/slash-3.0.0.tgz", + "integrity": "sha512-g9Q1haeby36OSStwb4ntCGGGaKsaVSjQ68fBxoQcutl5fS1vuY18H3wSt3jFyFtrkx+Kz0V1G85A4MyAdDMi2Q==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/source-map-js": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/source-map-js/-/source-map-js-1.0.2.tgz", + "integrity": "sha512-R0XvVJ9WusLiqTCEiGCmICCMplcCkIwwR11mOSD9CR5u+IXYdiseeEuXCVAjS54zqwkLcPNnmU4OeJ6tUrWhDw==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/string-convert": { + "version": "0.2.1", + "resolved": "https://registry.npmjs.org/string-convert/-/string-convert-0.2.1.tgz", + "integrity": "sha512-u/1tdPl4yQnPBjnVrmdLo9gtuLvELKsAoRapekWggdiQNvvvum+jYF329d84NAa660KQw7pB2n36KrIKVoXa3A==" + }, + "node_modules/strip-ansi": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", + "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", + "dev": true, + "dependencies": { + "ansi-regex": "^5.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/strip-json-comments": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-3.1.1.tgz", + "integrity": "sha512-6fPc+R4ihwqP6N/aIv2f1gMH8lOVtWQHoqC4yK6oSDVVocumAsfCqjkXnqiYMhmMwS/mEHLp7Vehlt3ql6lEig==", + "dev": true, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/stylis": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/stylis/-/stylis-4.3.0.tgz", + "integrity": "sha512-E87pIogpwUsUwXw7dNyU4QDjdgVMy52m+XEOPEKUn161cCzWjjhPSQhByfd1CcNvrOLnXQ6OnnZDwnJrz/Z4YQ==" + }, + "node_modules/supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "dependencies": { + "has-flag": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/text-table": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/text-table/-/text-table-0.2.0.tgz", + "integrity": "sha512-N+8UisAXDGk8PFXP4HAzVR9nbfmVJ3zYLAWiTIoqC5v5isinhr+r5uaO8+7r3BMfuNIufIsA7RdpVgacC2cSpw==", + "dev": true + }, + "node_modules/throttle-debounce": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/throttle-debounce/-/throttle-debounce-5.0.0.tgz", + "integrity": "sha512-2iQTSgkkc1Zyk0MeVrt/3BvuOXYPl/R8Z0U2xxo9rjwNciaHDG3R+Lm6dh4EeUci49DanvBnuqI6jshoQQRGEg==", + "engines": { + "node": ">=12.22" + } + }, + "node_modules/to-regex-range": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz", + "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==", + "dev": true, + "dependencies": { + "is-number": "^7.0.0" + }, + "engines": { + "node": ">=8.0" + } + }, + "node_modules/toggle-selection": { + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/toggle-selection/-/toggle-selection-1.0.6.tgz", + "integrity": "sha512-BiZS+C1OS8g/q2RRbJmy59xpyghNBqrr6k5L/uKBGRsTfxmu3ffiRnd8mlGPUVayg8pvfi5urfnu8TU7DVOkLQ==" + }, + "node_modules/ts-api-utils": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/ts-api-utils/-/ts-api-utils-1.0.1.tgz", + "integrity": "sha512-lC/RGlPmwdrIBFTX59wwNzqh7aR2otPNPR/5brHZm/XKFYKsfqxihXUe9pU3JI+3vGkl+vyCoNNnPhJn3aLK1A==", + "dev": true, + "engines": { + "node": ">=16.13.0" + }, + "peerDependencies": { + "typescript": ">=4.2.0" + } + }, + "node_modules/type-check": { + "version": "0.4.0", + "resolved": "https://registry.npmjs.org/type-check/-/type-check-0.4.0.tgz", + "integrity": "sha512-XleUoc9uwGXqjWwXaUTZAmzMcFZ5858QA2vvx1Ur5xIcixXIP+8LnFDgRplU30us6teqdlskFfu+ae4K79Ooew==", + "dev": true, + "dependencies": { + "prelude-ls": "^1.2.1" + }, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/type-fest": { + "version": "0.20.2", + "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.20.2.tgz", + "integrity": "sha512-Ne+eE4r0/iWnpAxD852z3A+N0Bt5RN//NjJwRd2VFHEmrywxf5vsZlh4R6lixl6B+wz/8d+maTSAkN1FIkI3LQ==", + "dev": true, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/typescript": { + "version": "5.1.6", + "resolved": "https://registry.npmjs.org/typescript/-/typescript-5.1.6.tgz", + "integrity": "sha512-zaWCozRZ6DLEWAWFrVDz1H6FVXzUSfTy5FUMWsQlU8Ym5JP9eO4xkTIROFCQvhQf61z6O/G6ugw3SgAnvvm+HA==", + "dev": true, + "bin": { + "tsc": "bin/tsc", + "tsserver": "bin/tsserver" + }, + "engines": { + "node": ">=14.17" + } + }, + "node_modules/uri-js": { + "version": "4.4.1", + "resolved": "https://registry.npmjs.org/uri-js/-/uri-js-4.4.1.tgz", + "integrity": "sha512-7rKUyy33Q1yc98pQ1DAmLtwX109F7TIfWlW1Ydo8Wl1ii1SeHieeh0HHfPeL2fMXK6z0s8ecKs9frCuLJvndBg==", + "dev": true, + "dependencies": { + "punycode": "^2.1.0" + } + }, + "node_modules/vite": { + "version": "4.4.6", + "resolved": "https://registry.npmjs.org/vite/-/vite-4.4.6.tgz", + "integrity": "sha512-EY6Mm8vJ++S3D4tNAckaZfw3JwG3wa794Vt70M6cNJ6NxT87yhq7EC8Rcap3ahyHdo8AhCmV9PTk+vG1HiYn1A==", + "dev": true, + "dependencies": { + "esbuild": "^0.18.10", + "postcss": "^8.4.26", + "rollup": "^3.25.2" + }, + "bin": { + "vite": "bin/vite.js" + }, + "engines": { + "node": "^14.18.0 || >=16.0.0" + }, + "funding": { + "url": "https://github.com/vitejs/vite?sponsor=1" + }, + "optionalDependencies": { + "fsevents": "~2.3.2" + }, + "peerDependencies": { + "@types/node": ">= 14", + "less": "*", + "lightningcss": "^1.21.0", + "sass": "*", + "stylus": "*", + "sugarss": "*", + "terser": "^5.4.0" + }, + "peerDependenciesMeta": { + "@types/node": { + "optional": true + }, + "less": { + "optional": true + }, + "lightningcss": { + "optional": true + }, + "sass": { + "optional": true + }, + "stylus": { + "optional": true + }, + "sugarss": { + "optional": true + }, + "terser": { + "optional": true + } + } + }, + "node_modules/which": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz", + "integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==", + "dev": true, + "dependencies": { + "isexe": "^2.0.0" + }, + "bin": { + "node-which": "bin/node-which" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/wrappy": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", + "integrity": "sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ==", + "dev": true + }, + "node_modules/yallist": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/yallist/-/yallist-4.0.0.tgz", + "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==", + "dev": true + }, + "node_modules/yocto-queue": { + "version": "0.1.0", + "resolved": "https://registry.npmjs.org/yocto-queue/-/yocto-queue-0.1.0.tgz", + "integrity": "sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q==", + "dev": true, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + } + }, + "dependencies": { + "@aashutoshrathi/word-wrap": { + "version": "1.2.6", + "resolved": "https://registry.npmjs.org/@aashutoshrathi/word-wrap/-/word-wrap-1.2.6.tgz", + "integrity": "sha512-1Yjs2SvM8TflER/OD3cOjhWWOZb58A2t7wpE2S9XfBYTiIl+XFhQG2bjy4Pu1I+EAlCNUzRDYDdFwFYUKvXcIA==", + "dev": true + }, + "@ant-design/colors": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/@ant-design/colors/-/colors-7.0.0.tgz", + "integrity": "sha512-iVm/9PfGCbC0dSMBrz7oiEXZaaGH7ceU40OJEfKmyuzR9R5CRimJYPlRiFtMQGQcbNMea/ePcoIebi4ASGYXtg==", + "requires": { + "@ctrl/tinycolor": "^3.4.0" + } + }, + "@ant-design/cssinjs": { + "version": "1.13.2", + "resolved": "https://registry.npmjs.org/@ant-design/cssinjs/-/cssinjs-1.13.2.tgz", + "integrity": "sha512-II3QJx6V6boYkAIUiEd1/hquSeX1r67sUOtsvco35fTmV46JvkJz946/6K+ikiuODBSE1kLf7fJ5gHetQyUyww==", + "requires": { + "@babel/runtime": "^7.11.1", + "@emotion/hash": "^0.8.0", + "@emotion/unitless": "^0.7.5", + "classnames": "^2.3.1", + "csstype": "^3.0.10", + "rc-util": "^5.34.1", + "stylis": "^4.0.13" + } + }, + "@ant-design/icons": { + "version": "5.1.4", + "resolved": "https://registry.npmjs.org/@ant-design/icons/-/icons-5.1.4.tgz", + "integrity": "sha512-YHKL7Jx3bM12OxvtiYDon04BsBT/6LGitYEqar3GljzWaAyMOAD8i/uF1Rsi5Us/YNdWWXBGSvZV2OZWMpJlcA==", + "requires": { + "@ant-design/colors": "^7.0.0", + "@ant-design/icons-svg": "^4.2.1", + "@babel/runtime": "^7.11.2", + "classnames": "^2.2.6", + "rc-util": "^5.31.1" + } + }, + "@ant-design/icons-svg": { + "version": "4.2.1", + "resolved": "https://registry.npmjs.org/@ant-design/icons-svg/-/icons-svg-4.2.1.tgz", + "integrity": "sha512-EB0iwlKDGpG93hW8f85CTJTs4SvMX7tt5ceupvhALp1IF44SeUFOMhKUOYqpsoYWQKAOuTRDMqn75rEaKDp0Xw==" + }, + "@ant-design/react-slick": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/@ant-design/react-slick/-/react-slick-1.0.1.tgz", + "integrity": "sha512-ARM0TmpGdDuUVE10NwUCENQlJSInNKo5NiBjL5szu5BxWNEHNwQMcDrlVCqFbkvFLy+2CvywW8Y59QJtC0YDag==", + "requires": { + "@babel/runtime": "^7.10.4", + "classnames": "^2.2.5", + "json2mq": "^0.2.0", + "resize-observer-polyfill": "^1.5.1", + "throttle-debounce": "^5.0.0" + } + }, + "@babel/runtime": { + "version": "7.22.6", + "resolved": "https://registry.npmjs.org/@babel/runtime/-/runtime-7.22.6.tgz", + "integrity": "sha512-wDb5pWm4WDdF6LFUde3Jl8WzPA+3ZbxYqkC6xAXuD3irdEHN1k0NfTRrJD8ZD378SJ61miMLCqIOXYhd8x+AJQ==", + "requires": { + "regenerator-runtime": "^0.13.11" + } + }, + "@ctrl/tinycolor": { + "version": "3.6.0", + "resolved": "https://registry.npmjs.org/@ctrl/tinycolor/-/tinycolor-3.6.0.tgz", + "integrity": "sha512-/Z3l6pXthq0JvMYdUFyX9j0MaCltlIn6mfh9jLyQwg5aPKxkyNa0PTHtU1AlFXLNk55ZuAeJRcpvq+tmLfKmaQ==" + }, + "@emotion/hash": { + "version": "0.8.0", + "resolved": "https://registry.npmjs.org/@emotion/hash/-/hash-0.8.0.tgz", + "integrity": "sha512-kBJtf7PH6aWwZ6fka3zQ0p6SBYzx4fl1LoZXE2RrnYST9Xljm7WfKJrU4g/Xr3Beg72MLrp1AWNUmuYJTL7Cow==" + }, + "@emotion/unitless": { + "version": "0.7.5", + "resolved": "https://registry.npmjs.org/@emotion/unitless/-/unitless-0.7.5.tgz", + "integrity": "sha512-OWORNpfjMsSSUBVrRBVGECkhWcULOAJz9ZW8uK9qgxD+87M7jHRcvh/A96XXNhXTLmKcoYSQtBEX7lHMO7YRwg==" + }, + "@esbuild/android-arm": { + "version": "0.18.16", + "resolved": "https://registry.npmjs.org/@esbuild/android-arm/-/android-arm-0.18.16.tgz", + "integrity": "sha512-gCHjjQmA8L0soklKbLKA6pgsLk1byULuHe94lkZDzcO3/Ta+bbeewJioEn1Fr7kgy9NWNFy/C+MrBwC6I/WCug==", + "dev": true, + "optional": true + }, + "@esbuild/android-arm64": { + "version": "0.18.16", + "resolved": "https://registry.npmjs.org/@esbuild/android-arm64/-/android-arm64-0.18.16.tgz", + "integrity": "sha512-wsCqSPqLz+6Ov+OM4EthU43DyYVVyfn15S4j1bJzylDpc1r1jZFFfJQNfDuT8SlgwuqpmpJXK4uPlHGw6ve7eA==", + "dev": true, + "optional": true + }, + "@esbuild/android-x64": { + "version": "0.18.16", + "resolved": "https://registry.npmjs.org/@esbuild/android-x64/-/android-x64-0.18.16.tgz", + "integrity": "sha512-ldsTXolyA3eTQ1//4DS+E15xl0H/3DTRJaRL0/0PgkqDsI0fV/FlOtD+h0u/AUJr+eOTlZv4aC9gvfppo3C4sw==", + "dev": true, + "optional": true + }, + "@esbuild/darwin-arm64": { + "version": "0.18.16", + "resolved": "https://registry.npmjs.org/@esbuild/darwin-arm64/-/darwin-arm64-0.18.16.tgz", + "integrity": "sha512-aBxruWCII+OtluORR/KvisEw0ALuw/qDQWvkoosA+c/ngC/Kwk0lLaZ+B++LLS481/VdydB2u6tYpWxUfnLAIw==", + "dev": true, + "optional": true + }, + "@esbuild/darwin-x64": { + "version": "0.18.16", + "resolved": "https://registry.npmjs.org/@esbuild/darwin-x64/-/darwin-x64-0.18.16.tgz", + "integrity": "sha512-6w4Dbue280+rp3LnkgmriS1icOUZDyPuZo/9VsuMUTns7SYEiOaJ7Ca1cbhu9KVObAWfmdjUl4gwy9TIgiO5eA==", + "dev": true, + "optional": true + }, + "@esbuild/freebsd-arm64": { + "version": "0.18.16", + "resolved": "https://registry.npmjs.org/@esbuild/freebsd-arm64/-/freebsd-arm64-0.18.16.tgz", + "integrity": "sha512-x35fCebhe9s979DGKbVAwXUOcTmCIE32AIqB9CB1GralMIvxdnMLAw5CnID17ipEw9/3MvDsusj/cspYt2ZLNQ==", + "dev": true, + "optional": true + }, + "@esbuild/freebsd-x64": { + "version": "0.18.16", + "resolved": "https://registry.npmjs.org/@esbuild/freebsd-x64/-/freebsd-x64-0.18.16.tgz", + "integrity": "sha512-YM98f+PeNXF3GbxIJlUsj+McUWG1irguBHkszCIwfr3BXtXZsXo0vqybjUDFfu9a8Wr7uUD/YSmHib+EeGAFlg==", + "dev": true, + "optional": true + }, + "@esbuild/linux-arm": { + "version": "0.18.16", + "resolved": "https://registry.npmjs.org/@esbuild/linux-arm/-/linux-arm-0.18.16.tgz", + "integrity": "sha512-b5ABb+5Ha2C9JkeZXV+b+OruR1tJ33ePmv9ZwMeETSEKlmu/WJ45XTTG+l6a2KDsQtJJ66qo/hbSGBtk0XVLHw==", + "dev": true, + "optional": true + }, + "@esbuild/linux-arm64": { + "version": "0.18.16", + "resolved": "https://registry.npmjs.org/@esbuild/linux-arm64/-/linux-arm64-0.18.16.tgz", + "integrity": "sha512-XIqhNUxJiuy+zsR77+H5Z2f7s4YRlriSJKtvx99nJuG5ATuJPjmZ9n0ANgnGlPCpXGSReFpgcJ7O3SMtzIFeiQ==", + "dev": true, + "optional": true + }, + "@esbuild/linux-ia32": { + "version": "0.18.16", + "resolved": "https://registry.npmjs.org/@esbuild/linux-ia32/-/linux-ia32-0.18.16.tgz", + "integrity": "sha512-no+pfEpwnRvIyH+txbBAWtjxPU9grslmTBfsmDndj7bnBmr55rOo/PfQmRfz7Qg9isswt1FP5hBbWb23fRWnow==", + "dev": true, + "optional": true + }, + "@esbuild/linux-loong64": { + "version": "0.18.16", + "resolved": "https://registry.npmjs.org/@esbuild/linux-loong64/-/linux-loong64-0.18.16.tgz", + "integrity": "sha512-Zbnczs9ZXjmo0oZSS0zbNlJbcwKXa/fcNhYQjahDs4Xg18UumpXG/lwM2lcSvHS3mTrRyCYZvJbmzYc4laRI1g==", + "dev": true, + "optional": true + }, + "@esbuild/linux-mips64el": { + "version": "0.18.16", + "resolved": "https://registry.npmjs.org/@esbuild/linux-mips64el/-/linux-mips64el-0.18.16.tgz", + "integrity": "sha512-YMF7hih1HVR/hQVa/ot4UVffc5ZlrzEb3k2ip0nZr1w6fnYypll9td2qcoMLvd3o8j3y6EbJM3MyIcXIVzXvQQ==", + "dev": true, + "optional": true + }, + "@esbuild/linux-ppc64": { + "version": "0.18.16", + "resolved": "https://registry.npmjs.org/@esbuild/linux-ppc64/-/linux-ppc64-0.18.16.tgz", + "integrity": "sha512-Wkz++LZ29lDwUyTSEnzDaaP5OveOgTU69q9IyIw9WqLRxM4BjTBjz9un4G6TOvehWpf/J3gYVFN96TjGHrbcNQ==", + "dev": true, + "optional": true + }, + "@esbuild/linux-riscv64": { + "version": "0.18.16", + "resolved": "https://registry.npmjs.org/@esbuild/linux-riscv64/-/linux-riscv64-0.18.16.tgz", + "integrity": "sha512-LFMKZ30tk78/mUv1ygvIP+568bwf4oN6reG/uczXnz6SvFn4e2QUFpUpZY9iSJT6Qpgstrhef/nMykIXZtZWGQ==", + "dev": true, + "optional": true + }, + "@esbuild/linux-s390x": { + "version": "0.18.16", + "resolved": "https://registry.npmjs.org/@esbuild/linux-s390x/-/linux-s390x-0.18.16.tgz", + "integrity": "sha512-3ZC0BgyYHYKfZo3AV2/66TD/I9tlSBaW7eWTEIkrQQKfJIifKMMttXl9FrAg+UT0SGYsCRLI35Gwdmm96vlOjg==", + "dev": true, + "optional": true + }, + "@esbuild/linux-x64": { + "version": "0.18.16", + "resolved": "https://registry.npmjs.org/@esbuild/linux-x64/-/linux-x64-0.18.16.tgz", + "integrity": "sha512-xu86B3647DihHJHv/wx3NCz2Dg1gjQ8bbf9cVYZzWKY+gsvxYmn/lnVlqDRazObc3UMwoHpUhNYaZset4X8IPA==", + "dev": true, + "optional": true + }, + "@esbuild/netbsd-x64": { + "version": "0.18.16", + "resolved": "https://registry.npmjs.org/@esbuild/netbsd-x64/-/netbsd-x64-0.18.16.tgz", + "integrity": "sha512-uVAgpimx9Ffw3xowtg/7qQPwHFx94yCje+DoBx+LNm2ePDpQXHrzE+Sb0Si2VBObYz+LcRps15cq+95YM7gkUw==", + "dev": true, + "optional": true + }, + "@esbuild/openbsd-x64": { + "version": "0.18.16", + "resolved": "https://registry.npmjs.org/@esbuild/openbsd-x64/-/openbsd-x64-0.18.16.tgz", + "integrity": "sha512-6OjCQM9wf7z8/MBi6BOWaTL2AS/SZudsZtBziXMtNI8r/U41AxS9x7jn0ATOwVy08OotwkPqGRMkpPR2wcTJXA==", + "dev": true, + "optional": true + }, + "@esbuild/sunos-x64": { + "version": "0.18.16", + "resolved": "https://registry.npmjs.org/@esbuild/sunos-x64/-/sunos-x64-0.18.16.tgz", + "integrity": "sha512-ZoNkruFYJp9d1LbUYCh8awgQDvB9uOMZqlQ+gGEZR7v6C+N6u7vPr86c+Chih8niBR81Q/bHOSKGBK3brJyvkQ==", + "dev": true, + "optional": true + }, + "@esbuild/win32-arm64": { + "version": "0.18.16", + "resolved": "https://registry.npmjs.org/@esbuild/win32-arm64/-/win32-arm64-0.18.16.tgz", + "integrity": "sha512-+j4anzQ9hrs+iqO+/wa8UE6TVkKua1pXUb0XWFOx0FiAj6R9INJ+WE//1/Xo6FG1vB5EpH3ko+XcgwiDXTxcdw==", + "dev": true, + "optional": true + }, + "@esbuild/win32-ia32": { + "version": "0.18.16", + "resolved": "https://registry.npmjs.org/@esbuild/win32-ia32/-/win32-ia32-0.18.16.tgz", + "integrity": "sha512-5PFPmq3sSKTp9cT9dzvI67WNfRZGvEVctcZa1KGjDDu4n3H8k59Inbk0du1fz0KrAbKKNpJbdFXQMDUz7BG4rQ==", + "dev": true, + "optional": true + }, + "@esbuild/win32-x64": { + "version": "0.18.16", + "resolved": "https://registry.npmjs.org/@esbuild/win32-x64/-/win32-x64-0.18.16.tgz", + "integrity": "sha512-sCIVrrtcWN5Ua7jYXNG1xD199IalrbfV2+0k/2Zf2OyV2FtnQnMgdzgpRAbi4AWlKJj1jkX+M+fEGPQj6BQB4w==", + "dev": true, + "optional": true + }, + "@eslint-community/eslint-utils": { + "version": "4.4.0", + "resolved": "https://registry.npmjs.org/@eslint-community/eslint-utils/-/eslint-utils-4.4.0.tgz", + "integrity": "sha512-1/sA4dwrzBAyeUoQ6oxahHKmrZvsnLCg4RfxW3ZFGGmQkSNQPFNLV9CUEFQP1x9EYXHTo5p6xdhZM1Ne9p/AfA==", + "dev": true, + "requires": { + "eslint-visitor-keys": "^3.3.0" + } + }, + "@eslint-community/regexpp": { + "version": "4.6.0", + "resolved": "https://registry.npmjs.org/@eslint-community/regexpp/-/regexpp-4.6.0.tgz", + "integrity": "sha512-uiPeRISaglZnaZk8vwrjQZ1CxogZeY/4IYft6gBOTqu1WhVXWmCmZMWxUv2Q/pxSvPdp1JPaO62kLOcOkMqWrw==", + "dev": true + }, + "@eslint/eslintrc": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/@eslint/eslintrc/-/eslintrc-2.1.0.tgz", + "integrity": "sha512-Lj7DECXqIVCqnqjjHMPna4vn6GJcMgul/wuS0je9OZ9gsL0zzDpKPVtcG1HaDVc+9y+qgXneTeUMbCqXJNpH1A==", + "dev": true, + "requires": { + "ajv": "^6.12.4", + "debug": "^4.3.2", + "espree": "^9.6.0", + "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": { + "version": "8.44.0", + "resolved": "https://registry.npmjs.org/@eslint/js/-/js-8.44.0.tgz", + "integrity": "sha512-Ag+9YM4ocKQx9AarydN0KY2j0ErMHNIocPDrVo8zAE44xLTjEtz81OdR68/cydGtk6m6jDb5Za3r2useMzYmSw==", + "dev": true + }, + "@fullcalendar/core": { + "version": "6.1.8", + "resolved": "https://registry.npmjs.org/@fullcalendar/core/-/core-6.1.8.tgz", + "integrity": "sha512-i8JBIvZCWGO9dsMEDcx9bnsQZ9PtGSJdOXGgWbhLaGq2iq41OBdp9g9gM4b/Otv2oK8bL5Gl6CsMmb/HkDtA6Q==", + "requires": { + "preact": "~10.12.1" + } + }, + "@fullcalendar/daygrid": { + "version": "6.1.8", + "resolved": "https://registry.npmjs.org/@fullcalendar/daygrid/-/daygrid-6.1.8.tgz", + "integrity": "sha512-kCZxQFKb9Vqa3CZRX0v7rMSJ2mlTt4gDpyLfiNJKxUAq7W51uKurPaFZWicaXy1ESHVBxKNlbx5uNjBpyu50JQ==", + "requires": {} + }, + "@fullcalendar/interaction": { + "version": "6.1.8", + "resolved": "https://registry.npmjs.org/@fullcalendar/interaction/-/interaction-6.1.8.tgz", + "integrity": "sha512-r6W4E9ohaA87M2uPSlmpE2WT7Fzu7LN0u2pE6D/tThruCEaAPbN8Pw5+sqclsuyTIL09mg0eSJm/ggJekTabSA==", + "requires": {} + }, + "@fullcalendar/react": { + "version": "6.1.8", + "resolved": "https://registry.npmjs.org/@fullcalendar/react/-/react-6.1.8.tgz", + "integrity": "sha512-E8GQSQyZHkjpwxQW5Vci7iZgN7f33ntuRcvfGii4Fn35t9VHGz2SEyKAWXpVf38elcKTZKVgajU9ipStd+1LEg==", + "requires": {} + }, + "@humanwhocodes/config-array": { + "version": "0.11.10", + "resolved": "https://registry.npmjs.org/@humanwhocodes/config-array/-/config-array-0.11.10.tgz", + "integrity": "sha512-KVVjQmNUepDVGXNuoRRdmmEjruj0KfiGSbS8LVc12LMsWDQzRXJ0qdhN8L8uUigKpfEHRhlaQFY0ib1tnUbNeQ==", + "dev": true, + "requires": { + "@humanwhocodes/object-schema": "^1.2.1", + "debug": "^4.1.1", + "minimatch": "^3.0.5" + } + }, + "@humanwhocodes/module-importer": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/@humanwhocodes/module-importer/-/module-importer-1.0.1.tgz", + "integrity": "sha512-bxveV4V8v5Yb4ncFTT3rPSgZBOpCkjfK0y4oVVVJwIuDVBRMDXrPyXRL988i5ap9m9bnyEEjWfm5WkBmtffLfA==", + "dev": true + }, + "@humanwhocodes/object-schema": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/@humanwhocodes/object-schema/-/object-schema-1.2.1.tgz", + "integrity": "sha512-ZnQMnLV4e7hDlUvw8H+U8ASL02SS2Gn6+9Ac3wGGLIe7+je2AeAOxPY+izIPJDfFDb7eDjev0Us8MO1iFRN8hA==", + "dev": true + }, + "@nodelib/fs.scandir": { + "version": "2.1.5", + "resolved": "https://registry.npmjs.org/@nodelib/fs.scandir/-/fs.scandir-2.1.5.tgz", + "integrity": "sha512-vq24Bq3ym5HEQm2NKCr3yXDwjc7vTsEThRDnkp2DK9p1uqLR+DHurm/NOTo0KG7HYHU7eppKZj3MyqYuMBf62g==", + "dev": true, + "requires": { + "@nodelib/fs.stat": "2.0.5", + "run-parallel": "^1.1.9" + } + }, + "@nodelib/fs.stat": { + "version": "2.0.5", + "resolved": "https://registry.npmjs.org/@nodelib/fs.stat/-/fs.stat-2.0.5.tgz", + "integrity": "sha512-RkhPPp2zrqDAQA/2jNhnztcPAlv64XdhIp7a7454A5ovI7Bukxgt7MX7udwAu3zg1DcpPU0rz3VV1SeaqvY4+A==", + "dev": true + }, + "@nodelib/fs.walk": { + "version": "1.2.8", + "resolved": "https://registry.npmjs.org/@nodelib/fs.walk/-/fs.walk-1.2.8.tgz", + "integrity": "sha512-oGB+UxlgWcgQkgwo8GcEGwemoTFt3FIO9ababBmaGwXIoBKZ+GTy0pP185beGg7Llih/NSHSV2XAs1lnznocSg==", + "dev": true, + "requires": { + "@nodelib/fs.scandir": "2.1.5", + "fastq": "^1.6.0" + } + }, + "@rc-component/color-picker": { + "version": "1.4.1", + "resolved": "https://registry.npmjs.org/@rc-component/color-picker/-/color-picker-1.4.1.tgz", + "integrity": "sha512-vh5EWqnsayZa/JwUznqDaPJz39jznx/YDbyBuVJntv735tKXKwEUZZb2jYEldOg+NKWZwtALjGMrNeGBmqFoEw==", + "requires": { + "@babel/runtime": "^7.10.1", + "@ctrl/tinycolor": "^3.6.0", + "classnames": "^2.2.6", + "rc-util": "^5.30.0" + } + }, + "@rc-component/context": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/@rc-component/context/-/context-1.3.0.tgz", + "integrity": "sha512-6QdaCJ7Wn5UZLJs15IEfqy4Ru3OaL5ctqpQYWd5rlfV9wwzrzdt6+kgAQZV/qdB0MUPN4nhyBfRembQCIvBf+w==", + "requires": { + "@babel/runtime": "^7.10.1", + "rc-util": "^5.27.0" + } + }, + "@rc-component/mini-decimal": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/@rc-component/mini-decimal/-/mini-decimal-1.1.0.tgz", + "integrity": "sha512-jS4E7T9Li2GuYwI6PyiVXmxTiM6b07rlD9Ge8uGZSCz3WlzcG5ZK7g5bbuKNeZ9pgUuPK/5guV781ujdVpm4HQ==", + "requires": { + "@babel/runtime": "^7.18.0" + } + }, + "@rc-component/mutate-observer": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/@rc-component/mutate-observer/-/mutate-observer-1.0.0.tgz", + "integrity": "sha512-okqRJSfNisXdI6CUeOLZC5ukBW/8kir2Ii4PJiKpUt+3+uS7dxwJUMxsUZquxA1rQuL8YcEmKVp/TCnR+yUdZA==", + "requires": { + "@babel/runtime": "^7.18.0", + "classnames": "^2.3.2", + "rc-util": "^5.24.4" + } + }, + "@rc-component/portal": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/@rc-component/portal/-/portal-1.1.2.tgz", + "integrity": "sha512-6f813C0IsasTZms08kfA8kPAGxbbkYToa8ALaiDIGGECU4i9hj8Plgbx0sNJDrey3EtHO30hmdaxtT0138xZcg==", + "requires": { + "@babel/runtime": "^7.18.0", + "classnames": "^2.3.2", + "rc-util": "^5.24.4" + } + }, + "@rc-component/tour": { + "version": "1.8.1", + "resolved": "https://registry.npmjs.org/@rc-component/tour/-/tour-1.8.1.tgz", + "integrity": "sha512-CsrQnfKgNArxx2j1RNHVLZgVA+rLrEj06lIsl4KSynMqADsqz8eKvVkr0F3p9PA10948M6WEEZt5a/FGAbGR2A==", + "requires": { + "@babel/runtime": "^7.18.0", + "@rc-component/portal": "^1.0.0-9", + "@rc-component/trigger": "^1.3.6", + "classnames": "^2.3.2", + "rc-util": "^5.24.4" + } + }, + "@rc-component/trigger": { + "version": "1.14.3", + "resolved": "https://registry.npmjs.org/@rc-component/trigger/-/trigger-1.14.3.tgz", + "integrity": "sha512-kIPL8srsAb3RWGWicLZyTMS/6IM1bmjcaqXS3GfgIe6EoPBxzGRx6NAvpDJg198zDSZ/RXGFLrqguCItnaN/Aw==", + "requires": { + "@babel/runtime": "^7.18.3", + "@rc-component/portal": "^1.1.0", + "classnames": "^2.3.2", + "rc-align": "^4.0.0", + "rc-motion": "^2.0.0", + "rc-resize-observer": "^1.3.1", + "rc-util": "^5.33.0" + } + }, + "@remix-run/router": { + "version": "1.7.2", + "resolved": "https://registry.npmjs.org/@remix-run/router/-/router-1.7.2.tgz", + "integrity": "sha512-7Lcn7IqGMV+vizMPoEl5F0XDshcdDYtMI6uJLQdQz5CfZAwy3vvGKYSUk789qndt5dEC4HfSjviSYlSoHGL2+A==" + }, + "@swc/core": { + "version": "1.3.70", + "resolved": "https://registry.npmjs.org/@swc/core/-/core-1.3.70.tgz", + "integrity": "sha512-LWVWlEDLlOD25PvA2NEz41UzdwXnlDyBiZbe69s3zM0DfCPwZXLUm79uSqH9ItsOjTrXSL5/1+XUL6C/BZwChA==", + "dev": true, + "requires": { + "@swc/core-darwin-arm64": "1.3.70", + "@swc/core-darwin-x64": "1.3.70", + "@swc/core-linux-arm-gnueabihf": "1.3.70", + "@swc/core-linux-arm64-gnu": "1.3.70", + "@swc/core-linux-arm64-musl": "1.3.70", + "@swc/core-linux-x64-gnu": "1.3.70", + "@swc/core-linux-x64-musl": "1.3.70", + "@swc/core-win32-arm64-msvc": "1.3.70", + "@swc/core-win32-ia32-msvc": "1.3.70", + "@swc/core-win32-x64-msvc": "1.3.70" + } + }, + "@swc/core-darwin-arm64": { + "version": "1.3.70", + "resolved": "https://registry.npmjs.org/@swc/core-darwin-arm64/-/core-darwin-arm64-1.3.70.tgz", + "integrity": "sha512-31+mcl0dgdRHvZRjhLOK9V6B+qJ7nxDZYINr9pBlqGWxknz37Vld5KK19Kpr79r0dXUZvaaelLjCnJk9dA2PcQ==", + "dev": true, + "optional": true + }, + "@swc/core-darwin-x64": { + "version": "1.3.70", + "resolved": "https://registry.npmjs.org/@swc/core-darwin-x64/-/core-darwin-x64-1.3.70.tgz", + "integrity": "sha512-GMFJ65E18zQC80t0os+TZvI+8lbRuitncWVge/RXmXbVLPRcdykP4EJ87cqzcG5Ah0z18/E0T+ixD6jHRisrYQ==", + "dev": true, + "optional": true + }, + "@swc/core-linux-arm-gnueabihf": { + "version": "1.3.70", + "resolved": "https://registry.npmjs.org/@swc/core-linux-arm-gnueabihf/-/core-linux-arm-gnueabihf-1.3.70.tgz", + "integrity": "sha512-wjhCwS8LCiAq2VedF1b4Bryyw68xZnfMED4pLRazAl8BaUlDFANfRBORNunxlfHQj4V3x39IaiLgCZRHMdzXBg==", + "dev": true, + "optional": true + }, + "@swc/core-linux-arm64-gnu": { + "version": "1.3.70", + "resolved": "https://registry.npmjs.org/@swc/core-linux-arm64-gnu/-/core-linux-arm64-gnu-1.3.70.tgz", + "integrity": "sha512-9D/Rx67cAOnMiexvCqARxvhj7coRajTp5HlJHuf+rfwMqI2hLhpO9/pBMQxBUAWxODO/ksQ/OF+GJRjmtWw/2A==", + "dev": true, + "optional": true + }, + "@swc/core-linux-arm64-musl": { + "version": "1.3.70", + "resolved": "https://registry.npmjs.org/@swc/core-linux-arm64-musl/-/core-linux-arm64-musl-1.3.70.tgz", + "integrity": "sha512-gkjxBio7XD+1GlQVVyPP/qeFkLu83VhRHXaUrkNYpr5UZG9zZurBERT9nkS6Y+ouYh+Q9xmw57aIyd2KvD2zqQ==", + "dev": true, + "optional": true + }, + "@swc/core-linux-x64-gnu": { + "version": "1.3.70", + "resolved": "https://registry.npmjs.org/@swc/core-linux-x64-gnu/-/core-linux-x64-gnu-1.3.70.tgz", + "integrity": "sha512-/nCly+V4xfMVwfEUoLLAukxUSot/RcSzsf6GdsGTjFcrp5sZIntAjokYRytm3VT1c2TK321AfBorsi9R5w8Y7Q==", + "dev": true, + "optional": true + }, + "@swc/core-linux-x64-musl": { + "version": "1.3.70", + "resolved": "https://registry.npmjs.org/@swc/core-linux-x64-musl/-/core-linux-x64-musl-1.3.70.tgz", + "integrity": "sha512-HoOsPJbt361KGKaivAK0qIiYARkhzlxeAfvF5NlnKxkIMOZpQ46Lwj3tR0VWohKbrhS+cYKFlVuDi5XnDkx0XA==", + "dev": true, + "optional": true + }, + "@swc/core-win32-arm64-msvc": { + "version": "1.3.70", + "resolved": "https://registry.npmjs.org/@swc/core-win32-arm64-msvc/-/core-win32-arm64-msvc-1.3.70.tgz", + "integrity": "sha512-hm4IBK/IaRil+aj1cWU6f0GyAdHpw/Jr5nyFYLM2c/tt7w2t5hgb8NjzM2iM84lOClrig1fG6edj2vCF1dFzNQ==", + "dev": true, + "optional": true + }, + "@swc/core-win32-ia32-msvc": { + "version": "1.3.70", + "resolved": "https://registry.npmjs.org/@swc/core-win32-ia32-msvc/-/core-win32-ia32-msvc-1.3.70.tgz", + "integrity": "sha512-5cgKUKIT/9Fp5fCA+zIjYCQ4dSvjFYOeWGZR3QiTXGkC4bGa1Ji9SEPyeIAX0iruUnKjYaZB9RvHK2tNn7RLrQ==", + "dev": true, + "optional": true + }, + "@swc/core-win32-x64-msvc": { + "version": "1.3.70", + "resolved": "https://registry.npmjs.org/@swc/core-win32-x64-msvc/-/core-win32-x64-msvc-1.3.70.tgz", + "integrity": "sha512-LE8lW46+TQBzVkn2mHBlk8DIElPIZ2dO5P8AbJiARNBAnlqQWu67l9gWM89UiZ2l33J2cI37pHzON3tKnT8f9g==", + "dev": true, + "optional": true + }, + "@types/json-schema": { + "version": "7.0.12", + "resolved": "https://registry.npmjs.org/@types/json-schema/-/json-schema-7.0.12.tgz", + "integrity": "sha512-Hr5Jfhc9eYOQNPYO5WLDq/n4jqijdHNlDXjuAQkkt+mWdQR+XJToOHrsD4cPaMXpn6KO7y2+wM8AZEs8VpBLVA==", + "dev": true + }, + "@types/node": { + "version": "20.4.4", + "resolved": "https://registry.npmjs.org/@types/node/-/node-20.4.4.tgz", + "integrity": "sha512-CukZhumInROvLq3+b5gLev+vgpsIqC2D0deQr/yS1WnxvmYLlJXZpaQrQiseMY+6xusl79E04UjWoqyr+t1/Ew==", + "dev": true + }, + "@types/prop-types": { + "version": "15.7.5", + "resolved": "https://registry.npmjs.org/@types/prop-types/-/prop-types-15.7.5.tgz", + "integrity": "sha512-JCB8C6SnDoQf0cNycqd/35A7MjcnK+ZTqE7judS6o7utxUCg6imJg3QK2qzHKszlTjcj2cn+NwMB2i96ubpj7w==", + "dev": true + }, + "@types/react": { + "version": "18.2.15", + "resolved": "https://registry.npmjs.org/@types/react/-/react-18.2.15.tgz", + "integrity": "sha512-oEjE7TQt1fFTFSbf8kkNuc798ahTUzn3Le67/PWjE8MAfYAD/qB7O8hSTcromLFqHCt9bcdOg5GXMokzTjJ5SA==", + "dev": true, + "requires": { + "@types/prop-types": "*", + "@types/scheduler": "*", + "csstype": "^3.0.2" + } + }, + "@types/react-dom": { + "version": "18.2.7", + "resolved": "https://registry.npmjs.org/@types/react-dom/-/react-dom-18.2.7.tgz", + "integrity": "sha512-GRaAEriuT4zp9N4p1i8BDBYmEyfo+xQ3yHjJU4eiK5NDa1RmUZG+unZABUTK4/Ox/M+GaHwb6Ow8rUITrtjszA==", + "dev": true, + "requires": { + "@types/react": "*" + } + }, + "@types/scheduler": { + "version": "0.16.3", + "resolved": "https://registry.npmjs.org/@types/scheduler/-/scheduler-0.16.3.tgz", + "integrity": "sha512-5cJ8CB4yAx7BH1oMvdU0Jh9lrEXyPkar6F9G/ERswkCuvP4KQZfZkSjcMbAICCpQTN4OuZn8tz0HiKv9TGZgrQ==", + "dev": true + }, + "@types/semver": { + "version": "7.5.0", + "resolved": "https://registry.npmjs.org/@types/semver/-/semver-7.5.0.tgz", + "integrity": "sha512-G8hZ6XJiHnuhQKR7ZmysCeJWE08o8T0AXtk5darsCaTVsYZhhgUrq53jizaR2FvsoeCwJhlmwTjkXBY5Pn/ZHw==", + "dev": true + }, + "@typescript-eslint/eslint-plugin": { + "version": "6.1.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/eslint-plugin/-/eslint-plugin-6.1.0.tgz", + "integrity": "sha512-qg7Bm5TyP/I7iilGyp6DRqqkt8na00lI6HbjWZObgk3FFSzH5ypRwAHXJhJkwiRtTcfn+xYQIMOR5kJgpo6upw==", + "dev": true, + "requires": { + "@eslint-community/regexpp": "^4.5.1", + "@typescript-eslint/scope-manager": "6.1.0", + "@typescript-eslint/type-utils": "6.1.0", + "@typescript-eslint/utils": "6.1.0", + "@typescript-eslint/visitor-keys": "6.1.0", + "debug": "^4.3.4", + "graphemer": "^1.4.0", + "ignore": "^5.2.4", + "natural-compare": "^1.4.0", + "natural-compare-lite": "^1.4.0", + "semver": "^7.5.4", + "ts-api-utils": "^1.0.1" + } + }, + "@typescript-eslint/parser": { + "version": "6.1.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/parser/-/parser-6.1.0.tgz", + "integrity": "sha512-hIzCPvX4vDs4qL07SYzyomamcs2/tQYXg5DtdAfj35AyJ5PIUqhsLf4YrEIFzZcND7R2E8tpQIZKayxg8/6Wbw==", + "dev": true, + "requires": { + "@typescript-eslint/scope-manager": "6.1.0", + "@typescript-eslint/types": "6.1.0", + "@typescript-eslint/typescript-estree": "6.1.0", + "@typescript-eslint/visitor-keys": "6.1.0", + "debug": "^4.3.4" + } + }, + "@typescript-eslint/scope-manager": { + "version": "6.1.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/scope-manager/-/scope-manager-6.1.0.tgz", + "integrity": "sha512-AxjgxDn27hgPpe2rQe19k0tXw84YCOsjDJ2r61cIebq1t+AIxbgiXKvD4999Wk49GVaAcdJ/d49FYel+Pp3jjw==", + "dev": true, + "requires": { + "@typescript-eslint/types": "6.1.0", + "@typescript-eslint/visitor-keys": "6.1.0" + } + }, + "@typescript-eslint/type-utils": { + "version": "6.1.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/type-utils/-/type-utils-6.1.0.tgz", + "integrity": "sha512-kFXBx6QWS1ZZ5Ni89TyT1X9Ag6RXVIVhqDs0vZE/jUeWlBv/ixq2diua6G7ece6+fXw3TvNRxP77/5mOMusx2w==", + "dev": true, + "requires": { + "@typescript-eslint/typescript-estree": "6.1.0", + "@typescript-eslint/utils": "6.1.0", + "debug": "^4.3.4", + "ts-api-utils": "^1.0.1" + } + }, + "@typescript-eslint/types": { + "version": "6.1.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/types/-/types-6.1.0.tgz", + "integrity": "sha512-+Gfd5NHCpDoHDOaU/yIF3WWRI2PcBRKKpP91ZcVbL0t5tQpqYWBs3z/GGhvU+EV1D0262g9XCnyqQh19prU0JQ==", + "dev": true + }, + "@typescript-eslint/typescript-estree": { + "version": "6.1.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/typescript-estree/-/typescript-estree-6.1.0.tgz", + "integrity": "sha512-nUKAPWOaP/tQjU1IQw9sOPCDavs/iU5iYLiY/6u7gxS7oKQoi4aUxXS1nrrVGTyBBaGesjkcwwHkbkiD5eBvcg==", + "dev": true, + "requires": { + "@typescript-eslint/types": "6.1.0", + "@typescript-eslint/visitor-keys": "6.1.0", + "debug": "^4.3.4", + "globby": "^11.1.0", + "is-glob": "^4.0.3", + "semver": "^7.5.4", + "ts-api-utils": "^1.0.1" + } + }, + "@typescript-eslint/utils": { + "version": "6.1.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/utils/-/utils-6.1.0.tgz", + "integrity": "sha512-wp652EogZlKmQoMS5hAvWqRKplXvkuOnNzZSE0PVvsKjpexd/XznRVHAtrfHFYmqaJz0DFkjlDsGYC9OXw+OhQ==", + "dev": true, + "requires": { + "@eslint-community/eslint-utils": "^4.4.0", + "@types/json-schema": "^7.0.12", + "@types/semver": "^7.5.0", + "@typescript-eslint/scope-manager": "6.1.0", + "@typescript-eslint/types": "6.1.0", + "@typescript-eslint/typescript-estree": "6.1.0", + "semver": "^7.5.4" + } + }, + "@typescript-eslint/visitor-keys": { + "version": "6.1.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/visitor-keys/-/visitor-keys-6.1.0.tgz", + "integrity": "sha512-yQeh+EXhquh119Eis4k0kYhj9vmFzNpbhM3LftWQVwqVjipCkwHBQOZutcYW+JVkjtTG9k8nrZU1UoNedPDd1A==", + "dev": true, + "requires": { + "@typescript-eslint/types": "6.1.0", + "eslint-visitor-keys": "^3.4.1" + } + }, + "@vitejs/plugin-react-swc": { + "version": "3.3.2", + "resolved": "https://registry.npmjs.org/@vitejs/plugin-react-swc/-/plugin-react-swc-3.3.2.tgz", + "integrity": "sha512-VJFWY5sfoZerQRvJrh518h3AcQt6f/yTuWn4/TRB+dqmYU0NX1qz7qM5Wfd+gOQqUzQW4gxKqKN3KpE/P3+zrA==", + "dev": true, + "requires": { + "@swc/core": "^1.3.61" + } + }, + "acorn": { + "version": "8.10.0", + "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.10.0.tgz", + "integrity": "sha512-F0SAmZ8iUtS//m8DmCTA0jlh6TDKkHQyK6xc6V4KDTyZKA9dnvX9/3sRTVQrWm79glUAZbnmmNcdYwUIHWVybw==", + "dev": true + }, + "acorn-jsx": { + "version": "5.3.2", + "resolved": "https://registry.npmjs.org/acorn-jsx/-/acorn-jsx-5.3.2.tgz", + "integrity": "sha512-rq9s+JNhf0IChjtDXxllJ7g41oZk5SlXtp0LHwyA5cejwn7vKmKp4pPri6YEePv2PU65sAsegbXtIinmDFDXgQ==", + "dev": true, + "requires": {} + }, + "ajv": { + "version": "6.12.6", + "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.12.6.tgz", + "integrity": "sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g==", + "dev": true, + "requires": { + "fast-deep-equal": "^3.1.1", + "fast-json-stable-stringify": "^2.0.0", + "json-schema-traverse": "^0.4.1", + "uri-js": "^4.2.2" + } + }, + "ansi-regex": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz", + "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==", + "dev": true + }, + "ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "requires": { + "color-convert": "^2.0.1" + } + }, + "antd": { + "version": "5.7.3", + "resolved": "https://registry.npmjs.org/antd/-/antd-5.7.3.tgz", + "integrity": "sha512-7sQeE86XkUrYDIKGu/Qu7kl+NWYzkVSGbGqWGIbITHkFZorCyOvvqgF63fiWo/tp2lZWbEOO0Cm7IiYnoeWh9A==", + "requires": { + "@ant-design/colors": "^7.0.0", + "@ant-design/cssinjs": "^1.13.2", + "@ant-design/icons": "^5.1.0", + "@ant-design/react-slick": "~1.0.0", + "@babel/runtime": "^7.18.3", + "@ctrl/tinycolor": "^3.6.0", + "@rc-component/color-picker": "~1.4.0", + "@rc-component/mutate-observer": "^1.0.0", + "@rc-component/tour": "~1.8.1", + "@rc-component/trigger": "^1.13.0", + "classnames": "^2.2.6", + "copy-to-clipboard": "^3.2.0", + "dayjs": "^1.11.1", + "qrcode.react": "^3.1.0", + "rc-cascader": "~3.12.0", + "rc-checkbox": "~3.1.0", + "rc-collapse": "~3.7.0", + "rc-dialog": "~9.1.0", + "rc-drawer": "~6.2.0", + "rc-dropdown": "~4.1.0", + "rc-field-form": "~1.34.0", + "rc-image": "~7.0.0", + "rc-input": "~1.1.0", + "rc-input-number": "~8.0.2", + "rc-mentions": "~2.5.0", + "rc-menu": "~9.10.0", + "rc-motion": "^2.7.3", + "rc-notification": "~5.0.4", + "rc-pagination": "~3.5.0", + "rc-picker": "~3.10.0", + "rc-progress": "~3.4.1", + "rc-rate": "~2.12.0", + "rc-resize-observer": "^1.2.0", + "rc-segmented": "~2.2.0", + "rc-select": "~14.5.0", + "rc-slider": "~10.1.0", + "rc-steps": "~6.0.1", + "rc-switch": "~4.1.0", + "rc-table": "~7.32.1", + "rc-tabs": "~12.9.0", + "rc-textarea": "~1.3.3", + "rc-tooltip": "~6.0.0", + "rc-tree": "~5.7.6", + "rc-tree-select": "~5.9.0", + "rc-upload": "~4.3.0", + "rc-util": "^5.32.0", + "scroll-into-view-if-needed": "^3.0.3", + "throttle-debounce": "^5.0.0" + } + }, + "argparse": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/argparse/-/argparse-2.0.1.tgz", + "integrity": "sha512-8+9WqebbFzpX9OR+Wa6O29asIogeRMzcGtAINdpMHHyAg10f05aSFVBbcEqGf/PXw1EjAZ+q2/bEBg3DvurK3Q==", + "dev": true + }, + "array-tree-filter": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/array-tree-filter/-/array-tree-filter-2.1.0.tgz", + "integrity": "sha512-4ROwICNlNw/Hqa9v+rk5h22KjmzB1JGTMVKP2AKJBOCgb0yL0ASf0+YvCcLNNwquOHNX48jkeZIJ3a+oOQqKcw==" + }, + "array-union": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/array-union/-/array-union-2.1.0.tgz", + "integrity": "sha512-HGyxoOTYUyCM6stUe6EJgnd4EoewAI7zMdfqO+kGjnlZmBDz/cR5pf8r/cR4Wq60sL/p0IkcjUEEPwS3GFrIyw==", + "dev": true + }, + "async-validator": { + "version": "4.2.5", + "resolved": "https://registry.npmjs.org/async-validator/-/async-validator-4.2.5.tgz", + "integrity": "sha512-7HhHjtERjqlNbZtqNqy2rckN/SpOOlmDliet+lP7k+eKZEjPk3DgyeU9lIXLdeLz0uBbbVp+9Qdow9wJWgwwfg==" + }, + "asynckit": { + "version": "0.4.0", + "resolved": "https://registry.npmjs.org/asynckit/-/asynckit-0.4.0.tgz", + "integrity": "sha512-Oei9OH4tRh0YqU3GxhX79dM/mwVgvbZJaSNaRk+bshkj0S5cfHcgYakreBjrHwatXKbz+IoIdYLxrKim2MjW0Q==" + }, + "axios": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/axios/-/axios-1.4.0.tgz", + "integrity": "sha512-S4XCWMEmzvo64T9GfvQDOXgYRDJ/wsSZc7Jvdgx5u1sd0JwsuPLqb3SYmusag+edF6ziyMensPVqLTSc1PiSEA==", + "requires": { + "follow-redirects": "^1.15.0", + "form-data": "^4.0.0", + "proxy-from-env": "^1.1.0" + } + }, + "balanced-match": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz", + "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==", + "dev": true + }, + "brace-expansion": { + "version": "1.1.11", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz", + "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==", + "dev": true, + "requires": { + "balanced-match": "^1.0.0", + "concat-map": "0.0.1" + } + }, + "braces": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.2.tgz", + "integrity": "sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==", + "dev": true, + "requires": { + "fill-range": "^7.0.1" + } + }, + "callsites": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/callsites/-/callsites-3.1.0.tgz", + "integrity": "sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ==", + "dev": true + }, + "chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "requires": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + } + }, + "classnames": { + "version": "2.3.2", + "resolved": "https://registry.npmjs.org/classnames/-/classnames-2.3.2.tgz", + "integrity": "sha512-CSbhY4cFEJRe6/GQzIk5qXZ4Jeg5pcsP7b5peFSDpffpe1cqjASH/n9UTjBwOp6XpMSTwQ8Za2K5V02ueA7Tmw==" + }, + "color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "requires": { + "color-name": "~1.1.4" + } + }, + "color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true + }, + "combined-stream": { + "version": "1.0.8", + "resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.8.tgz", + "integrity": "sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==", + "requires": { + "delayed-stream": "~1.0.0" + } + }, + "compute-scroll-into-view": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/compute-scroll-into-view/-/compute-scroll-into-view-3.0.3.tgz", + "integrity": "sha512-nadqwNxghAGTamwIqQSG433W6OADZx2vCo3UXHNrzTRHK/htu+7+L0zhjEoaeaQVNAi3YgqWDv8+tzf0hRfR+A==" + }, + "concat-map": { + "version": "0.0.1", + "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", + "integrity": "sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg==", + "dev": true + }, + "copy-to-clipboard": { + "version": "3.3.3", + "resolved": "https://registry.npmjs.org/copy-to-clipboard/-/copy-to-clipboard-3.3.3.tgz", + "integrity": "sha512-2KV8NhB5JqC3ky0r9PMCAZKbUHSwtEo4CwCs0KXgruG43gX5PMqDEBbVU4OUzw2MuAWUfsuFmWvEKG5QRfSnJA==", + "requires": { + "toggle-selection": "^1.0.6" + } + }, + "cross-spawn": { + "version": "7.0.3", + "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.3.tgz", + "integrity": "sha512-iRDPJKUPVEND7dHPO8rkbOnPpyDygcDFtWjpeWNCgy8WP2rXcxXL8TskReQl6OrB2G7+UJrags1q15Fudc7G6w==", + "dev": true, + "requires": { + "path-key": "^3.1.0", + "shebang-command": "^2.0.0", + "which": "^2.0.1" + } + }, + "csstype": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/csstype/-/csstype-3.1.2.tgz", + "integrity": "sha512-I7K1Uu0MBPzaFKg4nI5Q7Vs2t+3gWWW648spaF+Rg7pI9ds18Ugn+lvg4SHczUdKlHI5LWBXyqfS8+DufyBsgQ==" + }, + "dayjs": { + "version": "1.11.9", + "resolved": "https://registry.npmjs.org/dayjs/-/dayjs-1.11.9.tgz", + "integrity": "sha512-QvzAURSbQ0pKdIye2txOzNaHmxtUBXerpY0FJsFXUMKbIZeFm5ht1LS/jFsrncjnmtv8HsG0W2g6c0zUjZWmpA==" + }, + "debug": { + "version": "4.3.4", + "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.4.tgz", + "integrity": "sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ==", + "dev": true, + "requires": { + "ms": "2.1.2" + } + }, + "deep-is": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/deep-is/-/deep-is-0.1.4.tgz", + "integrity": "sha512-oIPzksmTg4/MriiaYGO+okXDT7ztn/w3Eptv/+gSIdMdKsJo0u4CfYNFJPy+4SKMuCqGw2wxnA+URMg3t8a/bQ==", + "dev": true + }, + "delayed-stream": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz", + "integrity": "sha512-ZySD7Nf91aLB0RxL4KGrKHBXl7Eds1DAmEdcoVawXnLD7SDhpNgtuII2aAkg7a7QS41jxPSZ17p4VdGnMHk3MQ==" + }, + "dir-glob": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/dir-glob/-/dir-glob-3.0.1.tgz", + "integrity": "sha512-WkrWp9GR4KXfKGYzOLmTuGVi1UWFfws377n9cc55/tb6DuqyF6pcQ5AbiHEshaDpY9v6oaSr2XCDidGmMwdzIA==", + "dev": true, + "requires": { + "path-type": "^4.0.0" + } + }, + "doctrine": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/doctrine/-/doctrine-3.0.0.tgz", + "integrity": "sha512-yS+Q5i3hBf7GBkd4KG8a7eBNNWNGLTaEwwYWUijIYM7zrlYDM0BFXHjjPWlWZ1Rg7UaddZeIDmi9jF3HmqiQ2w==", + "dev": true, + "requires": { + "esutils": "^2.0.2" + } + }, + "dom-align": { + "version": "1.12.4", + "resolved": "https://registry.npmjs.org/dom-align/-/dom-align-1.12.4.tgz", + "integrity": "sha512-R8LUSEay/68zE5c8/3BDxiTEvgb4xZTF0RKmAHfiEVN3klfIpXfi2/QCoiWPccVQ0J/ZGdz9OjzL4uJEP/MRAw==" + }, + "dotenv": { + "version": "16.3.1", + "resolved": "https://registry.npmjs.org/dotenv/-/dotenv-16.3.1.tgz", + "integrity": "sha512-IPzF4w4/Rd94bA9imS68tZBaYyBWSCE47V1RGuMrB94iyTOIEwRmVL2x/4An+6mETpLrKJ5hQkB8W4kFAadeIQ==", + "dev": true + }, + "esbuild": { + "version": "0.18.16", + "resolved": "https://registry.npmjs.org/esbuild/-/esbuild-0.18.16.tgz", + "integrity": "sha512-1xLsOXrDqwdHxyXb/x/SOyg59jpf/SH7YMvU5RNSU7z3TInaASNJWNFJ6iRvLvLETZMasF3d1DdZLg7sgRimRQ==", + "dev": true, + "requires": { + "@esbuild/android-arm": "0.18.16", + "@esbuild/android-arm64": "0.18.16", + "@esbuild/android-x64": "0.18.16", + "@esbuild/darwin-arm64": "0.18.16", + "@esbuild/darwin-x64": "0.18.16", + "@esbuild/freebsd-arm64": "0.18.16", + "@esbuild/freebsd-x64": "0.18.16", + "@esbuild/linux-arm": "0.18.16", + "@esbuild/linux-arm64": "0.18.16", + "@esbuild/linux-ia32": "0.18.16", + "@esbuild/linux-loong64": "0.18.16", + "@esbuild/linux-mips64el": "0.18.16", + "@esbuild/linux-ppc64": "0.18.16", + "@esbuild/linux-riscv64": "0.18.16", + "@esbuild/linux-s390x": "0.18.16", + "@esbuild/linux-x64": "0.18.16", + "@esbuild/netbsd-x64": "0.18.16", + "@esbuild/openbsd-x64": "0.18.16", + "@esbuild/sunos-x64": "0.18.16", + "@esbuild/win32-arm64": "0.18.16", + "@esbuild/win32-ia32": "0.18.16", + "@esbuild/win32-x64": "0.18.16" + } + }, + "escape-string-regexp": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-4.0.0.tgz", + "integrity": "sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA==", + "dev": true + }, + "eslint": { + "version": "8.45.0", + "resolved": "https://registry.npmjs.org/eslint/-/eslint-8.45.0.tgz", + "integrity": "sha512-pd8KSxiQpdYRfYa9Wufvdoct3ZPQQuVuU5O6scNgMuOMYuxvH0IGaYK0wUFjo4UYYQQCUndlXiMbnxopwvvTiw==", + "dev": true, + "requires": { + "@eslint-community/eslint-utils": "^4.2.0", + "@eslint-community/regexpp": "^4.4.0", + "@eslint/eslintrc": "^2.1.0", + "@eslint/js": "8.44.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.6.0", + "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", + "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.3", + "strip-ansi": "^6.0.1", + "text-table": "^0.2.0" + } + }, + "eslint-plugin-react-hooks": { + "version": "4.6.0", + "resolved": "https://registry.npmjs.org/eslint-plugin-react-hooks/-/eslint-plugin-react-hooks-4.6.0.tgz", + "integrity": "sha512-oFc7Itz9Qxh2x4gNHStv3BqJq54ExXmfC+a1NjAta66IAN87Wu0R/QArgIS9qKzX3dXKPI9H5crl9QchNMY9+g==", + "dev": true, + "requires": {} + }, + "eslint-plugin-react-refresh": { + "version": "0.4.3", + "resolved": "https://registry.npmjs.org/eslint-plugin-react-refresh/-/eslint-plugin-react-refresh-0.4.3.tgz", + "integrity": "sha512-Hh0wv8bUNY877+sI0BlCUlsS0TYYQqvzEwJsJJPM2WF4RnTStSnSR3zdJYa2nPOJgg3UghXi54lVyMSmpCalzA==", + "dev": true, + "requires": {} + }, + "eslint-scope": { + "version": "7.2.1", + "resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-7.2.1.tgz", + "integrity": "sha512-CvefSOsDdaYYvxChovdrPo/ZGt8d5lrJWleAc1diXRKhHGiTYEI26cvo8Kle/wGnsizoCJjK73FMg1/IkIwiNA==", + "dev": true, + "requires": { + "esrecurse": "^4.3.0", + "estraverse": "^5.2.0" + } + }, + "eslint-visitor-keys": { + "version": "3.4.1", + "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-3.4.1.tgz", + "integrity": "sha512-pZnmmLwYzf+kWaM/Qgrvpen51upAktaaiI01nsJD/Yr3lMOdNtq0cxkrrg16w64VtisN6okbs7Q8AfGqj4c9fA==", + "dev": true + }, + "espree": { + "version": "9.6.1", + "resolved": "https://registry.npmjs.org/espree/-/espree-9.6.1.tgz", + "integrity": "sha512-oruZaFkjorTpF32kDSI5/75ViwGeZginGGy2NoOSg3Q9bnwlnmDm4HLnkl0RE3n+njDXR037aY1+x58Z/zFdwQ==", + "dev": true, + "requires": { + "acorn": "^8.9.0", + "acorn-jsx": "^5.3.2", + "eslint-visitor-keys": "^3.4.1" + } + }, + "esquery": { + "version": "1.5.0", + "resolved": "https://registry.npmjs.org/esquery/-/esquery-1.5.0.tgz", + "integrity": "sha512-YQLXUplAwJgCydQ78IMJywZCceoqk1oH01OERdSAJc/7U2AylwjhSCLDEtqwg811idIS/9fIU5GjG73IgjKMVg==", + "dev": true, + "requires": { + "estraverse": "^5.1.0" + } + }, + "esrecurse": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/esrecurse/-/esrecurse-4.3.0.tgz", + "integrity": "sha512-KmfKL3b6G+RXvP8N1vr3Tq1kL/oCFgn2NYXEtqP8/L3pKapUA4G8cFVaoF3SU323CD4XypR/ffioHmkti6/Tag==", + "dev": true, + "requires": { + "estraverse": "^5.2.0" + } + }, + "estraverse": { + "version": "5.3.0", + "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-5.3.0.tgz", + "integrity": "sha512-MMdARuVEQziNTeJD8DgMqmhwR11BRQ/cBP+pLtYdSTnf3MIO8fFeiINEbX36ZdNlfU/7A9f3gUw49B3oQsvwBA==", + "dev": true + }, + "esutils": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/esutils/-/esutils-2.0.3.tgz", + "integrity": "sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g==", + "dev": true + }, + "fast-deep-equal": { + "version": "3.1.3", + "resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-3.1.3.tgz", + "integrity": "sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q==", + "dev": true + }, + "fast-glob": { + "version": "3.3.1", + "resolved": "https://registry.npmjs.org/fast-glob/-/fast-glob-3.3.1.tgz", + "integrity": "sha512-kNFPyjhh5cKjrUltxs+wFx+ZkbRaxxmZ+X0ZU31SOsxCEtP9VPgtq2teZw1DebupL5GmDaNQ6yKMMVcM41iqDg==", + "dev": true, + "requires": { + "@nodelib/fs.stat": "^2.0.2", + "@nodelib/fs.walk": "^1.2.3", + "glob-parent": "^5.1.2", + "merge2": "^1.3.0", + "micromatch": "^4.0.4" + }, + "dependencies": { + "glob-parent": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.2.tgz", + "integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==", + "dev": true, + "requires": { + "is-glob": "^4.0.1" + } + } + } + }, + "fast-json-stable-stringify": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/fast-json-stable-stringify/-/fast-json-stable-stringify-2.1.0.tgz", + "integrity": "sha512-lhd/wF+Lk98HZoTCtlVraHtfh5XYijIjalXck7saUtuanSDyLMxnHhSXEDJqHxD7msR8D0uCmqlkwjCV8xvwHw==", + "dev": true + }, + "fast-levenshtein": { + "version": "2.0.6", + "resolved": "https://registry.npmjs.org/fast-levenshtein/-/fast-levenshtein-2.0.6.tgz", + "integrity": "sha512-DCXu6Ifhqcks7TZKY3Hxp3y6qphY5SJZmrWMDrKcERSOXWQdMhU9Ig/PYrzyw/ul9jOIyh0N4M0tbC5hodg8dw==", + "dev": true + }, + "fastq": { + "version": "1.15.0", + "resolved": "https://registry.npmjs.org/fastq/-/fastq-1.15.0.tgz", + "integrity": "sha512-wBrocU2LCXXa+lWBt8RoIRD89Fi8OdABODa/kEnyeyjS5aZO5/GNvI5sEINADqP/h8M29UHTHUb53sUu5Ihqdw==", + "dev": true, + "requires": { + "reusify": "^1.0.4" + } + }, + "file-entry-cache": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/file-entry-cache/-/file-entry-cache-6.0.1.tgz", + "integrity": "sha512-7Gps/XWymbLk2QLYK4NzpMOrYjMhdIxXuIvy2QBsLE6ljuodKvdkWs/cpyJJ3CVIVpH0Oi1Hvg1ovbMzLdFBBg==", + "dev": true, + "requires": { + "flat-cache": "^3.0.4" + } + }, + "fill-range": { + "version": "7.0.1", + "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.0.1.tgz", + "integrity": "sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ==", + "dev": true, + "requires": { + "to-regex-range": "^5.0.1" + } + }, + "find-up": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/find-up/-/find-up-5.0.0.tgz", + "integrity": "sha512-78/PXT1wlLLDgTzDs7sjq9hzz0vXD+zn+7wypEe4fXQxCmdmqfGsEPQxmiCSQI3ajFV91bVSsvNtrJRiW6nGng==", + "dev": true, + "requires": { + "locate-path": "^6.0.0", + "path-exists": "^4.0.0" + } + }, + "flat-cache": { + "version": "3.0.4", + "resolved": "https://registry.npmjs.org/flat-cache/-/flat-cache-3.0.4.tgz", + "integrity": "sha512-dm9s5Pw7Jc0GvMYbshN6zchCA9RgQlzzEZX3vylR9IqFfS8XciblUXOKfW6SiuJ0e13eDYZoZV5wdrev7P3Nwg==", + "dev": true, + "requires": { + "flatted": "^3.1.0", + "rimraf": "^3.0.2" + } + }, + "flatted": { + "version": "3.2.7", + "resolved": "https://registry.npmjs.org/flatted/-/flatted-3.2.7.tgz", + "integrity": "sha512-5nqDSxl8nn5BSNxyR3n4I6eDmbolI6WT+QqR547RwxQapgjQBmtktdP+HTBb/a/zLsbzERTONyUB5pefh5TtjQ==", + "dev": true + }, + "follow-redirects": { + "version": "1.15.2", + "resolved": "https://registry.npmjs.org/follow-redirects/-/follow-redirects-1.15.2.tgz", + "integrity": "sha512-VQLG33o04KaQ8uYi2tVNbdrWp1QWxNNea+nmIB4EVM28v0hmP17z7aG1+wAkNzVq4KeXTq3221ye5qTJP91JwA==" + }, + "form-data": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/form-data/-/form-data-4.0.0.tgz", + "integrity": "sha512-ETEklSGi5t0QMZuiXoA/Q6vcnxcLQP5vdugSpuAyi6SVGi2clPPp+xgEhuMaHC+zGgn31Kd235W35f7Hykkaww==", + "requires": { + "asynckit": "^0.4.0", + "combined-stream": "^1.0.8", + "mime-types": "^2.1.12" + } + }, + "fs.realpath": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz", + "integrity": "sha512-OO0pH2lK6a0hZnAdau5ItzHPI6pUlvI7jMVnxUQRtw4owF2wk8lOSabtGDCTP4Ggrg2MbGnWO9X8K1t4+fGMDw==", + "dev": true + }, + "fsevents": { + "version": "2.3.2", + "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.2.tgz", + "integrity": "sha512-xiqMQR4xAeHTuB9uWm+fFRcIOgKBMiOBP+eXiyT7jsgVCq1bkVygt00oASowB7EdtpOHaaPgKt812P9ab+DDKA==", + "dev": true, + "optional": true + }, + "glob": { + "version": "7.2.3", + "resolved": "https://registry.npmjs.org/glob/-/glob-7.2.3.tgz", + "integrity": "sha512-nFR0zLpU2YCaRxwoCJvL6UvCH2JFyFVIvwTLsIf21AuHlMskA1hhTdk+LlYJtOlYt9v6dvszD2BGRqBL+iQK9Q==", + "dev": true, + "requires": { + "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-parent": { + "version": "6.0.2", + "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-6.0.2.tgz", + "integrity": "sha512-XxwI8EOhVQgWp6iDL+3b0r86f4d6AX6zSU55HfB4ydCEuXLXc5FcYeOu+nnGftS4TEju/11rt4KJPTMgbfmv4A==", + "dev": true, + "requires": { + "is-glob": "^4.0.3" + } + }, + "globals": { + "version": "13.20.0", + "resolved": "https://registry.npmjs.org/globals/-/globals-13.20.0.tgz", + "integrity": "sha512-Qg5QtVkCy/kv3FUSlu4ukeZDVf9ee0iXLAUYX13gbR17bnejFTzr4iS9bY7kwCf1NztRNm1t91fjOiyx4CSwPQ==", + "dev": true, + "requires": { + "type-fest": "^0.20.2" + } + }, + "globby": { + "version": "11.1.0", + "resolved": "https://registry.npmjs.org/globby/-/globby-11.1.0.tgz", + "integrity": "sha512-jhIXaOzy1sb8IyocaruWSn1TjmnBVs8Ayhcy83rmxNJ8q2uWKCAj3CnJY+KpGSXCueAPc0i05kVvVKtP1t9S3g==", + "dev": true, + "requires": { + "array-union": "^2.1.0", + "dir-glob": "^3.0.1", + "fast-glob": "^3.2.9", + "ignore": "^5.2.0", + "merge2": "^1.4.1", + "slash": "^3.0.0" + } + }, + "graphemer": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/graphemer/-/graphemer-1.4.0.tgz", + "integrity": "sha512-EtKwoO6kxCL9WO5xipiHTZlSzBm7WLT627TqC/uVRd0HKmq8NXyebnNYxDoBi7wt8eTWrUrKXCOVaFq9x1kgag==", + "dev": true + }, + "hamt_plus": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/hamt_plus/-/hamt_plus-1.0.2.tgz", + "integrity": "sha512-t2JXKaehnMb9paaYA7J0BX8QQAY8lwfQ9Gjf4pg/mk4krt+cmwmU652HOoWonf+7+EQV97ARPMhhVgU1ra2GhA==" + }, + "has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true + }, + "ignore": { + "version": "5.2.4", + "resolved": "https://registry.npmjs.org/ignore/-/ignore-5.2.4.tgz", + "integrity": "sha512-MAb38BcSbH0eHNBxn7ql2NH/kX33OkB3lZ1BNdh7ENeRChHTYsTvWrMubiIAMNS2llXEEgZ1MUOBtXChP3kaFQ==", + "dev": true + }, + "import-fresh": { + "version": "3.3.0", + "resolved": "https://registry.npmjs.org/import-fresh/-/import-fresh-3.3.0.tgz", + "integrity": "sha512-veYYhQa+D1QBKznvhUHxb8faxlrwUnxseDAbAp457E0wLNio2bOSKnjYDhMj+YiAq61xrMGhQk9iXVk5FzgQMw==", + "dev": true, + "requires": { + "parent-module": "^1.0.0", + "resolve-from": "^4.0.0" + } + }, + "imurmurhash": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz", + "integrity": "sha512-JmXMZ6wuvDmLiHEml9ykzqO6lwFbof0GG4IkcGaENdCRDDmMVnny7s5HsIgHCbaq0w2MyPhDqkhTUgS2LU2PHA==", + "dev": true + }, + "inflight": { + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz", + "integrity": "sha512-k92I/b08q4wvFscXCLvqfsHCrjrF7yiXsQuIVvVE7N82W3+aqpzuUdBbfhWcy/FZR3/4IgflMgKLOsvPDrGCJA==", + "dev": true, + "requires": { + "once": "^1.3.0", + "wrappy": "1" + } + }, + "inherits": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz", + "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==", + "dev": true + }, + "is-extglob": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", + "integrity": "sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ==", + "dev": true + }, + "is-glob": { + "version": "4.0.3", + "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.3.tgz", + "integrity": "sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==", + "dev": true, + "requires": { + "is-extglob": "^2.1.1" + } + }, + "is-number": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz", + "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==", + "dev": true + }, + "is-path-inside": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/is-path-inside/-/is-path-inside-3.0.3.tgz", + "integrity": "sha512-Fd4gABb+ycGAmKou8eMftCupSir5lRxqf4aD/vd0cD2qc4HL07OjCeuHMr8Ro4CoMaeCKDB0/ECBOVWjTwUvPQ==", + "dev": true + }, + "isexe": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz", + "integrity": "sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw==", + "dev": true + }, + "js-tokens": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-4.0.0.tgz", + "integrity": "sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==" + }, + "js-yaml": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-4.1.0.tgz", + "integrity": "sha512-wpxZs9NoxZaJESJGIZTyDEaYpl0FKSA+FB9aJiyemKhMwkxQg63h4T1KJgUGHpTqPDNRcmmYLugrRjJlBtWvRA==", + "dev": true, + "requires": { + "argparse": "^2.0.1" + } + }, + "json-schema-traverse": { + "version": "0.4.1", + "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz", + "integrity": "sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg==", + "dev": true + }, + "json-stable-stringify-without-jsonify": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/json-stable-stringify-without-jsonify/-/json-stable-stringify-without-jsonify-1.0.1.tgz", + "integrity": "sha512-Bdboy+l7tA3OGW6FjyFHWkP5LuByj1Tk33Ljyq0axyzdk9//JSi2u3fP1QSmd1KNwq6VOKYGlAu87CisVir6Pw==", + "dev": true + }, + "json2mq": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/json2mq/-/json2mq-0.2.0.tgz", + "integrity": "sha512-SzoRg7ux5DWTII9J2qkrZrqV1gt+rTaoufMxEzXbS26Uid0NwaJd123HcoB80TgubEppxxIGdNxCx50fEoEWQA==", + "requires": { + "string-convert": "^0.2.0" + } + }, + "levn": { + "version": "0.4.1", + "resolved": "https://registry.npmjs.org/levn/-/levn-0.4.1.tgz", + "integrity": "sha512-+bT2uH4E5LGE7h/n3evcS/sQlJXCpIp6ym8OWJ5eV6+67Dsql/LaaT7qJBAt2rzfoa/5QBGBhxDix1dMt2kQKQ==", + "dev": true, + "requires": { + "prelude-ls": "^1.2.1", + "type-check": "~0.4.0" + } + }, + "locate-path": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-6.0.0.tgz", + "integrity": "sha512-iPZK6eYjbxRu3uB4/WZ3EsEIMJFMqAoopl3R+zuq0UjcAm/MO6KCweDgPfP3elTztoKP3KtnVHxTn2NHBSDVUw==", + "dev": true, + "requires": { + "p-locate": "^5.0.0" + } + }, + "lodash.merge": { + "version": "4.6.2", + "resolved": "https://registry.npmjs.org/lodash.merge/-/lodash.merge-4.6.2.tgz", + "integrity": "sha512-0KpjqXRVvrYyCsX1swR/XTK0va6VQkQM6MNo7PqW77ByjAhoARA8EfrP1N4+KlKj8YS0ZUCtRT/YUuhyYDujIQ==", + "dev": true + }, + "loose-envify": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/loose-envify/-/loose-envify-1.4.0.tgz", + "integrity": "sha512-lyuxPGr/Wfhrlem2CL/UcnUc1zcqKAImBDzukY7Y5F/yQiNdko6+fRLevlw1HgMySw7f611UIY408EtxRSoK3Q==", + "requires": { + "js-tokens": "^3.0.0 || ^4.0.0" + } + }, + "lru-cache": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-6.0.0.tgz", + "integrity": "sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA==", + "dev": true, + "requires": { + "yallist": "^4.0.0" + } + }, + "merge2": { + "version": "1.4.1", + "resolved": "https://registry.npmjs.org/merge2/-/merge2-1.4.1.tgz", + "integrity": "sha512-8q7VEgMJW4J8tcfVPy8g09NcQwZdbwFEqhe/WZkoIzjn/3TGDwtOCYtXGxA3O8tPzpczCCDgv+P2P5y00ZJOOg==", + "dev": true + }, + "micromatch": { + "version": "4.0.5", + "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-4.0.5.tgz", + "integrity": "sha512-DMy+ERcEW2q8Z2Po+WNXuw3c5YaUSFjAO5GsJqfEl7UjvtIuFKO6ZrKvcItdy98dwFI2N1tg3zNIdKaQT+aNdA==", + "dev": true, + "requires": { + "braces": "^3.0.2", + "picomatch": "^2.3.1" + } + }, + "mime-db": { + "version": "1.52.0", + "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.52.0.tgz", + "integrity": "sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==" + }, + "mime-types": { + "version": "2.1.35", + "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.35.tgz", + "integrity": "sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==", + "requires": { + "mime-db": "1.52.0" + } + }, + "minimatch": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", + "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", + "dev": true, + "requires": { + "brace-expansion": "^1.1.7" + } + }, + "moment": { + "version": "2.29.4", + "resolved": "https://registry.npmjs.org/moment/-/moment-2.29.4.tgz", + "integrity": "sha512-5LC9SOxjSc2HF6vO2CyuTDNivEdoz2IvyJJGj6X8DJ0eFyfszE0QiEd+iXmBvUP3WHxSjFH/vIsA0EN00cgr8w==", + "optional": true, + "peer": true + }, + "ms": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", + "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==", + "dev": true + }, + "nanoid": { + "version": "3.3.6", + "resolved": "https://registry.npmjs.org/nanoid/-/nanoid-3.3.6.tgz", + "integrity": "sha512-BGcqMMJuToF7i1rt+2PWSNVnWIkGCU78jBG3RxO/bZlnZPK2Cmi2QaffxGO/2RvWi9sL+FAiRiXMgsyxQ1DIDA==", + "dev": true + }, + "natural-compare": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/natural-compare/-/natural-compare-1.4.0.tgz", + "integrity": "sha512-OWND8ei3VtNC9h7V60qff3SVobHr996CTwgxubgyQYEpg290h9J0buyECNNJexkFm5sOajh5G116RYA1c8ZMSw==", + "dev": true + }, + "natural-compare-lite": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/natural-compare-lite/-/natural-compare-lite-1.4.0.tgz", + "integrity": "sha512-Tj+HTDSJJKaZnfiuw+iaF9skdPpTo2GtEly5JHnWV/hfv2Qj/9RKsGISQtLh2ox3l5EAGw487hnBee0sIJ6v2g==", + "dev": true + }, + "once": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz", + "integrity": "sha512-lNaJgI+2Q5URQBkccEKHTQOPaXdUxnZZElQTZY0MFUAuaEqe1E+Nyvgdz/aIyNi6Z9MzO5dv1H8n58/GELp3+w==", + "dev": true, + "requires": { + "wrappy": "1" + } + }, + "optionator": { + "version": "0.9.3", + "resolved": "https://registry.npmjs.org/optionator/-/optionator-0.9.3.tgz", + "integrity": "sha512-JjCoypp+jKn1ttEFExxhetCKeJt9zhAgAve5FXHixTvFDW/5aEktX9bufBKLRRMdU7bNtpLfcGu94B3cdEJgjg==", + "dev": true, + "requires": { + "@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": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-3.1.0.tgz", + "integrity": "sha512-TYOanM3wGwNGsZN2cVTYPArw454xnXj5qmWF1bEoAc4+cU/ol7GVh7odevjp1FNHduHc3KZMcFduxU5Xc6uJRQ==", + "dev": true, + "requires": { + "yocto-queue": "^0.1.0" + } + }, + "p-locate": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-5.0.0.tgz", + "integrity": "sha512-LaNjtRWUBY++zB5nE/NwcaoMylSPk+S+ZHNB1TzdbMJMny6dynpAGt7X/tl/QYq3TIeE6nxHppbo2LGymrG5Pw==", + "dev": true, + "requires": { + "p-limit": "^3.0.2" + } + }, + "parent-module": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/parent-module/-/parent-module-1.0.1.tgz", + "integrity": "sha512-GQ2EWRpQV8/o+Aw8YqtfZZPfNRWZYkbidE9k5rpl/hC3vtHHBfGm2Ifi6qWV+coDGkrUKZAxE3Lot5kcsRlh+g==", + "dev": true, + "requires": { + "callsites": "^3.0.0" + } + }, + "path-exists": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-4.0.0.tgz", + "integrity": "sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==", + "dev": true + }, + "path-is-absolute": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/path-is-absolute/-/path-is-absolute-1.0.1.tgz", + "integrity": "sha512-AVbw3UJ2e9bq64vSaS9Am0fje1Pa8pbGqTTsmXfaIiMpnr5DlDhfJOuLj9Sf95ZPVDAUerDfEk88MPmPe7UCQg==", + "dev": true + }, + "path-key": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/path-key/-/path-key-3.1.1.tgz", + "integrity": "sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q==", + "dev": true + }, + "path-type": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/path-type/-/path-type-4.0.0.tgz", + "integrity": "sha512-gDKb8aZMDeD/tZWs9P6+q0J9Mwkdl6xMV8TjnGP3qJVJ06bdMgkbBlLU8IdfOsIsFz2BW1rNVT3XuNEl8zPAvw==", + "dev": true + }, + "picocolors": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/picocolors/-/picocolors-1.0.0.tgz", + "integrity": "sha512-1fygroTLlHu66zi26VoTDv8yRgm0Fccecssto+MhsZ0D/DGW2sm8E8AjW7NU5VVTRt5GxbeZ5qBuJr+HyLYkjQ==", + "dev": true + }, + "picomatch": { + "version": "2.3.1", + "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.1.tgz", + "integrity": "sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==", + "dev": true + }, + "postcss": { + "version": "8.4.27", + "resolved": "https://registry.npmjs.org/postcss/-/postcss-8.4.27.tgz", + "integrity": "sha512-gY/ACJtJPSmUFPDCHtX78+01fHa64FaU4zaaWfuh1MhGJISufJAH4cun6k/8fwsHYeK4UQmENQK+tRLCFJE8JQ==", + "dev": true, + "requires": { + "nanoid": "^3.3.6", + "picocolors": "^1.0.0", + "source-map-js": "^1.0.2" + } + }, + "preact": { + "version": "10.12.1", + "resolved": "https://registry.npmjs.org/preact/-/preact-10.12.1.tgz", + "integrity": "sha512-l8386ixSsBdbreOAkqtrwqHwdvR35ID8c3rKPa8lCWuO86dBi32QWHV4vfsZK1utLLFMvw+Z5Ad4XLkZzchscg==" + }, + "prelude-ls": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/prelude-ls/-/prelude-ls-1.2.1.tgz", + "integrity": "sha512-vkcDPrRZo1QZLbn5RLGPpg/WmIQ65qoWWhcGKf/b5eplkkarX0m9z8ppCat4mlOqUsWpyNuYgO3VRyrYHSzX5g==", + "dev": true + }, + "proxy-from-env": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/proxy-from-env/-/proxy-from-env-1.1.0.tgz", + "integrity": "sha512-D+zkORCbA9f1tdWRK0RaCR3GPv50cMxcrz4X8k5LTSUD1Dkw47mKJEZQNunItRTkWwgtaUSo1RVFRIG9ZXiFYg==" + }, + "punycode": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/punycode/-/punycode-2.3.0.tgz", + "integrity": "sha512-rRV+zQD8tVFys26lAGR9WUuS4iUAngJScM+ZRSKtvl5tKeZ2t5bvdNFdNHBW9FWR4guGHlgmsZ1G7BSm2wTbuA==", + "dev": true + }, + "qrcode.react": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/qrcode.react/-/qrcode.react-3.1.0.tgz", + "integrity": "sha512-oyF+Urr3oAMUG/OiOuONL3HXM+53wvuH3mtIWQrYmsXoAq0DkvZp2RYUWFSMFtbdOpuS++9v+WAkzNVkMlNW6Q==", + "requires": {} + }, + "queue-microtask": { + "version": "1.2.3", + "resolved": "https://registry.npmjs.org/queue-microtask/-/queue-microtask-1.2.3.tgz", + "integrity": "sha512-NuaNSa6flKT5JaSYQzJok04JzTL1CA6aGhv5rfLW3PgqA+M2ChpZQnAC8h8i4ZFkBS8X5RqkDBHA7r4hej3K9A==", + "dev": true + }, + "rc-align": { + "version": "4.0.15", + "resolved": "https://registry.npmjs.org/rc-align/-/rc-align-4.0.15.tgz", + "integrity": "sha512-wqJtVH60pka/nOX7/IspElA8gjPNQKIx/ZqJ6heATCkXpe1Zg4cPVrMD2vC96wjsFFL8WsmhPbx9tdMo1qqlIA==", + "requires": { + "@babel/runtime": "^7.10.1", + "classnames": "2.x", + "dom-align": "^1.7.0", + "rc-util": "^5.26.0", + "resize-observer-polyfill": "^1.5.1" + } + }, + "rc-cascader": { + "version": "3.12.1", + "resolved": "https://registry.npmjs.org/rc-cascader/-/rc-cascader-3.12.1.tgz", + "integrity": "sha512-g6In2y6eudHXS/Fs9dKFhp9acvHRUPqem/7xReR9ng8M1pNAE137uGBOt9WNpgsKT/cDGudXZQVehaBwAKg6hQ==", + "requires": { + "@babel/runtime": "^7.12.5", + "array-tree-filter": "^2.1.0", + "classnames": "^2.3.1", + "rc-select": "~14.5.0", + "rc-tree": "~5.7.0", + "rc-util": "^5.6.1" + } + }, + "rc-checkbox": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/rc-checkbox/-/rc-checkbox-3.1.0.tgz", + "integrity": "sha512-PAwpJFnBa3Ei+5pyqMMXdcKYKNBMS+TvSDiLdDnARnMJHC8ESxwPfm4Ao1gJiKtWLdmGfigascnCpwrHFgoOBQ==", + "requires": { + "@babel/runtime": "^7.10.1", + "classnames": "^2.3.2", + "rc-util": "^5.25.2" + } + }, + "rc-collapse": { + "version": "3.7.0", + "resolved": "https://registry.npmjs.org/rc-collapse/-/rc-collapse-3.7.0.tgz", + "integrity": "sha512-Cir1c89cENiK5wryd9ut+XltrIfx/+KH1/63uJIVjuXkgfrIvIy6W1fYGgEYtttbHW2fEfxg1s31W+Vm98fSRw==", + "requires": { + "@babel/runtime": "^7.10.1", + "classnames": "2.x", + "rc-motion": "^2.3.4", + "rc-util": "^5.27.0" + } + }, + "rc-dialog": { + "version": "9.1.0", + "resolved": "https://registry.npmjs.org/rc-dialog/-/rc-dialog-9.1.0.tgz", + "integrity": "sha512-5ry+JABAWEbaKyYsmITtrJbZbJys8CtMyzV8Xn4LYuXMeUx5XVHNyJRoqLFE4AzBuXXzOWeaC49cg+XkxK6kHA==", + "requires": { + "@babel/runtime": "^7.10.1", + "@rc-component/portal": "^1.0.0-8", + "classnames": "^2.2.6", + "rc-motion": "^2.3.0", + "rc-util": "^5.21.0" + } + }, + "rc-drawer": { + "version": "6.2.0", + "resolved": "https://registry.npmjs.org/rc-drawer/-/rc-drawer-6.2.0.tgz", + "integrity": "sha512-spPkZ3WvP0U0vy5dyzSwlUJ/+vLFtjP/cTwSwejhQRoDBaexSZHsBhELoCZcEggI7LQ7typmtG30lAue2HEhvA==", + "requires": { + "@babel/runtime": "^7.10.1", + "@rc-component/portal": "^1.1.1", + "classnames": "^2.2.6", + "rc-motion": "^2.6.1", + "rc-util": "^5.21.2" + } + }, + "rc-dropdown": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/rc-dropdown/-/rc-dropdown-4.1.0.tgz", + "integrity": "sha512-VZjMunpBdlVzYpEdJSaV7WM7O0jf8uyDjirxXLZRNZ+tAC+NzD3PXPEtliFwGzVwBBdCmGuSqiS9DWcOLxQ9tw==", + "requires": { + "@babel/runtime": "^7.18.3", + "@rc-component/trigger": "^1.7.0", + "classnames": "^2.2.6", + "rc-util": "^5.17.0" + } + }, + "rc-field-form": { + "version": "1.34.2", + "resolved": "https://registry.npmjs.org/rc-field-form/-/rc-field-form-1.34.2.tgz", + "integrity": "sha512-BdciU5C7dBO51/9ZKcMvK2f8zaaO12Lt1eBhlAo8nNv+6htlNcgY9DAkUlZ7gfyWjnCc1Oo4hHIXau1m6tLw1A==", + "requires": { + "@babel/runtime": "^7.18.0", + "async-validator": "^4.1.0", + "rc-util": "^5.32.2" + } + }, + "rc-image": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/rc-image/-/rc-image-7.0.0.tgz", + "integrity": "sha512-pOr/LYthg5a+R2LDlFPv8u2ndX4aJQNghWCiWxflmLglC3p0uts/NIWLAituQOKvV1wO1aFI1CZtLMT7jrU3vA==", + "requires": { + "@babel/runtime": "^7.11.2", + "@rc-component/portal": "^1.0.2", + "classnames": "^2.2.6", + "rc-dialog": "~9.1.0", + "rc-motion": "^2.6.2", + "rc-util": "^5.34.1" + } + }, + "rc-input": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/rc-input/-/rc-input-1.1.0.tgz", + "integrity": "sha512-izuNXPABQPh4KD7ANFcTrIGp9EZU0FkjTw6AvwCQ/rGPrdDsUTHLsp/Wju/kzGMLJFJWKNF3smbmXRNO23DtXA==", + "requires": { + "@babel/runtime": "^7.11.1", + "classnames": "^2.2.1", + "rc-util": "^5.18.1" + } + }, + "rc-input-number": { + "version": "8.0.3", + "resolved": "https://registry.npmjs.org/rc-input-number/-/rc-input-number-8.0.3.tgz", + "integrity": "sha512-GHfWvufXEmwF/wtR8oPZNTuMdFb/rvx/+Sp2bZfaPftM+LFFdO8o3/PaeTk8DKt0Tv+u5Zuf68lqLdGCkmAXRg==", + "requires": { + "@babel/runtime": "^7.10.1", + "@rc-component/mini-decimal": "^1.0.1", + "classnames": "^2.2.5", + "rc-input": "~1.1.0", + "rc-util": "^5.28.0" + } + }, + "rc-mentions": { + "version": "2.5.0", + "resolved": "https://registry.npmjs.org/rc-mentions/-/rc-mentions-2.5.0.tgz", + "integrity": "sha512-rERXsbUTNVrb5T/iDC0ki/SRGWJnOVraDy6O25Us3FSpuUZ3uq2TPZB4fRk0Hss5kyiEPzz2sprhkI4b+F4jUw==", + "requires": { + "@babel/runtime": "^7.22.5", + "@rc-component/trigger": "^1.5.0", + "classnames": "^2.2.6", + "rc-input": "~1.1.0", + "rc-menu": "~9.10.0", + "rc-textarea": "~1.3.0", + "rc-util": "^5.22.5" + } + }, + "rc-menu": { + "version": "9.10.0", + "resolved": "https://registry.npmjs.org/rc-menu/-/rc-menu-9.10.0.tgz", + "integrity": "sha512-g27kpXaAoJh/fkPZF65/d4V+w4DhDeqomBdPcGnkFAcJnEM4o21TnVccrBUoDedLKzC7wJRw1Q7VTqEsfEufmw==", + "requires": { + "@babel/runtime": "^7.10.1", + "@rc-component/trigger": "^1.6.2", + "classnames": "2.x", + "rc-motion": "^2.4.3", + "rc-overflow": "^1.3.1", + "rc-util": "^5.27.0" + } + }, + "rc-motion": { + "version": "2.7.3", + "resolved": "https://registry.npmjs.org/rc-motion/-/rc-motion-2.7.3.tgz", + "integrity": "sha512-2xUvo8yGHdOHeQbdI8BtBsCIrWKchEmFEIskf0nmHtJsou+meLd/JE+vnvSX2JxcBrJtXY2LuBpxAOxrbY/wMQ==", + "requires": { + "@babel/runtime": "^7.11.1", + "classnames": "^2.2.1", + "rc-util": "^5.21.0" + } + }, + "rc-notification": { + "version": "5.0.5", + "resolved": "https://registry.npmjs.org/rc-notification/-/rc-notification-5.0.5.tgz", + "integrity": "sha512-uEz2jggourwv/rR0obe7RHEa63UchqX4k+e+Qt2c3LaY7U9Tc+L6ANhzgCKYSA/afm0ebjmNZHoB5Cv47xEOcA==", + "requires": { + "@babel/runtime": "^7.10.1", + "classnames": "2.x", + "rc-motion": "^2.6.0", + "rc-util": "^5.20.1" + } + }, + "rc-overflow": { + "version": "1.3.1", + "resolved": "https://registry.npmjs.org/rc-overflow/-/rc-overflow-1.3.1.tgz", + "integrity": "sha512-RY0nVBlfP9CkxrpgaLlGzkSoh9JhjJLu6Icqs9E7CW6Ewh9s0peF9OHIex4OhfoPsR92LR0fN6BlCY9Z4VoUtA==", + "requires": { + "@babel/runtime": "^7.11.1", + "classnames": "^2.2.1", + "rc-resize-observer": "^1.0.0", + "rc-util": "^5.19.2" + } + }, + "rc-pagination": { + "version": "3.5.0", + "resolved": "https://registry.npmjs.org/rc-pagination/-/rc-pagination-3.5.0.tgz", + "integrity": "sha512-lUBVtVVUn7gGsq4mTyVpcZQr+AMcljbMiL/HcCmSdFrcsK0iZVKwwbXDxhz2IV0JXUs9Hzepr5sQFaF+9ad/pQ==", + "requires": { + "@babel/runtime": "^7.10.1", + "classnames": "^2.2.1", + "rc-util": "^5.32.2" + } + }, + "rc-picker": { + "version": "3.10.0", + "resolved": "https://registry.npmjs.org/rc-picker/-/rc-picker-3.10.0.tgz", + "integrity": "sha512-Euki50qtEct6ByOeYlnA4TLs/LcXz7BAYS4cmCTKJ3dWg2sNTVtredLdbS9aJ/9fhMacxGAYAlcQJpQx+av43A==", + "requires": { + "@babel/runtime": "^7.10.1", + "@rc-component/trigger": "^1.5.0", + "classnames": "^2.2.1", + "rc-util": "^5.30.0" + } + }, + "rc-progress": { + "version": "3.4.2", + "resolved": "https://registry.npmjs.org/rc-progress/-/rc-progress-3.4.2.tgz", + "integrity": "sha512-iAGhwWU+tsayP+Jkl9T4+6rHeQTG9kDz8JAHZk4XtQOcYN5fj9H34NXNEdRdZx94VUDHMqCb1yOIvi8eJRh67w==", + "requires": { + "@babel/runtime": "^7.10.1", + "classnames": "^2.2.6", + "rc-util": "^5.16.1" + } + }, + "rc-rate": { + "version": "2.12.0", + "resolved": "https://registry.npmjs.org/rc-rate/-/rc-rate-2.12.0.tgz", + "integrity": "sha512-g092v5iZCdVzbjdn28FzvWebK2IutoVoiTeqoLTj9WM7SjA/gOJIw5/JFZMRyJYYVe1jLAU2UhAfstIpCNRozg==", + "requires": { + "@babel/runtime": "^7.10.1", + "classnames": "^2.2.5", + "rc-util": "^5.0.1" + } + }, + "rc-resize-observer": { + "version": "1.3.1", + "resolved": "https://registry.npmjs.org/rc-resize-observer/-/rc-resize-observer-1.3.1.tgz", + "integrity": "sha512-iFUdt3NNhflbY3mwySv5CA1TC06zdJ+pfo0oc27xpf4PIOvfZwZGtD9Kz41wGYqC4SLio93RVAirSSpYlV/uYg==", + "requires": { + "@babel/runtime": "^7.20.7", + "classnames": "^2.2.1", + "rc-util": "^5.27.0", + "resize-observer-polyfill": "^1.5.1" + } + }, + "rc-segmented": { + "version": "2.2.2", + "resolved": "https://registry.npmjs.org/rc-segmented/-/rc-segmented-2.2.2.tgz", + "integrity": "sha512-Mq52M96QdHMsNdE/042ibT5vkcGcD5jxKp7HgPC2SRofpia99P5fkfHy1pEaajLMF/kj0+2Lkq1UZRvqzo9mSA==", + "requires": { + "@babel/runtime": "^7.11.1", + "classnames": "^2.2.1", + "rc-motion": "^2.4.4", + "rc-util": "^5.17.0" + } + }, + "rc-select": { + "version": "14.5.2", + "resolved": "https://registry.npmjs.org/rc-select/-/rc-select-14.5.2.tgz", + "integrity": "sha512-Np/lDHvxCnVhVsheQjSV1I/OMJTWJf1n10wq8q1AGy3ytyYLfjNpi6uaz/pmjsbbiSddSWzJnNZCli9LmgBZsA==", + "requires": { + "@babel/runtime": "^7.10.1", + "@rc-component/trigger": "^1.5.0", + "classnames": "2.x", + "rc-motion": "^2.0.1", + "rc-overflow": "^1.0.0", + "rc-util": "^5.16.1", + "rc-virtual-list": "^3.5.2" + } + }, + "rc-slider": { + "version": "10.1.1", + "resolved": "https://registry.npmjs.org/rc-slider/-/rc-slider-10.1.1.tgz", + "integrity": "sha512-gn8oXazZISEhnmRinI89Z/JD/joAaM35jp+gDtIVSTD/JJMCCBqThqLk1SVJmvtfeiEF/kKaFY0+qt4SDHFUDw==", + "requires": { + "@babel/runtime": "^7.10.1", + "classnames": "^2.2.5", + "rc-util": "^5.27.0" + } + }, + "rc-steps": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/rc-steps/-/rc-steps-6.0.1.tgz", + "integrity": "sha512-lKHL+Sny0SeHkQKKDJlAjV5oZ8DwCdS2hFhAkIjuQt1/pB81M0cA0ErVFdHq9+jmPmFw1vJB2F5NBzFXLJxV+g==", + "requires": { + "@babel/runtime": "^7.16.7", + "classnames": "^2.2.3", + "rc-util": "^5.16.1" + } + }, + "rc-switch": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/rc-switch/-/rc-switch-4.1.0.tgz", + "integrity": "sha512-TI8ufP2Az9oEbvyCeVE4+90PDSljGyuwix3fV58p7HV2o4wBnVToEyomJRVyTaZeqNPAp+vqeo4Wnj5u0ZZQBg==", + "requires": { + "@babel/runtime": "^7.21.0", + "classnames": "^2.2.1", + "rc-util": "^5.30.0" + } + }, + "rc-table": { + "version": "7.32.1", + "resolved": "https://registry.npmjs.org/rc-table/-/rc-table-7.32.1.tgz", + "integrity": "sha512-fHMQteKMocUC9I9Vex3eBLH7QsiaMR/qtzh3B1Ty2PoNGwVTwVdDFyRL05zch+JU3KnNNczgQeVvtf/p//gdrQ==", + "requires": { + "@babel/runtime": "^7.10.1", + "@rc-component/context": "^1.3.0", + "classnames": "^2.2.5", + "rc-resize-observer": "^1.1.0", + "rc-util": "^5.27.1" + } + }, + "rc-tabs": { + "version": "12.9.0", + "resolved": "https://registry.npmjs.org/rc-tabs/-/rc-tabs-12.9.0.tgz", + "integrity": "sha512-2HnVowgMVrq0DfQtyu4mCd9E6pXlWNdM6VaDvOOHMsLYqPmpY+7zBqUC6YrrQ9xYXHciTS0e7TtjOHIvpVCHLQ==", + "requires": { + "@babel/runtime": "^7.11.2", + "classnames": "2.x", + "rc-dropdown": "~4.1.0", + "rc-menu": "~9.10.0", + "rc-motion": "^2.6.2", + "rc-resize-observer": "^1.0.0", + "rc-util": "^5.16.0" + } + }, + "rc-textarea": { + "version": "1.3.3", + "resolved": "https://registry.npmjs.org/rc-textarea/-/rc-textarea-1.3.3.tgz", + "integrity": "sha512-846kjD/RYZx/th32FW4T80IrRTt2dT7+kxdToI7pwzJPlsfmZyo8e2F2m0FLcvriv6rtAUMSqQRH1HC3i+sAbw==", + "requires": { + "@babel/runtime": "^7.10.1", + "classnames": "^2.2.1", + "rc-input": "~1.1.0", + "rc-resize-observer": "^1.0.0", + "rc-util": "^5.27.0" + } + }, + "rc-tooltip": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/rc-tooltip/-/rc-tooltip-6.0.1.tgz", + "integrity": "sha512-MdvPlsD1fDSxKp9+HjXrc/CxLmA/s11QYIh1R7aExxfodKP7CZA++DG1AjrW80F8IUdHYcR43HAm0Y2BYPelHA==", + "requires": { + "@babel/runtime": "^7.11.2", + "@rc-component/trigger": "^1.0.4", + "classnames": "^2.3.1" + } + }, + "rc-tree": { + "version": "5.7.9", + "resolved": "https://registry.npmjs.org/rc-tree/-/rc-tree-5.7.9.tgz", + "integrity": "sha512-1hKkToz/EVjJlMVwmZnpXeLXt/1iQMsaAq9m+GNkUbK746gkc7QpJXSN/TzjhTI5Hi+LOSlrMaXLMT0bHPqILQ==", + "requires": { + "@babel/runtime": "^7.10.1", + "classnames": "2.x", + "rc-motion": "^2.0.1", + "rc-util": "^5.16.1", + "rc-virtual-list": "^3.5.1" + } + }, + "rc-tree-select": { + "version": "5.9.0", + "resolved": "https://registry.npmjs.org/rc-tree-select/-/rc-tree-select-5.9.0.tgz", + "integrity": "sha512-oh3blESzLfLCBPSiVDtZ2irzrWWZUMeHvnSwRvFo79br8Z+K/1OhXhXBZmROvfKwaH8YUugAQy8B2j5EGQbdyA==", + "requires": { + "@babel/runtime": "^7.10.1", + "classnames": "2.x", + "rc-select": "~14.5.0", + "rc-tree": "~5.7.0", + "rc-util": "^5.16.1" + } + }, + "rc-upload": { + "version": "4.3.4", + "resolved": "https://registry.npmjs.org/rc-upload/-/rc-upload-4.3.4.tgz", + "integrity": "sha512-uVbtHFGNjHG/RyAfm9fluXB6pvArAGyAx8z7XzXXyorEgVIWj6mOlriuDm0XowDHYz4ycNK0nE0oP3cbFnzxiQ==", + "requires": { + "@babel/runtime": "^7.18.3", + "classnames": "^2.2.5", + "rc-util": "^5.2.0" + } + }, + "rc-util": { + "version": "5.34.1", + "resolved": "https://registry.npmjs.org/rc-util/-/rc-util-5.34.1.tgz", + "integrity": "sha512-SqiUT8Ssgh5C+hu4y887xwCrMNcxLm6ScOo8AFlWYYF3z9uNNiPpwwSjvicqOlWd79rNw1g44rnP7tz9MrO1ZQ==", + "requires": { + "@babel/runtime": "^7.18.3", + "react-is": "^16.12.0" + } + }, + "rc-virtual-list": { + "version": "3.5.3", + "resolved": "https://registry.npmjs.org/rc-virtual-list/-/rc-virtual-list-3.5.3.tgz", + "integrity": "sha512-rG6IuD4EYM8K6oZ8Shu2BC/CmcTdqng4yBWkc/5fjWhB20bl6QwR2Upyt7+MxvfscoVm8zOQY+tcpEO5cu4GaQ==", + "requires": { + "@babel/runtime": "^7.20.0", + "classnames": "^2.2.6", + "rc-resize-observer": "^1.0.0", + "rc-util": "^5.15.0" + } + }, + "react": { + "version": "18.2.0", + "resolved": "https://registry.npmjs.org/react/-/react-18.2.0.tgz", + "integrity": "sha512-/3IjMdb2L9QbBdWiW5e3P2/npwMBaU9mHCSCUzNln0ZCYbcfTsGbTJrU/kGemdH2IWmB2ioZ+zkxtmq6g09fGQ==", + "requires": { + "loose-envify": "^1.1.0" + } + }, + "react-dom": { + "version": "18.2.0", + "resolved": "https://registry.npmjs.org/react-dom/-/react-dom-18.2.0.tgz", + "integrity": "sha512-6IMTriUmvsjHUjNtEDudZfuDQUoWXVxKHhlEGSk81n4YFS+r/Kl99wXiwlVXtPBtJenozv2P+hxDsw9eA7Xo6g==", + "requires": { + "loose-envify": "^1.1.0", + "scheduler": "^0.23.0" + } + }, + "react-is": { + "version": "16.13.1", + "resolved": "https://registry.npmjs.org/react-is/-/react-is-16.13.1.tgz", + "integrity": "sha512-24e6ynE2H+OKt4kqsOvNd8kBpV65zoxbA4BVsEOB3ARVWQki/DHzaUoC5KuON/BiccDaCCTZBuOcfZs70kR8bQ==" + }, + "react-router": { + "version": "6.14.2", + "resolved": "https://registry.npmjs.org/react-router/-/react-router-6.14.2.tgz", + "integrity": "sha512-09Zss2dE2z+T1D03IheqAFtK4UzQyX8nFPWx6jkwdYzGLXd5ie06A6ezS2fO6zJfEb/SpG6UocN2O1hfD+2urQ==", + "requires": { + "@remix-run/router": "1.7.2" + } + }, + "react-router-dom": { + "version": "6.14.2", + "resolved": "https://registry.npmjs.org/react-router-dom/-/react-router-dom-6.14.2.tgz", + "integrity": "sha512-5pWX0jdKR48XFZBuJqHosX3AAHjRAzygouMTyimnBPOLdY3WjzUSKhus2FVMihUFWzeLebDgr4r8UeQFAct7Bg==", + "requires": { + "@remix-run/router": "1.7.2", + "react-router": "6.14.2" + } + }, + "recoil": { + "version": "0.7.7", + "resolved": "https://registry.npmjs.org/recoil/-/recoil-0.7.7.tgz", + "integrity": "sha512-8Og5KPQW9LwC577Vc7Ug2P0vQshkv1y3zG3tSSkWMqkWSwHmE+by06L8JtnGocjW6gcCvfwB3YtrJG6/tWivNQ==", + "requires": { + "hamt_plus": "1.0.2" + } + }, + "regenerator-runtime": { + "version": "0.13.11", + "resolved": "https://registry.npmjs.org/regenerator-runtime/-/regenerator-runtime-0.13.11.tgz", + "integrity": "sha512-kY1AZVr2Ra+t+piVaJ4gxaFaReZVH40AKNo7UCX6W+dEwBo/2oZJzqfuN1qLq1oL45o56cPaTXELwrTh8Fpggg==" + }, + "resize-observer-polyfill": { + "version": "1.5.1", + "resolved": "https://registry.npmjs.org/resize-observer-polyfill/-/resize-observer-polyfill-1.5.1.tgz", + "integrity": "sha512-LwZrotdHOo12nQuZlHEmtuXdqGoOD0OhaxopaNFxWzInpEgaLWoVuAMbTzixuosCx2nEG58ngzW3vxdWoxIgdg==" + }, + "resolve-from": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-4.0.0.tgz", + "integrity": "sha512-pb/MYmXstAkysRFx8piNI1tGFNQIFA3vkE3Gq4EuA1dF6gHp/+vgZqsCGJapvy8N3Q+4o7FwvquPJcnZ7RYy4g==", + "dev": true + }, + "reusify": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/reusify/-/reusify-1.0.4.tgz", + "integrity": "sha512-U9nH88a3fc/ekCF1l0/UP1IosiuIjyTh7hBvXVMHYgVcfGvt897Xguj2UOLDeI5BG2m7/uwyaLVT6fbtCwTyzw==", + "dev": true + }, + "rimraf": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-3.0.2.tgz", + "integrity": "sha512-JZkJMZkAGFFPP2YqXZXPbMlMBgsxzE8ILs4lMIX/2o0L9UBw9O/Y3o6wFw/i9YLapcUJWwqbi3kdxIPdC62TIA==", + "dev": true, + "requires": { + "glob": "^7.1.3" + } + }, + "rollup": { + "version": "3.26.3", + "resolved": "https://registry.npmjs.org/rollup/-/rollup-3.26.3.tgz", + "integrity": "sha512-7Tin0C8l86TkpcMtXvQu6saWH93nhG3dGQ1/+l5V2TDMceTxO7kDiK6GzbfLWNNxqJXm591PcEZUozZm51ogwQ==", + "dev": true, + "requires": { + "fsevents": "~2.3.2" + } + }, + "run-parallel": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/run-parallel/-/run-parallel-1.2.0.tgz", + "integrity": "sha512-5l4VyZR86LZ/lDxZTR6jqL8AFE2S0IFLMP26AbjsLVADxHdhB/c0GUsH+y39UfCi3dzz8OlQuPmnaJOMoDHQBA==", + "dev": true, + "requires": { + "queue-microtask": "^1.2.2" + } + }, + "scheduler": { + "version": "0.23.0", + "resolved": "https://registry.npmjs.org/scheduler/-/scheduler-0.23.0.tgz", + "integrity": "sha512-CtuThmgHNg7zIZWAXi3AsyIzA3n4xx7aNyjwC2VJldO2LMVDhFK+63xGqq6CsJH4rTAt6/M+N4GhZiDYPx9eUw==", + "requires": { + "loose-envify": "^1.1.0" + } + }, + "scroll-into-view-if-needed": { + "version": "3.0.10", + "resolved": "https://registry.npmjs.org/scroll-into-view-if-needed/-/scroll-into-view-if-needed-3.0.10.tgz", + "integrity": "sha512-t44QCeDKAPf1mtQH3fYpWz8IM/DyvHLjs8wUvvwMYxk5moOqCzrMSxK6HQVD0QVmVjXFavoFIPRVrMuJPKAvtg==", + "requires": { + "compute-scroll-into-view": "^3.0.2" + } + }, + "semver": { + "version": "7.5.4", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.5.4.tgz", + "integrity": "sha512-1bCSESV6Pv+i21Hvpxp3Dx+pSD8lIPt8uVjRrxAUt/nbswYc+tK6Y2btiULjd4+fnq15PX+nqQDC7Oft7WkwcA==", + "dev": true, + "requires": { + "lru-cache": "^6.0.0" + } + }, + "shebang-command": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-2.0.0.tgz", + "integrity": "sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA==", + "dev": true, + "requires": { + "shebang-regex": "^3.0.0" + } + }, + "shebang-regex": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-3.0.0.tgz", + "integrity": "sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==", + "dev": true + }, + "slash": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/slash/-/slash-3.0.0.tgz", + "integrity": "sha512-g9Q1haeby36OSStwb4ntCGGGaKsaVSjQ68fBxoQcutl5fS1vuY18H3wSt3jFyFtrkx+Kz0V1G85A4MyAdDMi2Q==", + "dev": true + }, + "source-map-js": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/source-map-js/-/source-map-js-1.0.2.tgz", + "integrity": "sha512-R0XvVJ9WusLiqTCEiGCmICCMplcCkIwwR11mOSD9CR5u+IXYdiseeEuXCVAjS54zqwkLcPNnmU4OeJ6tUrWhDw==", + "dev": true + }, + "string-convert": { + "version": "0.2.1", + "resolved": "https://registry.npmjs.org/string-convert/-/string-convert-0.2.1.tgz", + "integrity": "sha512-u/1tdPl4yQnPBjnVrmdLo9gtuLvELKsAoRapekWggdiQNvvvum+jYF329d84NAa660KQw7pB2n36KrIKVoXa3A==" + }, + "strip-ansi": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", + "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", + "dev": true, + "requires": { + "ansi-regex": "^5.0.1" + } + }, + "strip-json-comments": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-3.1.1.tgz", + "integrity": "sha512-6fPc+R4ihwqP6N/aIv2f1gMH8lOVtWQHoqC4yK6oSDVVocumAsfCqjkXnqiYMhmMwS/mEHLp7Vehlt3ql6lEig==", + "dev": true + }, + "stylis": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/stylis/-/stylis-4.3.0.tgz", + "integrity": "sha512-E87pIogpwUsUwXw7dNyU4QDjdgVMy52m+XEOPEKUn161cCzWjjhPSQhByfd1CcNvrOLnXQ6OnnZDwnJrz/Z4YQ==" + }, + "supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "requires": { + "has-flag": "^4.0.0" + } + }, + "text-table": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/text-table/-/text-table-0.2.0.tgz", + "integrity": "sha512-N+8UisAXDGk8PFXP4HAzVR9nbfmVJ3zYLAWiTIoqC5v5isinhr+r5uaO8+7r3BMfuNIufIsA7RdpVgacC2cSpw==", + "dev": true + }, + "throttle-debounce": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/throttle-debounce/-/throttle-debounce-5.0.0.tgz", + "integrity": "sha512-2iQTSgkkc1Zyk0MeVrt/3BvuOXYPl/R8Z0U2xxo9rjwNciaHDG3R+Lm6dh4EeUci49DanvBnuqI6jshoQQRGEg==" + }, + "to-regex-range": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz", + "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==", + "dev": true, + "requires": { + "is-number": "^7.0.0" + } + }, + "toggle-selection": { + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/toggle-selection/-/toggle-selection-1.0.6.tgz", + "integrity": "sha512-BiZS+C1OS8g/q2RRbJmy59xpyghNBqrr6k5L/uKBGRsTfxmu3ffiRnd8mlGPUVayg8pvfi5urfnu8TU7DVOkLQ==" + }, + "ts-api-utils": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/ts-api-utils/-/ts-api-utils-1.0.1.tgz", + "integrity": "sha512-lC/RGlPmwdrIBFTX59wwNzqh7aR2otPNPR/5brHZm/XKFYKsfqxihXUe9pU3JI+3vGkl+vyCoNNnPhJn3aLK1A==", + "dev": true, + "requires": {} + }, + "type-check": { + "version": "0.4.0", + "resolved": "https://registry.npmjs.org/type-check/-/type-check-0.4.0.tgz", + "integrity": "sha512-XleUoc9uwGXqjWwXaUTZAmzMcFZ5858QA2vvx1Ur5xIcixXIP+8LnFDgRplU30us6teqdlskFfu+ae4K79Ooew==", + "dev": true, + "requires": { + "prelude-ls": "^1.2.1" + } + }, + "type-fest": { + "version": "0.20.2", + "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.20.2.tgz", + "integrity": "sha512-Ne+eE4r0/iWnpAxD852z3A+N0Bt5RN//NjJwRd2VFHEmrywxf5vsZlh4R6lixl6B+wz/8d+maTSAkN1FIkI3LQ==", + "dev": true + }, + "typescript": { + "version": "5.1.6", + "resolved": "https://registry.npmjs.org/typescript/-/typescript-5.1.6.tgz", + "integrity": "sha512-zaWCozRZ6DLEWAWFrVDz1H6FVXzUSfTy5FUMWsQlU8Ym5JP9eO4xkTIROFCQvhQf61z6O/G6ugw3SgAnvvm+HA==", + "dev": true + }, + "uri-js": { + "version": "4.4.1", + "resolved": "https://registry.npmjs.org/uri-js/-/uri-js-4.4.1.tgz", + "integrity": "sha512-7rKUyy33Q1yc98pQ1DAmLtwX109F7TIfWlW1Ydo8Wl1ii1SeHieeh0HHfPeL2fMXK6z0s8ecKs9frCuLJvndBg==", + "dev": true, + "requires": { + "punycode": "^2.1.0" + } + }, + "vite": { + "version": "4.4.6", + "resolved": "https://registry.npmjs.org/vite/-/vite-4.4.6.tgz", + "integrity": "sha512-EY6Mm8vJ++S3D4tNAckaZfw3JwG3wa794Vt70M6cNJ6NxT87yhq7EC8Rcap3ahyHdo8AhCmV9PTk+vG1HiYn1A==", + "dev": true, + "requires": { + "esbuild": "^0.18.10", + "fsevents": "~2.3.2", + "postcss": "^8.4.26", + "rollup": "^3.25.2" + } + }, + "which": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz", + "integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==", + "dev": true, + "requires": { + "isexe": "^2.0.0" + } + }, + "wrappy": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", + "integrity": "sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ==", + "dev": true + }, + "yallist": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/yallist/-/yallist-4.0.0.tgz", + "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==", + "dev": true + }, + "yocto-queue": { + "version": "0.1.0", + "resolved": "https://registry.npmjs.org/yocto-queue/-/yocto-queue-0.1.0.tgz", + "integrity": "sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q==", + "dev": true + } + } +} diff --git a/package.json b/package.json new file mode 100644 index 00000000..1d26310e --- /dev/null +++ b/package.json @@ -0,0 +1,39 @@ +{ + "name": "kdt5-mini", + "private": true, + "version": "0.0.0", + "type": "module", + "scripts": { + "dev": "vite", + "build": "tsc && vite build", + "lint": "eslint . --ext ts,tsx --report-unused-disable-directives --max-warnings 0", + "preview": "vite preview" + }, + "dependencies": { + "@fullcalendar/core": "^6.1.8", + "@fullcalendar/daygrid": "^6.1.8", + "@fullcalendar/interaction": "^6.1.8", + "@fullcalendar/react": "^6.1.8", + "antd": "^5.7.2", + "axios": "^1.4.0", + "dayjs": "^1.11.9", + "react": "^18.2.0", + "react-dom": "^18.2.0", + "react-router-dom": "^6.14.2", + "recoil": "^0.7.7" + }, + "devDependencies": { + "@types/node": "^20.4.4", + "@types/react": "^18.2.15", + "@types/react-dom": "^18.2.7", + "@typescript-eslint/eslint-plugin": "^6.0.0", + "@typescript-eslint/parser": "^6.0.0", + "@vitejs/plugin-react-swc": "^3.3.2", + "dotenv": "^16.3.1", + "eslint": "^8.45.0", + "eslint-plugin-react-hooks": "^4.6.0", + "eslint-plugin-react-refresh": "^0.4.3", + "typescript": "^5.0.2", + "vite": "^4.4.5" + } +} diff --git a/public/vite.svg b/public/vite.svg new file mode 100644 index 00000000..e7b8dfb1 --- /dev/null +++ b/public/vite.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/src/App.tsx b/src/App.tsx new file mode 100644 index 00000000..ebad3073 --- /dev/null +++ b/src/App.tsx @@ -0,0 +1,47 @@ +import { ConfigProvider } from 'antd'; +import { Route, Routes } from 'react-router-dom'; +import koKR from 'antd/locale/ko_KR'; +import MyLayout from '@/components/MyLayout'; +import Home from '@/page/home/home'; +import ProtectedRoute from '@/components/ProtectedRoute'; +import MyAccountLayout from '@/page/myAccount/myAccoutLayout'; +import MyAccount from '@/page/myAccount/myAccount'; +import Vacation from '@/page/myAccount/vacation'; +import ProtectedManagerRoute from '@/components/ProtectedManagerRoute'; +import AdminLayout from '@/page/myAccount/admin/adminLayout'; +import Approve from '@/page/myAccount/admin/approve'; +import Promote from '@/page/myAccount/admin/promote'; +import NotFound from '@/page/notFound/notFountd'; +import Signup from '@/page/signup/signup'; + +export default function App() { + const theme = { + token: { + colorPrimary: '#03bafc', + }, + }; + + return ( + + + } /> + }> + } /> + }> + }> + } /> + } /> + }> + }> + } /> + } /> + + + + + } /> + + + + ); +} diff --git a/src/api/auth/checkEmail.ts b/src/api/auth/checkEmail.ts new file mode 100644 index 00000000..a7e0d25b --- /dev/null +++ b/src/api/auth/checkEmail.ts @@ -0,0 +1,20 @@ +import { customAxios } from '@/api/customAxios'; + +export const checkEmail = async (userEmail: string) => { + // 추후에 /v1/auth/check-email 변경 + const response = await customAxios.post('/v1/auth/check-email', userEmail, { + headers: { 'Content-Type': 'application/json' }, + }); + return response; +}; + +export const checkEmailAuth = async (data: { + userEmail: string; + userEmailAuth: string; +}) => { + // 추후에 변경 /v1/auth/check-email-auth + const response = await customAxios.post('/v1/auth/check-email-auth', data, { + headers: { 'Content-Type': 'application/json' }, + }); + return response; +}; diff --git a/src/api/auth/cloudinary.ts b/src/api/auth/cloudinary.ts new file mode 100644 index 00000000..7b8dd6af --- /dev/null +++ b/src/api/auth/cloudinary.ts @@ -0,0 +1,20 @@ +import axios from 'axios'; + +const cloudinaryName = import.meta.env.VITE_CLOUD_NAME; + +export const handleUpload = async (file: string | Blob) => { + const formData = new FormData(); + formData.append('file', file); + formData.append('upload_preset', 'dvmzwc5k'); + + if (!file) { + return null; + } + + const response = await axios.post( + `https://api.cloudinary.com/v1_1/${cloudinaryName}/image/upload/`, + formData, + ); + + return response; +}; diff --git a/src/api/auth/signin.ts b/src/api/auth/signin.ts new file mode 100644 index 00000000..bf720b5c --- /dev/null +++ b/src/api/auth/signin.ts @@ -0,0 +1,11 @@ +import { customAxios } from '@/api/customAxios'; + +export const signin = async (loginData: { + userEmail: string; + userPassword: string; +}) => { + const response = await customAxios.post('/v2/auth/signin', loginData, { + withCredentials: true, + }); + return response; +}; diff --git a/src/api/auth/signout.ts b/src/api/auth/signout.ts new file mode 100644 index 00000000..881dc482 --- /dev/null +++ b/src/api/auth/signout.ts @@ -0,0 +1,6 @@ +import { customAxios } from '@/api/customAxios'; + +export const signout = async () => { + const response = await customAxios.post('/v1/auth/signout'); + return response; +}; diff --git a/src/api/auth/signup.ts b/src/api/auth/signup.ts new file mode 100644 index 00000000..3c1f9f3c --- /dev/null +++ b/src/api/auth/signup.ts @@ -0,0 +1,21 @@ +import { customAxios } from '@/api/customAxios'; + +interface valuseType { + confirm_password: string; + phone: string; + position: string; + profileThumbUrl: string; + userEmail: string; + userPassword: string; + userName: string; +} + +export const signup = async (values: valuseType) => { + // eslint-disable-next-line @typescript-eslint/no-unused-vars + const { confirm_password, ...otherData } = values; + + const response = await customAxios.post('/v1/auth/signup', otherData, { + withCredentials: true, + }); + return response; +}; diff --git a/src/api/auth/verification.ts b/src/api/auth/verification.ts new file mode 100644 index 00000000..32cb8226 --- /dev/null +++ b/src/api/auth/verification.ts @@ -0,0 +1,9 @@ +import { customAxios } from '@/api/customAxios'; + +export const verificationEmail = async (userEmail: string) => { + // 추후에 변경 /v1/auth/send-email + const response = await customAxios.post('/v1/auth/send-email', userEmail, { + headers: { 'Content-Type': 'application/json' }, + }); + return response; +}; diff --git a/src/api/customAxios.ts b/src/api/customAxios.ts new file mode 100644 index 00000000..e6290549 --- /dev/null +++ b/src/api/customAxios.ts @@ -0,0 +1,115 @@ +import { BASE_API_URL } from '@/data/constants'; +import { + deleteAccessTokenFromCookie, + getAccessTokenFromCookie, + setAccessTokenToCookie, +} from '@/utils/cookies'; +import axios from 'axios'; +import getPayloadFromJWT from '@/utils/getPayloadFromJWT'; + +export const customAxios = axios.create({ + baseURL: BASE_API_URL, + timeout: 7000, // 5초간 아무 응답이 없으면 취소 +}); + +customAxios.interceptors.request.use( + async (req) => { + const accessToken = getAccessTokenFromCookie(); + // accessToken이 없는 경우 === 로그아웃을 한 상태에서하는 요청들 : 회원가입, 이메일중복체크, 로그인), 그런데 있어도 되는듯 + if (!accessToken) { + return req; + } + + // 로그인 요청, 회원가입 등... 을 제외한 모든 요청은 accessToken을 필요로함 + req.headers.Authorization = `Bearer ${accessToken}`; + + // // 만료시간 표시 + // access토큰의 만료시간을 초로 나타낸 시간 + const expirationTime = getPayloadFromJWT(accessToken).exp as number; + // 현재시간을 초로 나타냄 + const currentTime = Math.floor(new Date().getTime() / 1000); + // 만료시간 > 5분 남은 경우 + const expirationLeft = expirationTime - currentTime; + console.log(`만료 ${Math.floor(expirationLeft / 60)}분 남았습니다.`); + + return req; + }, + (error) => { + return Promise.reject(error); + }, +); + +// 여러개의 요청이 밀렸을 경우 리프레시토큰 api가 여러번 실행되는 것을 막음 +let isRefreshing = false; + +// 만료된 토큰으로 인해 pending상태가 된 기존의 요청들을 배열에 담음, 새로운 토큰이 발행되면 이들 요청을 진행 +// eslint-disable-next-line no-unused-vars +let refreshSubscribers: ((accessToken: string) => void)[] = []; + +customAxios.interceptors.response.use( + (response) => { + return response; + }, + async (error) => { + const status = error.response?.data.error.status; + + if (status === 401) { + const refreshToken = localStorage.getItem('refreshToken'); + + if (!refreshToken) { + return Promise.reject(error); + } + + if (!isRefreshing) { + isRefreshing = true; + + try { + const response = await axios.post( + `${BASE_API_URL}/v2/auth/refresh-token`, + { + refreshToken, + }, + ); + + if (response.status === 200) { + localStorage.setItem( + 'refreshToken', + response.data.response.refreshToken, + ); + setAccessTokenToCookie(response.data.response.accessToken); + + const config = error.config; + config.headers.Authorization = `Bearer ${response.data.response.accessToken}`; + + const retryOriginalRequest = new Promise((resolve) => { + resolve(axios(config)); + }); + + isRefreshing = false; + + refreshSubscribers.forEach((callback) => + callback(response.data.response.accessToken), + ); + refreshSubscribers = []; + + return retryOriginalRequest; + } + } catch (error) { + deleteAccessTokenFromCookie(); + localStorage.removeItem('refreshToken'); + isRefreshing = false; + return Promise.reject(error); + } + } else { + return new Promise((resolve) => { + refreshSubscribers.push((accessToken: string) => { + error.config.headers.Authorization = `Bearer ${accessToken}`; + resolve(axios(error.config)); + }); + }); + } + } else { + return Promise.reject(error); + } + }, +); diff --git a/src/api/home/getUserHeader.ts b/src/api/home/getUserHeader.ts new file mode 100644 index 00000000..99b66d08 --- /dev/null +++ b/src/api/home/getUserHeader.ts @@ -0,0 +1,6 @@ +import { customAxios } from '@/api/customAxios'; + +export const getUserHeader = async () => { + const response = await customAxios('/v1/user/userHeader'); + return response; +}; diff --git a/src/api/home/pendingList.ts b/src/api/home/pendingList.ts new file mode 100644 index 00000000..7f69e1dd --- /dev/null +++ b/src/api/home/pendingList.ts @@ -0,0 +1,8 @@ +import { customAxios } from '@/api/customAxios'; + +export const pendingList = async (year: number) => { + const response = await customAxios( + `/v1/user/schedule/pending-list?year=${year}`, + ); + return response; +}; diff --git a/src/api/home/scheduleList.ts b/src/api/home/scheduleList.ts new file mode 100644 index 00000000..48759e64 --- /dev/null +++ b/src/api/home/scheduleList.ts @@ -0,0 +1,6 @@ +import { customAxios } from '@/api/customAxios'; + +export const scheduleList = async (year: number) => { + const response = await customAxios(`/v1/schedule/list?year=${year}`); + return response; +}; diff --git a/src/api/myAccount/admin.ts b/src/api/myAccount/admin.ts new file mode 100644 index 00000000..34de7f61 --- /dev/null +++ b/src/api/myAccount/admin.ts @@ -0,0 +1,32 @@ +import { customAxios } from '@/api/customAxios'; + +export const getVacationRequests = async () => { + const response = await customAxios.get('/v1/admin/list', {}); + return response; +}; + +export const approveRejectPending = async ( + id: number, + type: 'APPROVE' | 'REJECT' | 'PENDING', +) => { + const response = await customAxios.post(`/v1/admin/${type.toLowerCase()}`, { + id, + }); + return response; +}; + +export const getWorkers = async () => { + const response = await customAxios.get('/v1/admin/worker-list', {}); + return response; +}; + +export const changePosition = async ( + id: number, + position: 'LEVEL1' | 'LEVEL2' | 'LEVEL3' | 'LEVEL4', +) => { + const response = await customAxios.post(`/v1/admin/change-position`, { + id, + position, + }); + return response; +}; diff --git a/src/api/myAccount/changeMyInfo.ts b/src/api/myAccount/changeMyInfo.ts new file mode 100644 index 00000000..118491e2 --- /dev/null +++ b/src/api/myAccount/changeMyInfo.ts @@ -0,0 +1,10 @@ +import { customAxios } from '@/api/customAxios'; + +export const changeMyInfo = async (data: { + userPassword?: string; + phoneNumber?: string; + profileThumbUrl?: string; +}) => { + const response = await customAxios.patch('/v1/user/info', data, {}); + return response; +}; diff --git a/src/api/myAccount/getMyAccount.ts b/src/api/myAccount/getMyAccount.ts new file mode 100644 index 00000000..360646d2 --- /dev/null +++ b/src/api/myAccount/getMyAccount.ts @@ -0,0 +1,7 @@ +import { customAxios } from '@/api/customAxios'; + +export const getMyAccount = async () => { + const response = await customAxios('/v1/user/info'); + + return response; +}; diff --git a/src/api/mySchedule.ts b/src/api/mySchedule.ts new file mode 100644 index 00000000..452c4c3d --- /dev/null +++ b/src/api/mySchedule.ts @@ -0,0 +1,23 @@ +import { customAxios } from '@/api/customAxios'; + +export const addScheduleRequest = async (scheduleData: { + scheduleType: string; + startDate: string; + endDate: string; +}) => { + const response = await customAxios.post( + '/v1/user/schedule/add', + scheduleData, + ); + return response; +}; + +export const cancelScheduleRequest = async (id: number) => { + const response = await customAxios.post('/v1/user/schedule/cancel', { id }); + return response; +}; + +export const getMySchedule = async (year: number) => { + const response = await customAxios(`/v1/user/schedule?year=${year}`); + return response; +}; diff --git a/src/assets/defaultProfile.png b/src/assets/defaultProfile.png new file mode 100644 index 0000000000000000000000000000000000000000..e172ffd4d0399d67ae972181c82e5cd27e216fe9 GIT binary patch literal 45192 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

PC*lziZcG?(ceG4;MTgzpNm_ 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 literal 0 HcmV?d00001 diff --git a/src/components/MyHeader.tsx b/src/components/MyHeader.tsx new file mode 100644 index 00000000..6af6b0ba --- /dev/null +++ b/src/components/MyHeader.tsx @@ -0,0 +1,175 @@ +import { signout } from '@/api/auth/signout'; +import { AccessTokenAtom } from '@/recoil/AccessTokkenAtom'; +import { Button, Skeleton, Space, message, theme } from 'antd'; +import { Header } from 'antd/es/layout/layout'; +import { useState, useEffect } from 'react'; +import { Link } from 'react-router-dom'; +import { useRecoilState, useSetRecoilState, useRecoilValue } from 'recoil'; +import UserInfo from '@/components/UserInfo'; +import { deleteAccessTokenFromCookie } from '@/utils/cookies'; +import { IsManagerAtom } from '@/recoil/IsManagerAtom'; +import { getUserHeader } from '@/api/home/getUserHeader'; +import { ReRenderStateAtom } from '@/recoil/ReRenderStateAtom'; +import { UserEmailAtom } from '@/recoil/UserEmailAtom'; + +export default function MyHeader() { + // antd theme + const { + token: { colorPrimaryBg }, + } = theme.useToken(); + + // antd message(화면 상단에 뜨는 메세지)기능 + const [messageApi, contextHolder] = message.useMessage(); + + // 리코일 전역 access토큰 + const [accessToken, setAccessToken] = useRecoilState(AccessTokenAtom); + + const reRender = useRecoilValue(ReRenderStateAtom); + + // **네브바에 있는 유저 정보 GET요청** + // 네브바에 표시될 유저 정보들 + const [userHeaderInfo, setUserHeaderInfo] = useState({ + userName: '', + profileThumbNail: '', + position: '', + usedVacation: '', + }); + + // 매니저여부, 사용자 이메일 set하는 함수 + const setIsManager = useSetRecoilState(IsManagerAtom); + const setUserEmail = useSetRecoilState(UserEmailAtom); + + // 통신 loading + const [isMyHeaderLoading, setIsMyHeaderLoading] = useState(false); + + useEffect(() => { + const getData = async () => { + if (!accessToken) { + return; + } + try { + setIsMyHeaderLoading(true); + const response = await getUserHeader(); + if (response.status === 200) { + const userData = response.data.response; + setUserHeaderInfo({ + profileThumbNail: userData.profileThumbNail, + usedVacation: userData.usedVacation.toString(), + userName: userData.userName, + position: userData.position, + }); + // 헤더정보는 항상 노출이 되는 부분이기 때문에 관리자 여부, 사용자 이메일을 여기에서 세팅해줌 + setIsManager(userData.position === 'MANAGER'); + setUserEmail(userData.userEmail); + return; + } + // eslint-disable-next-line @typescript-eslint/no-explicit-any + } catch (error: any) { + console.log('헤더 유저정보 로딩 중 에러 발생:', error); + } finally { + setIsMyHeaderLoading(false); + } + }; + getData(); + // eslint-disable-next-line react-hooks/exhaustive-deps + }, [accessToken, reRender]); + + // 로그아웃 통신 로딩 ui + const [isSigningout, setIsSigningout] = useState(false); + + const handleSignout = async () => { + try { + setIsSigningout(true); + await signout(); + } catch (error) { + console.log('로그아웃 중 에러발생 : ', error); + } finally { + // 로그아웃은 통신이 성공하든 실패하든 상관없이 토큰을 삭제해주면 된다. + // 그러면 통신을 왜하냐고 물어볼 수 있는데 서버쪽에서 refresh 토큰을 삭제하기 위해서라고 함 + + // 쿠키에서 삭제 + deleteAccessTokenFromCookie(); + + // recoil 초기화 + setAccessToken(null); + + // 로딩 ui종료 + setIsSigningout(false); + + // 로컬저장소에서 리프레시토큰 삭제 + localStorage.removeItem('refreshToken'); + + // 오류가 났다고 하더라도 로그아웃 성공메세지를 보여줌 + messageApi.open({ + type: 'success', + content: '로그아웃이 완료 되었습니다.', // 서버에서 오는 성공메세지와 동일 + }); + } + }; + + return ( + <> + {contextHolder} +

+
+ + 🏠 + + {accessToken ? ( + + {isMyHeaderLoading ? ( +
+ + +
+ ) : ( + + )} + + +
+ ) : ( + <> + )} +
+
+ + ); +} diff --git a/src/components/MyLayout.tsx b/src/components/MyLayout.tsx new file mode 100644 index 00000000..fa8eb3e4 --- /dev/null +++ b/src/components/MyLayout.tsx @@ -0,0 +1,15 @@ +import { Layout } from 'antd'; +import { Content } from 'antd/es/layout/layout'; +import { Outlet } from 'react-router-dom'; +import MyHeader from '@/components/MyHeader'; + +export default function MyLayout() { + return ( + + + + + + + ); +} diff --git a/src/components/ProtectedManagerRoute.tsx b/src/components/ProtectedManagerRoute.tsx new file mode 100644 index 00000000..0ec8da09 --- /dev/null +++ b/src/components/ProtectedManagerRoute.tsx @@ -0,0 +1,12 @@ +import { IsManagerAtom } from '@/recoil/IsManagerAtom'; +import { Navigate, Outlet } from 'react-router-dom'; +import { useRecoilValue } from 'recoil'; + +export default function ProtectedManagerRoute() { + const isManager = useRecoilValue(IsManagerAtom); + + if (isManager) { + return ; + } + return ; +} diff --git a/src/components/ProtectedRoute.tsx b/src/components/ProtectedRoute.tsx new file mode 100644 index 00000000..5f00cd9d --- /dev/null +++ b/src/components/ProtectedRoute.tsx @@ -0,0 +1,12 @@ +import { AccessTokenAtom } from '@/recoil/AccessTokkenAtom'; +import { Navigate, Outlet } from 'react-router-dom'; +import { useRecoilValue } from 'recoil'; + +export default function ProtectedRoute() { + const accessToken = useRecoilValue(AccessTokenAtom); + + if (accessToken) { + return ; + } + return ; +} diff --git a/src/components/RequesTag.tsx b/src/components/RequesTag.tsx new file mode 100644 index 00000000..c4b08e69 --- /dev/null +++ b/src/components/RequesTag.tsx @@ -0,0 +1,23 @@ +import { REQUEST_STATE } from '@/data/constants'; +import { Tag } from 'antd'; + +export default function RequesTag({ + small, + state, +}: { + small?: boolean; + state: 'APPROVE' | 'REJECT' | 'PENDING'; +}) { + return ( + + {REQUEST_STATE[state]?.label} + + ); +} diff --git a/src/components/UserInfo.tsx b/src/components/UserInfo.tsx new file mode 100644 index 00000000..6c25de10 --- /dev/null +++ b/src/components/UserInfo.tsx @@ -0,0 +1,41 @@ +import { Badge, Typography, Avatar } from 'antd'; +import { Link } from 'react-router-dom'; +import { POSITIONS } from '@/data/constants'; + +const { Text } = Typography; + +export default function UserInfo({ + userHeaderInfo, +}: { + userHeaderInfo: { + userName: string; + profileThumbNail: string; + position: string; + usedVacation: string; + }; +}) { + return ( +
+ + {userHeaderInfo.userName} + + + 남은연차{' '} + {POSITIONS[userHeaderInfo?.position]?.total_vacation - + Number(userHeaderInfo.usedVacation)} + 일 + +
+ ); +} diff --git a/src/data/constants.ts b/src/data/constants.ts new file mode 100644 index 00000000..781ec6c1 --- /dev/null +++ b/src/data/constants.ts @@ -0,0 +1,48 @@ +export const BASE_API_URL = 'https://prettyawesome.duckdns.org:8080/api'; + +export const EMAIL_REGEX = /^[a-zA-Z0-9._%+-]+@[a-zA-Z0-9.-]+\.[a-zA-Z]{2,}$/; + +export const PASSWORD_REGEX = + /^(?=.*\d)(?=.*[!@#$%^&*()-+=])(?=.*[a-zA-Z]).{8,16}$/; + +export const POSITIONS: { + [key: string]: { label: string; total_vacation: number; color: string }; +} = { + LEVEL1: { + label: '레벨1', + total_vacation: 15, + color: 'lime', + }, + LEVEL2: { + label: '레벨2', + total_vacation: 18, + color: 'geekblue', + }, + LEVEL3: { + label: '레벨3', + total_vacation: 21, + color: 'magenta', + }, + LEVEL4: { + label: '레벨4', + total_vacation: 24, + color: 'volcano', + }, + MANAGER: { + label: '매니저', + total_vacation: 27, + color: 'gold', + }, +}; + +export const REQUEST_STATE = { + PENDING: { label: '심사중', color: '#f2cf50' }, + APPROVE: { label: '승인', color: '#50dcf2' }, + REJECT: { label: '거절', color: '#f26650' }, +}; +export const DUTY_ANNUAL: { + [key: string]: { label: string; color: string }; +} = { + DUTY: { label: '당직', color: '#f08080' }, + ANNUAL: { label: '연차', color: '#b1aee5' }, +}; diff --git a/src/data/dummyData.ts b/src/data/dummyData.ts new file mode 100644 index 00000000..ec1c48bc --- /dev/null +++ b/src/data/dummyData.ts @@ -0,0 +1,58 @@ +interface DataType { + key: number; + id: number; + profileThumbUrl: string; + userName: string; + position: 'LEVEL1' | 'LEVEL2' | 'LEVEL3' | 'LEVEL4'; + createAt: string; +} + +export const DUMMY_WORKERS: DataType[] = [ + { + key: 1, + id: 1, + userName: 'Kim', + profileThumbUrl: 'imageurl', + position: 'LEVEL1', + createAt: '2021-08-03', + }, + { + id: 2, + key: 2, + userName: 'lee', + profileThumbUrl: 'imageurl', + position: 'LEVEL2', + createAt: '2021-08-03', + }, + { + id: 3, + key: 3, + userName: 'park', + profileThumbUrl: 'imageurl', + position: 'LEVEL3', + createAt: '2021-08-03', + }, +]; + +export const DUMMY_MY_SCHEDULES = [ + { + scheduleType: 'ANNUAL', + state: 'REJECT', + startDate: '2023-08-01', + endDate: '2023-08-05', + }, + + { + scheduleType: 'ANNUAL', + state: 'PENDING', + startDate: '2023-08-03', + endDate: '2023-08-05', + }, + + { + scheduleType: 'DUTY', + state: 'APPROVE', + startDate: '2023-08-01', + endDate: '2023-08-01', + }, +]; diff --git a/src/index.css b/src/index.css new file mode 100644 index 00000000..7b16ef62 --- /dev/null +++ b/src/index.css @@ -0,0 +1,50 @@ +.profile_image img { + object-fit: cover; +} + +.icons { + cursor: pointer; + transition: all 0.2s ease-in-out; +} + +.icons:hover { + scale: 1.2; +} + +/* 달력 평일, 토요일, 일요일 색상 변경 */ +.fc-day-sun a { + color: #ff4d4f; +} + +.fc-day-sat a { + color: #69b1ff; +} + +.fc-day-mon a, +.fc-day-tue a, +.fc-day-wed a, +.fc-day-thu a, +.fc-day-fri a { + color: #595959; +} + +.fc-day { + font-weight: bold; +} + +.myScheduleRow { + font-size: 10px; +} + +.ant-table-wrapper .ant-table-thead > tr > th { + text-align: center; +} + +.booh { + display: none; +} +@media screen and (min-width: 1000px) { + .booh { + display: block; + } +} diff --git a/src/main.tsx b/src/main.tsx new file mode 100644 index 00000000..16103fbd --- /dev/null +++ b/src/main.tsx @@ -0,0 +1,16 @@ +import React from 'react'; +import ReactDOM from 'react-dom/client'; +import App from '@/App.tsx'; +import '@/index.css'; +import { BrowserRouter } from 'react-router-dom'; +import { RecoilRoot } from 'recoil'; + +ReactDOM.createRoot(document.getElementById('root')!).render( + + + + + + + , +); diff --git a/src/page/home/calendar.tsx b/src/page/home/calendar.tsx new file mode 100644 index 00000000..515f6b71 --- /dev/null +++ b/src/page/home/calendar.tsx @@ -0,0 +1,147 @@ +import { useState, useRef, Dispatch } from 'react'; +import FullCalendar from '@fullcalendar/react'; +import dayGridPlugin from '@fullcalendar/daygrid'; +import interactionPlugin from '@fullcalendar/interaction'; +import { Switch, Button, Space, Typography, Tooltip, Skeleton } from 'antd'; +import { LeftOutlined, RightOutlined } from '@ant-design/icons'; +import { ScheduleItem } from './home'; + +interface propsType { + mySchedule: ScheduleItem[]; + events: ScheduleItem[]; + year: number; + setYear: Dispatch>; + userYearlySchedulesLoading: boolean; +} + +export default function Calendar({ + mySchedule, + year, + setYear, + events, + userYearlySchedulesLoading, +}: propsType) { + // 데이터로 받아올 events를 상태관리 + // eslint-disable-next-line react-hooks/rules-of-hooks + + // 달력의 현재 월 상태관리 + const [month, setMonth] = useState(new Date().getMonth() + 1); + // switch 체크 상태관리 + const [isAllChecked, setIsAllChecked] = useState(true); + + // eslint-disable-next-line react-hooks/rules-of-hooks + const calendarRef = useRef(null); + + const { Title } = Typography; + + // eslint-disable-next-line @typescript-eslint/no-explicit-any + const renderDayCellContent = (args: any) => { + // '일' 문자 제거 + return args.date.getDate(); + }; + + // eslint-disable-next-line @typescript-eslint/no-explicit-any + const handleDateSet = (info: any) => { + const date = info.view.currentStart; + setYear(date.getFullYear()); + setMonth(date.getMonth() + 1); + }; + + const goPrev = () => { + const calendarApi = calendarRef.current?.getApi(); + calendarApi?.prev(); + }; + + const goNext = () => { + const calendarApi = calendarRef.current?.getApi(); + calendarApi?.next(); + }; + + const goToday = () => { + const calendarApi = calendarRef.current?.getApi(); + calendarApi?.today(); + }; + + return ( + <> +
+
+ + setIsAllChecked(check)} + loading={userYearlySchedulesLoading} + /> + +
+ +
+ + + + + {year}년 {month}월 + + + + +
+
+
+
+ {userYearlySchedulesLoading ? ( + + ) : ( + + )} +
+ + ); +} diff --git a/src/page/home/home.tsx b/src/page/home/home.tsx new file mode 100644 index 00000000..7a7955d2 --- /dev/null +++ b/src/page/home/home.tsx @@ -0,0 +1,371 @@ +import { AccessTokenAtom } from '@/recoil/AccessTokkenAtom'; +import { + DatePicker, + Layout, + Modal, + Button, + Select, + Space, + message, +} from 'antd'; +import { useRecoilValue, useSetRecoilState } from 'recoil'; +import Sider from 'antd/es/layout/Sider'; +import { Content } from 'antd/es/layout/layout'; +import { useEffect, useState } from 'react'; +import dayjs from 'dayjs'; +import Calendar from './calendar'; +import Signin from '@/page/home/signin'; +import { DatePickerProps, RangePickerProps } from 'antd/es/date-picker'; +import { addScheduleRequest } from '@/api/mySchedule'; +import MySchedule from '@/page/home/mySchedule'; +import { DUTY_ANNUAL } from '@/data/constants'; +import { ReRenderStateAtom } from '@/recoil/ReRenderStateAtom'; +import { scheduleList } from '@/api/home/scheduleList'; +import { UserEmailAtom } from '@/recoil/UserEmailAtom'; +import { pendingList } from '@/api/home/pendingList'; + +const { RangePicker } = DatePicker; + +export interface ScheduleItem { + userEmail: string; + userName: string; + scheduleType: string; + startDate: string; + endDate: string; + state: string; +} + +interface mySchedule extends ScheduleItem { + id: number; +} + +export default function Home() { + // antd message(화면 상단에 뜨는 메세지)기능 + + const [messageApi, contextHolder] = message.useMessage(); + + const [year, setYear] = useState(new Date().getFullYear()); + // const { + // token: { colorTextLabel }, + // } = theme.useToken(); + + const accessToken = useRecoilValue(AccessTokenAtom); + + const [isModalOpen, setIsModalOpen] = useState(!accessToken); + const setReRender = useSetRecoilState(ReRenderStateAtom); + const reRender = useRecoilValue(ReRenderStateAtom); + // 로그아웃을 하면 isModalOpen이 !accessToken의 상태를 바로 반영하지 않음 + // 따라서 useEffect로 반영이 되도록함 + useEffect(() => { + setIsModalOpen(!accessToken); + }, [accessToken]); + + const [scheduleInput, setScheduleInput] = useState<{ + scheduleType: string; + startDate: string; + endDate: string; + }>({ + scheduleType: '', + startDate: '', + endDate: '', + }); + + const userEmail = useRecoilValue(UserEmailAtom); + + const [events, setEvents] = useState([]); + + const [sideMySchedule, setSideMyschedule] = useState< + { + id: number; + key: number; + scheduleType: 'ANNUAL' | 'DUTY'; + startDate: string; + endDate: string; + state: 'REJECT' | 'APPROVE' | 'PENDING'; + }[] + >([]); + + const [myPendingScheduleList, setMyPendingScheduleList] = useState< + { + id: number; + key: number; + scheduleType: 'ANNUAL' | 'DUTY'; + startDate: string; + endDate: string; + state: 'PENDING'; + }[] + >([]); + + const [userYearlySchedulesLoading, setUserYearlySchedulesLoading] = + useState(false); + + const [pendingLoading, setPendingLoading] = useState(false); + useEffect(() => { + const getUsersYearlySchedules = async () => { + if (!accessToken) { + return; + } + try { + setUserYearlySchedulesLoading(true); + const listResponse = await scheduleList(year); + const listResponseData = listResponse.data.response; + + const sideMyScheduleData = listResponseData + .filter((item: mySchedule) => item.userEmail === userEmail) + .map((item: mySchedule) => { + return { + id: item.id, + key: item.id, + scheduleType: item.scheduleType, + startDate: item.startDate, + endDate: item.endDate, + state: item.state, + }; + }); + setSideMyschedule(sideMyScheduleData); + + const events = listResponseData.map((item: ScheduleItem) => { + const adjustEndDate = dayjs(item.endDate) + .add(1, 'day') + .format('YYYY-MM-DD'); + return { + userEmail: item.userEmail, + title: item.userName, + start: item.startDate, + end: adjustEndDate, + color: DUTY_ANNUAL[item.scheduleType].color, + }; + }); + setEvents(events); + } catch (error) { + console.log(error); + } finally { + setUserYearlySchedulesLoading(false); + } + }; + getUsersYearlySchedules(); + }, [year, accessToken, userEmail]); + + useEffect(() => { + const myPendingSchedule = async () => { + if (!accessToken) { + return; + } + try { + setPendingLoading(true); + const response = await pendingList(year); + const responseData = response.data.response; + + const myPendingScheduleData = responseData.map((item: mySchedule) => { + return { + id: item.id, + key: item.id, + scheduleType: item.scheduleType, + startDate: item.startDate, + endDate: item.endDate, + state: item.state, + }; + }); + setMyPendingScheduleList(myPendingScheduleData); + } catch (error) { + console.error(error); + } finally { + setPendingLoading(false); + } + }; + myPendingSchedule(); + }, [year, reRender, accessToken]); + + const handleSelect = (value: string) => { + setScheduleInput({ + startDate: '', + endDate: '', + scheduleType: value, + }); + }; + + const handleRangePicker = (value: RangePickerProps['value']) => { + const startDate = value && value[0]; + const endDate = value && value[1]; + setScheduleInput((prev) => ({ + ...prev, + startDate: dayjs(startDate).format('YYYY-MM-DD'), + endDate: dayjs(endDate).format('YYYY-MM-DD'), + })); + }; + + const handleDatePicker = (value: DatePickerProps['value']) => { + const startDate = dayjs(value).format('YYYY-MM-DD'); + setScheduleInput((prev) => ({ + ...prev, + startDate: dayjs(startDate).format('YYYY-MM-DD'), + endDate: dayjs(startDate).format('YYYY-MM-DD'), + })); + }; + + const [isAddingRequest, setIsAddingRequest] = useState(false); + + const handleSubmitSchedule = async () => { + if (!accessToken) { + return; + } + try { + setIsAddingRequest(true); + const response = await addScheduleRequest(scheduleInput); + if (response.status === 200) { + messageApi.open({ + type: 'success', + content: `${ + DUTY_ANNUAL[ + response.data.response.scheduleType as 'ANNUAL' | 'DUTY' + ]?.label + } 신청 완료`, + }); + setReRender((prev) => !prev); + } + // eslint-disable-next-line @typescript-eslint/no-explicit-any + } catch (error: any) { + messageApi.open({ + type: 'error', + content: + error.response?.data.error.message || + '연차, 당직 신청 등록 중 오류가 발생하였습니다.', + }); + } finally { + setIsAddingRequest(false); + } + }; + + const pastDates = (current: dayjs.Dayjs) => { + return current < dayjs().startOf('day'); + }; + + const mySchedule = events.filter((event) => event.userEmail === userEmail); + console.log(events); + return ( + <> + {contextHolder} + + {/* 로그인 창 */} + + + + + + +
+
+ +
+
+ +
+ + + + + + + + + 아직 회원가입을 하지 않으셨나요? + 회원가입 + + +
+ ); +} diff --git a/src/page/myAccount/admin/adminLayout.tsx b/src/page/myAccount/admin/adminLayout.tsx new file mode 100644 index 00000000..138c2457 --- /dev/null +++ b/src/page/myAccount/admin/adminLayout.tsx @@ -0,0 +1,32 @@ +import { ArrowUpOutlined, CheckOutlined } from '@ant-design/icons'; +import type { MenuProps } from 'antd'; +import { Menu } from 'antd'; +import { Outlet, useLocation, useNavigate } from 'react-router-dom'; + +export default function AdminLayout() { + const navigate = useNavigate(); + + const { pathname } = useLocation(); + + const items: MenuProps['items'] = [ + { + label: '연차 / 당직 승인', + key: '/myaccount/approve', + icon: , + onClick: () => navigate('/myaccount/approve'), + }, + { + label: '사원 직책 변경', + key: '/myaccount/promote', + icon: , + onClick: () => navigate('/myaccount/promote'), + }, + ]; + + return ( + <> + + + + ); +} diff --git a/src/page/myAccount/admin/approve.tsx b/src/page/myAccount/admin/approve.tsx new file mode 100644 index 00000000..d2676143 --- /dev/null +++ b/src/page/myAccount/admin/approve.tsx @@ -0,0 +1,222 @@ +import { + approveRejectPending, + getVacationRequests, +} from '@/api/myAccount/admin'; +import RequesTag from '@/components/RequesTag'; +import { DUTY_ANNUAL, POSITIONS } from '@/data/constants'; +import { AccessTokenAtom } from '@/recoil/AccessTokkenAtom'; +import { Badge, Button, Space, Table, message } from 'antd'; +import type { ColumnsType } from 'antd/es/table'; +import { useEffect, useState } from 'react'; +import { useRecoilValue } from 'recoil'; + +interface VacationRequestType { + key: number; + id: number; + userName: string; + position: 'LEVEL1' | 'LEVEL2' | 'LEVEL3' | 'LEVEL4'; + type: 'ANNUAL' | 'DUTY'; + startDate: string; + endDate: string; + state: 'PENDING' | 'APPROVE' | 'REJECT'; +} + +export default function Approve() { + // antd message(화면 상단에 뜨는 메세지)기능 + const [messageApi, contextHolder] = message.useMessage(); + + const [vacationRequests, setVacationRequests] = useState< + VacationRequestType[] + >([]); + + const [isvacationRequestsLoading, setIsvacationRequestsLoading] = + useState(false); + const [isAppoving, setIsApproving] = useState(false); + + const accessToken = useRecoilValue(AccessTokenAtom); + + useEffect(() => { + setIsvacationRequestsLoading(true); + const getData = async () => { + if (!accessToken) { + return; + } + try { + const response = await getVacationRequests(); + if (response.status === 200) { + const vacationRequestsData = response.data + .response as VacationRequestType[]; + // 성공했을때 + setVacationRequests( + vacationRequestsData.map((el) => { + return { ...el, key: el.id }; + }), + ); + return; + } + // eslint-disable-next-line @typescript-eslint/no-explicit-any + } catch (error: any) { + console.log('연가/당직 데이터를 불러오는 중 에러발생 : ', error); + } finally { + setIsvacationRequestsLoading(false); + } + }; + getData(); + }, [accessToken]); + + // 승인 + const handleRequest = async ( + id: number, + type: 'APPROVE' | 'REJECT' | 'PENDING', + ) => { + setIsApproving(true); + try { + const response = await approveRejectPending(id, type); + if (response.status === 200) { + setVacationRequests( + vacationRequests.map((request) => + request.id === id ? { ...request, state: type } : request, + ), + ); + messageApi.open({ + type: 'success', + content: response.data.response, + }); + } + // eslint-disable-next-line @typescript-eslint/no-explicit-any + } catch (error: any) { + console.log(error); + messageApi.open({ + type: 'error', + content: + error.response?.data.error.message || + '연차/당직 승인, 거절, 취소 실패', + }); + } finally { + setIsApproving(false); + } + }; + + const columns: ColumnsType = [ + { + title: '사원', + dataIndex: 'userName', + key: 'userName', + render: (_, { userName, position }) => ( + <> + + {userName} + + ), + }, + + { + title: '연차/당직', + dataIndex: 'type', + key: 'type', + align: 'center', + render: (_, { type }) => <>{DUTY_ANNUAL[type].label}, + }, + { + title: '시작일', + dataIndex: 'startDate', + key: 'startDate', + align: 'center', + sorter: (a, b) => + Number(a.startDate.replaceAll('-', '')) - + Number(b.startDate.replaceAll('-', '')), + render: (_, { startDate }) => <>{startDate.slice(0, 10)}, + }, + + { + title: '종료일', + dataIndex: 'endDate', + key: 'endDate', + align: 'center', + sorter: (a, b) => + Number(a.endDate.replaceAll('-', '')) - + Number(b.endDate.replaceAll('-', '')), + render: (_, { endDate }) => <>{endDate.slice(0, 10)}, + }, + + { + title: '승인여부', + key: 'tags', + dataIndex: 'tags', + align: 'center', + render: (_, { state }) => , + filters: [ + { + text: '심사중', + value: 'PENDING', + }, + { + text: '승인', + value: 'APPROVE', + }, + { + text: '거절', + value: 'REJECT', + }, + ], + onFilter: (value: string | number | boolean, record) => + record.state.includes(value as string), + }, + { + title: 'Action', + key: 'action', + align: 'center', + render: (_, { id, state }) => ( + + {state === 'PENDING' ? ( + <> + + + + ) : ( + + )} + + ), + }, + ]; + + return ( + <> + {contextHolder} + + + ); +} diff --git a/src/page/myAccount/admin/promote.tsx b/src/page/myAccount/admin/promote.tsx new file mode 100644 index 00000000..55714165 --- /dev/null +++ b/src/page/myAccount/admin/promote.tsx @@ -0,0 +1,131 @@ +import { POSITIONS } from '@/data/constants'; +import { Avatar, Badge, Button, Table } from 'antd'; +import type { ColumnsType } from 'antd/es/table'; +import PromotionModal from './promotionModal'; +import { useEffect, useState } from 'react'; +import { AccessTokenAtom } from '@/recoil/AccessTokkenAtom'; +import { useRecoilValue } from 'recoil'; +import { getWorkers } from '@/api/myAccount/admin'; + +interface WorkerType { + id: number; + key: number; + profileThumbUrl: string; + userName: string; + position: 'LEVEL1' | 'LEVEL2' | 'LEVEL3' | 'LEVEL4' | 'MANAGER'; + createAt: string; +} + +export default function Promote() { + const accessToken = useRecoilValue(AccessTokenAtom); + const [isWorkerListLoading, setIsWorkerListLoading] = useState(false); + const [isModalOpen, setIsModalOpen] = useState(false); + const [workers, setWorkers] = useState([]); + const [selectedWorker, setSelectedWorker] = useState<{ + userName: string; + id: number; + position: string; + }>(); + + useEffect(() => { + setIsWorkerListLoading(true); + const getData = async () => { + if (!accessToken) { + return; + } + try { + const response = await getWorkers(); + if (response.status === 200) { + const workersData = response.data.response as WorkerType[]; + // 성공했을때 + setWorkers( + workersData.map((el) => { + return { ...el, key: el.id }; + }), + ); + return; + } + // eslint-disable-next-line @typescript-eslint/no-explicit-any + } catch (error: any) { + console.log('사원데이터를 불러오지 못했습니다. : ', error); + } finally { + setIsWorkerListLoading(false); + } + }; + getData(); + }, [accessToken, selectedWorker]); + + const columns: ColumnsType = [ + { + title: '사진', + dataIndex: 'profileThumbUrl', + key: 'profileThumbUrl', + render: (_, { profileThumbUrl }) => , + align: 'center', + }, + { + title: '사원', + dataIndex: 'userName', + key: 'userName', + align: 'center', + }, + { + title: '직책', + dataIndex: 'position', + key: 'position', + render: (_, { position }) => ( + + ), + align: 'center', + }, + + { + title: '입사일', + key: 'createAt', + dataIndex: 'createAt', + align: 'center', + sorter: (a, b) => + Number(a.createAt.replaceAll('-', '')) - + Number(b.createAt.replaceAll('-', '')), + }, + { + title: 'Action', + key: 'action', + render: (_, { userName, position, id }) => ( + <> + {position === 'MANAGER' ? ( + <> + ) : ( + + )} + + ), + align: 'center', + }, + ]; + + return ( + <> +
+ + + ); +} diff --git a/src/page/myAccount/admin/promotionModal.tsx b/src/page/myAccount/admin/promotionModal.tsx new file mode 100644 index 00000000..5db093f7 --- /dev/null +++ b/src/page/myAccount/admin/promotionModal.tsx @@ -0,0 +1,145 @@ +import { changePosition } from '@/api/myAccount/admin'; +import { POSITIONS } from '@/data/constants'; +import { AccessTokenAtom } from '@/recoil/AccessTokkenAtom'; +import { Badge, Modal, Select, Space, message, Typography } from 'antd'; +import { useState } from 'react'; +import { useRecoilValue } from 'recoil'; + +const { Title } = Typography; + +interface PromotionModalProps { + isModalOpen: boolean; + selectedWorker?: { userName: string; id: number; position: string }; + setIsModalOpen: React.Dispatch>; + setSelectedWorker: React.Dispatch< + React.SetStateAction< + | { + userName: string; + id: number; + position: string; + } + | undefined + > + >; +} + +export default function PromotionModal({ + isModalOpen, + selectedWorker, + setIsModalOpen, + setSelectedWorker, +}: PromotionModalProps) { + const [messageApi, contextHolder] = message.useMessage(); + + const accessToken = useRecoilValue(AccessTokenAtom); + const [selectedPosition, setSelectedPosition] = useState< + '변경' | 'LEVEL1' | 'LEVEL2' | 'LEVEL3' | 'LEVEL4' + >('변경'); + + const handleChangePosition = async () => { + if (!accessToken) { + return; + } + if (selectedPosition === '변경') { + return; + } + try { + const response = await changePosition( + selectedWorker?.id as number, + selectedPosition as 'LEVEL1' | 'LEVEL2' | 'LEVEL3' | 'LEVEL4', + ); + if (response.status === 200) { + messageApi.open({ + type: 'success', + content: response.data.response, + }); + setSelectedWorker( + (prev) => + ({ + ...prev, + position: selectedPosition, + } as { userName: string; id: number; position: string }), + ); + setSelectedPosition('변경'); + return; + } + // eslint-disable-next-line @typescript-eslint/no-explicit-any + } catch (error: any) { + messageApi.open({ + type: 'error', + content: + error.response?.data.error.message || '직책 변경에 실패하였습니다.', + }); + } + }; + + return ( + <> + {contextHolder} + {selectedWorker?.userName} 직책 변경} + centered + closeIcon={false} + open={isModalOpen} + onOk={() => { + setIsModalOpen(false); + handleChangePosition(); + }} + onCancel={() => setIsModalOpen(false)} + > + + + + 에서 + + setEditPhonNumberInput(e.target.value)} + /> + + + + ) : ( + <>{formatPhoneNumber(myAccountInfo.phoneNumber)} + )} + + + + + + + {myAccountInfo.usedVacation}일 + + + {POSITIONS[myAccountInfo.position]?.total_vacation}일 + + + + 사진 + {editprofileThumbUrl ? ( + { + setEditProfileThumbUrl(false); + setEditPhonNumberInput(myAccountInfo.phoneNumber); + }} + /> + ) : ( + { + setEditProfileThumbUrl(true); + }} + style={{ marginLeft: 5, fontSize: 15 }} + className="icons" + /> + )} + + } + > + {editprofileThumbUrl ? ( + <> +
+ + false} /* showUploadList={false} */ + > + + + + + + + ) : ( + <> + + } + rootClassName="profile_image" + width={200} + height={200} + src={myAccountInfo.profileThumbUrl} + fallback={defaultProfile} + /> + + )} +
+ + + + + + ); +} diff --git a/src/page/myAccount/myAccoutLayout.tsx b/src/page/myAccount/myAccoutLayout.tsx new file mode 100644 index 00000000..1ed27bc5 --- /dev/null +++ b/src/page/myAccount/myAccoutLayout.tsx @@ -0,0 +1,97 @@ +import { Layout, Menu, MenuProps } from 'antd'; +import { Outlet, useLocation, useNavigate } from 'react-router-dom'; +import { useState } from 'react'; +import { useRecoilValue } from 'recoil'; +import { IsManagerAtom } from '@/recoil/IsManagerAtom'; +const { Content, Sider } = Layout; + +export default function MyAccountLayout() { + const { pathname } = useLocation(); + + const MY_ACCOUNT_MENU: { + [key: string]: { key: string }; + } = { + '/myaccount': { + key: '1', + }, + '/myaccount/vacation': { + key: '2', + }, + '/myaccount/approve': { + key: '3', + }, + '/myaccount/promote': { + key: '3', + }, + }; + + const [selectedMenuKey, setSelectedMenuKey] = useState( + MY_ACCOUNT_MENU[pathname].key, + ); + + const navigate = useNavigate(); + + // 현재 접속중인 url의 뒷부분을 가져옴 + const isManager = useRecoilValue(IsManagerAtom); + + const items: MenuProps['items'] = [ + { + key: '1', + label: '내 정보', + onClick: () => { + navigate('/myaccount'); + setSelectedMenuKey('1'); + }, + }, + { + key: '2', + label: '내 연차/당직', + onClick: () => { + navigate('/myaccount/vacation'); + setSelectedMenuKey('2'); + }, + }, + { + key: '3', + label: '관리자', + onClick: () => { + navigate('/myaccount/approve'); + setSelectedMenuKey('3'); + }, + }, + ]; + + return ( + + + + + + + + + + + ); +} diff --git a/src/page/myAccount/passwordChangeModal.tsx b/src/page/myAccount/passwordChangeModal.tsx new file mode 100644 index 00000000..c65a65cc --- /dev/null +++ b/src/page/myAccount/passwordChangeModal.tsx @@ -0,0 +1,180 @@ +import { changeMyInfo } from '@/api/myAccount/changeMyInfo'; +import { PASSWORD_REGEX } from '@/data/constants'; +import { AccessTokenAtom } from '@/recoil/AccessTokkenAtom'; +import { Button, Form, Input, Modal, Space, message } from 'antd'; +import { FormInstance, RuleObject } from 'antd/es/form'; +import { useCallback, useRef } from 'react'; +import { useRecoilValue } from 'recoil'; + +interface PasswordChangeModalProps { + isModalOpen: boolean; + setIsModalOpen: React.Dispatch>; +} + +export default function PasswordChangeModal({ + isModalOpen, + setIsModalOpen, +}: PasswordChangeModalProps) { + const formRef = useRef(null); + + // antd message(화면 상단에 뜨는 메세지)기능 + const [messageApi, contextHolder] = message.useMessage(); + + const accessToken = useRecoilValue(AccessTokenAtom); + + // 비밀번호 유효성 검사 + const validatePassword = useCallback((_: RuleObject, value: string) => { + const NUMBER_REGEX = /\d/; + const SPECIAL_REGEX = /[!@#$%^&*()-+=]/; + const ENGLISH_REGEX = /[a-zA-Z]/; + if (!value) { + return Promise.reject(new Error('비밀번호를 입력해주세요.')); + } + if (!NUMBER_REGEX.test(value) && !ENGLISH_REGEX.test(value)) { + return Promise.reject( + new Error( + '비밀번호에는 최소 하나의 숫자와 영어 대소문자가 포함되어야 합니다.', + ), + ); + } + if (!SPECIAL_REGEX.test(value) && !ENGLISH_REGEX.test(value)) { + return Promise.reject( + new Error( + '비밀번호에는 최소 하나의 특수문자와 영어 대소문자가 포함되어야 합니다.', + ), + ); + } + if (!NUMBER_REGEX.test(value) && !SPECIAL_REGEX.test(value)) { + return Promise.reject( + new Error( + '비밀번호에는 최소 하나의 숫자와 특수문자가 포함되어야 합니다.', + ), + ); + } + if (!NUMBER_REGEX.test(value)) { + return Promise.reject( + new Error('비밀번호에는 최소 하나의 숫자가 포함되어야 합니다.'), + ); + } + + if (!SPECIAL_REGEX.test(value)) { + return Promise.reject( + new Error('비밀번호에는 최소 하나의 특수문자가 포함되어야 합니다.'), + ); + } + if (!ENGLISH_REGEX.test(value)) { + return Promise.reject( + new Error( + '비밀번호에는 최소 하나의 영어 대소문자가 포함되어야 합니다.', + ), + ); + } + if (!PASSWORD_REGEX.test(value)) { + return Promise.reject(new Error('비밀번호는 8~16자 입니다.')); + } + return Promise.resolve(); + }, []); + + const handleChangePassword = async (values: { + newPassword: string; + confirmPassword: string; + }) => { + try { + if (!accessToken) { + return; + } + const response = await changeMyInfo({ + userPassword: values.newPassword, + }); + if (response.status === 200) { + setIsModalOpen(false); + messageApi.open({ + type: 'success', + content: '비밀번호를 수정하였습니다.', + }); + } + if (formRef.current) { + formRef.current.resetFields(); + } + // eslint-disable-next-line @typescript-eslint/no-explicit-any + } catch (error: any) { + messageApi.open({ + type: 'success', + content: + error.response.data.error.message || + '사용자 정보 수정에 실패하였습니다.', + }); + } + }; + + return ( + <> + {contextHolder} + { + setIsModalOpen(false); + }} + onCancel={() => { + setIsModalOpen(false); + }} + closeIcon={false} + > +
+ + + + + ({ + validator(_, value) { + if (!value || getFieldValue('newPassword') === value) { + return Promise.resolve(); + } + return Promise.reject( + new Error('비밀번호가 일치하지 않습니다.'), + ); + }, + }), + ]} + > + + + + + +
+ + ); +} diff --git a/src/page/myAccount/vacation.tsx b/src/page/myAccount/vacation.tsx new file mode 100644 index 00000000..4f1f53b5 --- /dev/null +++ b/src/page/myAccount/vacation.tsx @@ -0,0 +1,229 @@ +import { getMySchedule } from '@/api/mySchedule'; +import RequesTag from '@/components/RequesTag'; +import { DUTY_ANNUAL } from '@/data/constants'; +import { cancelScheduleRequest } from '@/api/mySchedule'; +import { AccessTokenAtom } from '@/recoil/AccessTokkenAtom'; +import { Select, Button, Table, message, Popconfirm } from 'antd'; +import type { ColumnsType } from 'antd/es/table'; +import { useEffect, useState } from 'react'; +import { useRecoilValue, useSetRecoilState } from 'recoil'; +import { ReRenderStateAtom } from '@/recoil/ReRenderStateAtom'; +import dayjs from 'dayjs'; + +interface CheckedVacationRequestType { + key: number; + id: number; + scheduleType: 'ANNUAL' | 'DUTY'; + startDate: string; + endDate: string; + state: 'PENDING' | 'APPROVE' | 'REJECT'; +} + +export default function Vaction() { + // antd message(화면 상단에 뜨는 메세지)기능 + const [messageApi, contextHolder] = message.useMessage(); + + const [checkedVacationRequests, setCheckedVacationRequests] = useState< + CheckedVacationRequestType[] + >([]); + + const [isvacationRequestsLoading, setIsvacationRequestsLoading] = + useState(false); + + const [isLoading, setIsLoading] = useState(false); + + const currentYear = new Date().getFullYear(); + + const [year, setYear] = useState(currentYear); + + const accessToken = useRecoilValue(AccessTokenAtom); + + const setReRender = useSetRecoilState(ReRenderStateAtom); + + const { Option } = Select; + + const fetchData = async () => { + if (!accessToken) { + return; + } + try { + const response = await getMySchedule(year); + if (response.status === 200) { + const CheckedVacationRequestsData = response.data + .response as CheckedVacationRequestType[]; + // 성공했을때 + setCheckedVacationRequests( + CheckedVacationRequestsData.map((el) => { + return { ...el, key: el.id }; + }), + ); + } + // eslint-disable-next-line @typescript-eslint/no-explicit-any + } catch (error: any) { + console.log('연가/당직 데이터를 불러오지 못했습니다. : ', error); + } finally { + setIsvacationRequestsLoading(false); + } + }; + + useEffect(() => { + setIsvacationRequestsLoading(true); + fetchData(); + }, [accessToken, year]); + + const handleCancelSchedule = async (id: number) => { + setIsLoading(true); + try { + const response = await cancelScheduleRequest(id); + if (response.status === 200) { + // 삭제가 성공하면 데이터 다시 불러오기 + fetchData(); + messageApi.open({ + type: 'success', + content: '삭제되었습니다.', + }); + setReRender((prev) => !prev); + } + } catch (error) { + console.log(error); + messageApi.open({ + type: 'error', + content: '요청 실패', + }); + } finally { + setIsLoading(false); + } + }; + + const pastDates = (current: dayjs.Dayjs) => { + return current < dayjs().startOf('day'); + }; + + const columns: ColumnsType = [ + { + title: '연차/당직', + dataIndex: 'scheduleType', + key: 'scheduleType', + align: 'center', + render: (_, { scheduleType }) => ( + + {DUTY_ANNUAL[scheduleType].label} + + ), + filters: [ + { + text: '연차', + value: 'ANNUAL', + }, + { + text: '당직', + value: 'DUTY', + }, + ], + onFilter: (value: string | number | boolean, record) => + record.scheduleType.includes(value as string), + }, + { + title: '시작일', + dataIndex: 'startDate', + key: 'startDate', + align: 'center', + defaultSortOrder: 'descend', + sorter: (a, b) => + Number(a.startDate.replaceAll('-', '')) - + Number(b.startDate.replaceAll('-', '')), + }, + + { + title: '종료일', + dataIndex: 'endDate', + key: 'endDate', + align: 'center', + sorter: (a, b) => + Number(a.endDate.replaceAll('-', '')) - + Number(b.endDate.replaceAll('-', '')), + }, + + { + title: '승인여부', + key: 'tags', + dataIndex: 'tags', + render: (_, { state }) => , + filters: [ + { + text: '심사중', + value: 'PENDING', + }, + { + text: '승인', + value: 'RESOLVE', + }, + { + text: '거절', + value: 'REJECT', + }, + ], + onFilter: (value: string | number | boolean, record) => + record.state.includes(value as string), + align: 'center', + }, + { + title: 'Action', + key: 'action', + align: 'center', + render: (_, { id, endDate, state }) => ( + handleCancelSchedule(id)} + okText="Yes" + cancelText="No" + disabled={pastDates(dayjs(endDate)) || state === 'REJECT'} + > + + + ), + }, + ]; + + const selectedYearsOptions = []; + for (let i = 0; i <= 5; i++) { + const year = currentYear - i; + selectedYearsOptions.push( + , + ); + } + + return ( + <> + {contextHolder} + +
+ + ); +} diff --git a/src/page/notFound/notFountd.tsx b/src/page/notFound/notFountd.tsx new file mode 100644 index 00000000..52e792e2 --- /dev/null +++ b/src/page/notFound/notFountd.tsx @@ -0,0 +1,12 @@ +import { Button, Result } from 'antd'; + +export default function NotFound() { + return ( + Back Home} + /> + ); +} diff --git a/src/page/signup/signup.tsx b/src/page/signup/signup.tsx new file mode 100644 index 00000000..a7dc8b73 --- /dev/null +++ b/src/page/signup/signup.tsx @@ -0,0 +1,549 @@ +import { useCallback, useState, useEffect } from 'react'; +import { + Button, + Input, + Form, + Select, + Card, + Upload, + message, + Space, +} from 'antd'; +import { PlusOutlined } from '@ant-design/icons'; +import { Link, useNavigate } from 'react-router-dom'; +import { EMAIL_REGEX, PASSWORD_REGEX, POSITIONS } from '@/data/constants'; +import { RuleObject } from 'antd/es/form'; +import { handleUpload } from '@/api/auth/cloudinary'; +import { signup } from '@/api/auth/signup'; +import { checkEmail, checkEmailAuth } from '@/api/auth/checkEmail'; +import { verificationEmail } from '@/api/auth/verification'; +import { useSetRecoilState } from 'recoil'; +import { AccessTokenAtom } from '@/recoil/AccessTokkenAtom'; +import { setAccessTokenToCookie } from '@/utils/cookies'; + +interface valuseType { + confirm_password: string; + phone: string; + position: string; + // eslint-disable-next-line @typescript-eslint/no-explicit-any + profileThumbUrl: any; + userEmail: string; + userPassword: string; + userName: string; +} + +export default function Signup() { + // 이메일 중복 체크 + const [isEmailCehck, setIsEmailCheck] = useState(false); + // 이메일 인증 번호 발송 + const [verification, setVerification] = useState(false); + // 이메일 인증 번호 발송 후에 제출 가능 타이머 + const [timer, setTimer] = useState(0); + // 이메일 인증 번호 재발송 ui + const [reSend, setReSend] = useState(false); + // 이메일 인증 과정 로딩 (중복체크, 인증번호 발송, 인증확인) + const [isLoading, setIsLoading] = useState(false); + // 이메일 인증번호 확인 + const [emailVerified, setEmailVerified] = useState(false); + // 회원가입 통신 과정 로딩 ui + const [isSigningUp, setIsSigningUp] = useState(false); + // antd message(화면 상단에 뜨는 메세지)기능 + const [messageApi, contextHolder] = message.useMessage(); + + const setAccessToken = useSetRecoilState(AccessTokenAtom); + const navigate = useNavigate(); + const { Option } = Select; + const [form] = Form.useForm(); + + const onFinish = async (values: valuseType) => { + // 클라우디너리로 전송한 이미지 url 가져옴 + const imageUrl = await getImageUrl(values); + + // 가져온 이미지 url만 profileThumbUrl에 전달하기 위해서 + // values객체를 전개연산자를 이용해서 newValues에 모든 속성을 복사하고, + // profileThumbUrl: imageUrl을 마지막에 사용해서 덮어 씌움 + const newValues = { + ...values, + profileThumbUrl: imageUrl, + }; + + setIsSigningUp(true); + try { + const response = await signup(newValues); + + // 로그인 성공 + if (response.status === 200) { + const { accessToken } = response.data.response; + + // 쿠키에 저장 + setAccessTokenToCookie(accessToken); + + // recoil에 저장 + setAccessToken(accessToken); + + // 안내메시지 + messageApi.open({ + type: 'success', + content: '회원가입이 완료되었습니다.', + }); + + // 회원가입이 성공한 경우 홈으로 이동 + setTimeout(() => { + navigate('/'); + }, 1000); + + return; + } + // eslint-disable-next-line @typescript-eslint/no-explicit-any + } catch (error: any) { + messageApi.open({ + type: 'error', + content: + error.response?.data.error.message || + '회원가입에 실패하였습니다. 관리자에게 문의하세요.', + }); + } finally { + setIsSigningUp(false); + } + }; + + // cloudinary를 이용해서 이미지url이 없으면 null, 있으면 이미지url을 반환 + const getImageUrl = async (values: valuseType) => { + let imageUrl = null; + + try { + if (values.profileThumbUrl && values.profileThumbUrl.length > 0) { + const response = await handleUpload( + values.profileThumbUrl[0].originFileObj, + ); + if (response?.status === 200) { + const data = response.data; + imageUrl = data.url; // 이미지 URL을 받아옴 + } else { + throw new Error('이미지 업로드에 실패하였습니다.'); + } + } + } catch (error) { + console.error('이미지 업로드중 오류 발생:', error); + imageUrl = null; // 오류가 발생했으므로 imageUrl을 null로 설정 + } + + return imageUrl; + }; + + // eslint-disable-next-line @typescript-eslint/no-explicit-any + const normFile = (e: any) => { + if (Array.isArray(e)) { + return e; + } + return e?.fileList; + }; + + // 이메일 유효성 검사 + const validateEmail = useCallback((_: RuleObject, value: string) => { + if (!value || EMAIL_REGEX.test(value)) { + return Promise.resolve(); + } + return Promise.reject(new Error('올바른 형식의 메일을 입력해주세요.')); + }, []); + + // 비밀번호 유효성 검사 + const validatePassword = useCallback((_: RuleObject, value: string) => { + const NUMBER_REGEX = /\d/; + const SPECIAL_REGEX = /[!@#$%^&*()-+=]/; + const ENGLISH_REGEX = /[a-zA-Z]/; + + if (!value) { + return Promise.reject(new Error('비밀번호를 입력해주세요.')); + } + + if (!NUMBER_REGEX.test(value) && !ENGLISH_REGEX.test(value)) { + return Promise.reject( + new Error('최소 하나의 숫자와 영어 대소문자가 포함되어야 합니다.'), + ); + } + + if (!SPECIAL_REGEX.test(value) && !ENGLISH_REGEX.test(value)) { + return Promise.reject( + new Error('최소 하나의 특수문자와 영어 대소문자가 포함되어야 합니다.'), + ); + } + + if (!NUMBER_REGEX.test(value) && !SPECIAL_REGEX.test(value)) { + return Promise.reject( + new Error('최소 하나의 숫자와 특수문자가 포함되어야 합니다.'), + ); + } + + if (!NUMBER_REGEX.test(value)) { + return Promise.reject(new Error('최소 하나의 숫자가 포함되어야 합니다.')); + } + + if (!SPECIAL_REGEX.test(value)) { + return Promise.reject( + new Error('최소 하나의 특수문자가 포함되어야 합니다.'), + ); + } + + if (!ENGLISH_REGEX.test(value)) { + return Promise.reject( + new Error('최소 하나의 영어 대소문자가 포함되어야 합니다.'), + ); + } + + if (!PASSWORD_REGEX.test(value)) { + return Promise.reject(new Error('비밀번호는 8~16자 입니다.')); + } + + return Promise.resolve(); + }, []); + + // 직급 선택 동적으로 생성 + 매니저 선택 옵션이 생기지 않게 filter를 이용해서 제거 + // key가 MANAGER가 아니면 통과 + const selectedPositionOptions = Object.keys(POSITIONS) + .filter((key) => key !== 'MANAGER') + .map((key) => { + return ( + + ); + }); + + // 이메일 중복체크 + const handleCheckEmail = async () => { + setIsLoading(true); + try { + // userEmail form을 찾아서 + const values = await form.validateFields(['userEmail']); + // userEmail을 가져옴 + const userEmail = values.userEmail; + // checkEmail에 userEmail을 매개변수로 전달달 + const response = await checkEmail(userEmail); + + if (response.data.success) { + setIsEmailCheck(true); // 중복되지 않은 이메일일 경우 + message.success('사용 가능한 이메일입니다.'); + } + // eslint-disable-next-line @typescript-eslint/no-explicit-any + } catch (error: any) { + messageApi.open({ + type: 'error', + content: + error.response?.data.error.message || + '이메일 중복체크에 실패하였습니다. 관리자에게 문의하세요.', + }); + } finally { + setIsLoading(false); + } + }; + + // 이메일 인증번호 발송 + const handleVerificationEmail = async () => { + setIsLoading(true); + try { + const values = await form.validateFields(['userEmail']); + const userEmail = values.userEmail; + const response = await verificationEmail(userEmail); + + if (response.status === 200) { + message.success('인증번호를 발송했습니다.'); + // 인증번호 발생이 정상적으로 이루어졌으면 타이머가 180으로 상태값 변경 + setTimer(180); + } else { + message.warning('이메일을 다시 확인해주세요.'); + } + return response; + } catch (error) { + message.error('인증번호 발송에 오류가 발생했습니다.'); + return null; + } finally { + setIsLoading(false); + setVerification(true); + } + }; + + const handleReSend = async () => { + if (reSend) { + setIsLoading(true); + try { + setReSend(false); + await handleVerificationEmail(); + } catch (error) { + console.error('이메일 인증 요청 중 오류 발생:', error); + } finally { + setIsLoading(false); + } + } + }; + + // 타이머 변경에 따른 리렌더링이 일어나게 + useEffect(() => { + // side effect를 멈추기 위해서 exactTimer라는 변수를 선언 + let exactTimer: string | number | NodeJS.Timeout | undefined; + // 타이머가 0보다 크면 timer의 상태값이 초당 -1씩 줄어들고 + // timer가 0초이고 verification state값이 true일 때, + // reSend state값이 true가 되어서 재발송 버튼을 클릭 할 수 있다. + if (timer > 0) { + exactTimer = setTimeout( + () => setTimer((prevTimer) => prevTimer - 1), + 1000, + ); + } else if (timer === 0 && verification) { + setReSend(true); + } + // 위의 동작들이 실행되면 타이머가 멈춘다. + return () => { + if (exactTimer) { + clearTimeout(exactTimer); + } + }; + }, [timer, verification]); + + // 이메일 인증번호 발송 받은 번호를 인증 + const handleEmailAuth = async () => { + try { + const userEmailData = await form.validateFields(['userEmail']); + const userEmail = userEmailData.userEmail; + const emailAuthData = await form.validateFields(['emailAuth']); + const userEmailAuth = emailAuthData.emailAuth; + const data = { userEmail, userEmailAuth }; + const response = await checkEmailAuth(data); + + if (response.data.success) { + message.success('이메일 인증 성공!'); + setEmailVerified(true); + setTimer(0); + } else { + message.error('이메일 인증 실패!'); + } + } catch (error) { + message.error('서버 요청에 실패하였습니다.'); + } + }; + + // 이메일 인증을 다시 처음부터 하게 하는 재인증 기능 + const handleReAuthentication = () => { + setIsEmailCheck(false); + setVerification(false); + setEmailVerified(false); + setReSend(false); + }; + + return ( +
+ {contextHolder} + +
+
+ + + + + + + + + { + if (!value) { + return Promise.reject('필수 진행 항목입니다.'); + } + return Promise.resolve(); + }, + }, + ]} + > + {isEmailCehck ? ( + verification ? ( + emailVerified ? ( + // 인증에 성공한 경우 - 재인증 버튼 표시 +
+ +
+ ) : ( + // 인증 번호 입력 창과 제출 버튼 +
+ + + + + +
남은 시간: {timer}초
+
+ ) + ) : ( + // 인증 번호 전송 버튼 + + ) + ) : ( + // 이메일 중복 체크 버튼 + + )} +
+ + + + + + ({ + validator(_, value) { + if (!value || getFieldValue('userPassword') === value) { + return Promise.resolve(); + } + return Promise.reject( + new Error('비밀번호가 일치하지 않습니다.'), + ); + }, + }), + ]} + > + + + + + + + + + + + false}> + + + +
+ +
+ + + + +
+ +
+
+ ); +} diff --git a/src/recoil/AccessTokkenAtom.ts b/src/recoil/AccessTokkenAtom.ts new file mode 100644 index 00000000..a888a124 --- /dev/null +++ b/src/recoil/AccessTokkenAtom.ts @@ -0,0 +1,7 @@ +import { getAccessTokenFromCookie } from '@/utils/cookies'; +import { atom } from 'recoil'; + +export const AccessTokenAtom = atom({ + key: 'AccessToken', + default: getAccessTokenFromCookie(), +}); diff --git a/src/recoil/IsManagerAtom.ts b/src/recoil/IsManagerAtom.ts new file mode 100644 index 00000000..ca1c4d5a --- /dev/null +++ b/src/recoil/IsManagerAtom.ts @@ -0,0 +1,6 @@ +import { atom } from 'recoil'; + +export const IsManagerAtom = atom({ + key: 'IsManagerAtom', + default: false, +}); diff --git a/src/recoil/ReRenderStateAtom.ts b/src/recoil/ReRenderStateAtom.ts new file mode 100644 index 00000000..0a650f7e --- /dev/null +++ b/src/recoil/ReRenderStateAtom.ts @@ -0,0 +1,6 @@ +import { atom } from 'recoil'; + +export const ReRenderStateAtom = atom({ + key: 'ReRenderStateAtom', + default: false, +}); diff --git a/src/recoil/UserEmailAtom.ts b/src/recoil/UserEmailAtom.ts new file mode 100644 index 00000000..e193ee48 --- /dev/null +++ b/src/recoil/UserEmailAtom.ts @@ -0,0 +1,6 @@ +import { atom } from 'recoil'; + +export const UserEmailAtom = atom({ + key: 'UserEmailAtom', + default: '', +}); diff --git a/src/types/IMySchdule.ts b/src/types/IMySchdule.ts new file mode 100644 index 00000000..cc0486f8 --- /dev/null +++ b/src/types/IMySchdule.ts @@ -0,0 +1,8 @@ +export interface IMySchedule { + id: number; + key: number; + scheduleType: 'ANNUAL' | 'DUTY'; + startDate: string; + endDate: string; + state: 'REJECT' | 'APPROVE' | 'PENDING'; +} diff --git a/src/utils/cookies.ts b/src/utils/cookies.ts new file mode 100644 index 00000000..09f253f7 --- /dev/null +++ b/src/utils/cookies.ts @@ -0,0 +1,27 @@ +export const deleteAccessTokenFromCookie = () => { + const date = new Date(); + // 쿠키는 따로 삭제 방법이 있지 않고 만료시간을 현재로 줘서 자동적으로 삭제되게 + const expires = 'expires=' + date.toUTCString(); + const cookieString = `accessToken=; ${expires}; path=/`; + document.cookie = cookieString; +}; + +export const getAccessTokenFromCookie = () => { + // 쿠키에 accessToken 있으면 가져오고 없으면 null + const cookie = document.cookie + .split('; ') + .find((cookie) => cookie.split('=')[0] === 'accessToken'); + return cookie ? cookie.split('=')[1] : null; +}; + +export const setAccessTokenToCookie = (accessToken: string) => { + const date = new Date(); + + // 30분 추가 + date.setTime(date.getTime() + 30 * 60 * 1000); + const expires = 'expires=' + date.toUTCString(); + + // 쿠키에 accessToke, 만료시간, path지정 + const cookieString = `accessToken=${accessToken}; ${expires}; path=/`; + document.cookie = cookieString; +}; diff --git a/src/utils/formatPhonenumber.ts b/src/utils/formatPhonenumber.ts new file mode 100644 index 00000000..2cb5f77a --- /dev/null +++ b/src/utils/formatPhonenumber.ts @@ -0,0 +1,6 @@ +export default function formatPhoneNumber(phoneNumber: string) { + return `${phoneNumber.substring(0, 3)}-${phoneNumber.substring( + 3, + 7, + )}-${phoneNumber.substring(7)}`; +} diff --git a/src/utils/getPayloadFromJWT.ts b/src/utils/getPayloadFromJWT.ts new file mode 100644 index 00000000..204d03bb --- /dev/null +++ b/src/utils/getPayloadFromJWT.ts @@ -0,0 +1,13 @@ +export default function getPayloadFromJWT(jwt: string | null) { + // jwt가 없으면 payload도 없는거니깐 undefined return + if (!jwt) { + return undefined; + } + // payload부분 base64 decoding하는 로직, jwt관련 외부 라이브러리 사용할 필요 없음 + const payload = jwt.split('.')[1]; + const base64 = payload.replace(/_/g, '/').replace(/-/g, '+'); + const padding = '='.repeat((4 - (base64.length % 4)) % 4); + const decodedPayload = atob(base64 + padding); + const payloadObject = JSON.parse(decodedPayload); + return payloadObject; +} diff --git a/src/vite-env.d.ts b/src/vite-env.d.ts new file mode 100644 index 00000000..11f02fe2 --- /dev/null +++ b/src/vite-env.d.ts @@ -0,0 +1 @@ +/// diff --git a/tsconfig.json b/tsconfig.json new file mode 100644 index 00000000..77b07003 --- /dev/null +++ b/tsconfig.json @@ -0,0 +1,29 @@ +{ + "compilerOptions": { + "target": "ES2020", + "useDefineForClassFields": true, + "lib": ["ES2021", "DOM", "DOM.Iterable"], + "module": "ESNext", + "skipLibCheck": true, + "baseUrl": ".", + "paths": { + "@/*": ["src/*"] + }, + + /* Bundler mode */ + "moduleResolution": "bundler", + "allowImportingTsExtensions": true, + "resolveJsonModule": true, + "isolatedModules": true, + "noEmit": true, + "jsx": "react-jsx", + + /* Linting */ + "strict": true, + "noUnusedLocals": true, + "noUnusedParameters": true, + "noFallthroughCasesInSwitch": true + }, + "include": ["src"], + "references": [{ "path": "./tsconfig.node.json" }] +} diff --git a/tsconfig.node.json b/tsconfig.node.json new file mode 100644 index 00000000..42872c59 --- /dev/null +++ b/tsconfig.node.json @@ -0,0 +1,10 @@ +{ + "compilerOptions": { + "composite": true, + "skipLibCheck": true, + "module": "ESNext", + "moduleResolution": "bundler", + "allowSyntheticDefaultImports": true + }, + "include": ["vite.config.ts"] +} diff --git a/vite.config.ts b/vite.config.ts new file mode 100644 index 00000000..445e88c4 --- /dev/null +++ b/vite.config.ts @@ -0,0 +1,13 @@ +import { defineConfig } from 'vite'; +import react from '@vitejs/plugin-react-swc'; +import path from 'path'; + +// https://vitejs.dev/config/ +export default defineConfig({ + plugins: [react()], + resolve: { + alias: { + '@': path.resolve(__dirname, './src'), + }, + }, +}); From 9be913cb209bbf01568e1baf1bac69fba885a030 Mon Sep 17 00:00:00 2001 From: howooking Date: Fri, 11 Aug 2023 20:19:59 +0900 Subject: [PATCH 2/3] =?UTF-8?q?=EC=A0=9C=EC=B6=9C?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- README.md | 268 +++++++++++++++++++++-- src/api/auth/checkEmail.ts | 1 - src/api/customAxios.ts | 4 +- src/page/home/home.tsx | 53 +++-- src/page/home/mySchedule.tsx | 2 +- src/page/home/signin.tsx | 4 +- src/page/myAccount/admin/approve.tsx | 8 +- src/page/myAccount/admin/promote.tsx | 5 +- src/page/signup/signup.tsx | 309 ++++++++++++++------------- vercel.json | 1 + 10 files changed, 465 insertions(+), 190 deletions(-) create mode 100644 vercel.json diff --git a/README.md b/README.md index 1ebe379f..15434b63 100644 --- a/README.md +++ b/README.md @@ -1,27 +1,257 @@ -# React + TypeScript + Vite +# KDT5-MINI with backend -This template provides a minimal setup to get React working in Vite with HMR and some ESLint rules. +## 프로젝트 소개 -Currently, two official plugins are available: +본 웹 어플리케이션은 50명 내외의 중소기업에서 사용하는 연차/당직 관리 프로그램입니다. -- [@vitejs/plugin-react](https://github.com/vitejs/vite-plugin-react/blob/main/packages/plugin-react/README.md) uses [Babel](https://babeljs.io/) for Fast Refresh -- [@vitejs/plugin-react-swc](https://github.com/vitejs/vite-plugin-react-swc) uses [SWC](https://swc.rs/) for Fast Refresh +패스트캠퍼스 백엔드 5기 3분과 팀을 이루어 협업을 진행하였습니다. -## Expanding the ESLint configuration +[결과물 보러가기](https://kdt-5-mini-team-11-eifz.vercel.app/) -If you are developing a production application, we recommend updating the configuration to enable type aware lint rules: +## 11조 개쩌는팀 소개 -- Configure the top-level `parserOptions` property like this: +| 팀원 | 박진영 | 남기훈 | 이정우 | 성규창 | 김용원 | 배종윤 | +| :--: | :--------------: | :--------------------: | :------------------------: | ------ | ------ | ------ | +| 담당 | 내계정
관리자 | 회원가입
캘린더
| 인증인가
연차/당직 신청 | 백엔드 | 백엔드 | 백엔드 | -```js - parserOptions: { - ecmaVersion: 'latest', - sourceType: 'module', - project: ['./tsconfig.json', './tsconfig.node.json'], - tsconfigRootDir: __dirname, - }, -``` +

-- Replace `plugin:@typescript-eslint/recommended` to `plugin:@typescript-eslint/recommended-type-checked` or `plugin:@typescript-eslint/strict-type-checked` -- Optionally add `plugin:@typescript-eslint/stylistic-type-checked` -- Install [eslint-plugin-react](https://github.com/jsx-eslint/eslint-plugin-react) and add `plugin:react/recommended` & `plugin:react/jsx-runtime` to the `extends` list +## 사용한 기술, 라이브러리 + +### Environment + +
+
+
+ +### Config + +
+
+ +### Frontend + +
+
+
+
+ +### Backend + +
+
+ +### Co-work + +
+
+
+ +## 화면 구성 + +### 로그인 + +image-1 + +- 사내에서 사용하는 프로그램으로 반드시 로그인이 필요함 + - 로그인, 회원가입 페이지를 제외한 모든 페이지는 protected route +- 로그인 창을 모달화하였으며 외부 배경을 blur 처리 +- 회원가입 버튼을 눌러 회원가입페이지로 이동 + +
+ +### 회원가입 + +![2023-08-11_14-32-12-1](https://github.com/KDT1-FE/KDT5-Mini/assets/87072568/1c14974d-3a5e-4dca-9270-5928830161a2) + +- 입력 항목별 유효성 검사 +- 이메일 중복 체크 이후에 이메일 인증해야 회원가입을 할 수 있다. + +
+ +#### 메인 화면 + +image-3 + +- 사이드바에서 내가 신청한 연차/당직 목록을 확인 할 수 있다. +- 사이드바에서 새로운 연차/당직을 신청할 수 있다. + - 신청한 연차/당직은 대기 상태로 등록이 되며 관리자가 승인 또는 거절을 한다. + - 심사중인 연차/당직은 취소가 가능하다. +- 우측 상단에 간략한 내 정보 및 로그아웃 버튼이 있다. +- 달력에는 모든 사원들의 승인된 일정이 표시된다. +- 달력 좌측 상단의 Switch 버튼을 눌러 내 일정만을 볼 수 있다. + +
+ +#### 내 정보 페이지 + +image-4 + +- 내 정보와 수정이 가능한 항목을 수정 할 수 있다. +- 유효성 검사 결과와 등록 결과 등을 팝업 메세지로 사용자에게 알려준다. +
+ +#### 내 연차/당직 페이지 + +image-6 + +- 내가 신청한 연차/당직 정보를 확인할 수 있다. +- 심사 중인 연차/당직은 삭제가 가능하다. +- 거절된 연차/당직은 삭제가 불가능하다. +- 승인된 연차/당직은 삭제가 가능하다. +- 과거의 연차/당직은 삭제가 불가능하다. + +
+ +#### 관리자 연차/당직 승인 페이지 + +image-7 + +- 관리자는 모든 사원의 연차/당직 신청을 승인, 거절, 취소할 수 있다. +- 관리자`MANAGER`가 메뉴가 보이지 않으며 접근 할 수 없다. + +
+ +#### 관리자 사원 직책 변경 페이지 + +![2023-08-11_14-52-36](https://github.com/KDT1-FE/KDT5-Mini/assets/87072568/aec52ed5-d6ee-4138-a958-af6d5495ae02) + +- 관리자는 사원의 직책을 변경할 수 있다. + +
+ +## 고찰 + +### 협업 + +- 처음으로 백엔드와 협업을 진행

+- 백엔드는 프론트 지식이 프론트는 백엔드 지식이 부족한 관계로 초반 소통과정이 원활하지 않음

+ - 가장 중요한 것은 소통, 지식 간극을 적극적인 소통으로 줄여나감 + - data base, 네트워크 통신, 보안에 대한 추가적인 공부가 필요함

+- 노션, 슬랙, 줌, 피그마 툴을 사용 + - 노션 : API 명세서 + image-13 +

+ - 슬랙 : 텍스트 형식의 소통 + image-10

+ - 줌 : 영상, 음성 형식의 실시간 소통 + image-12

+ - 피그마 : 와이어프레임 작성 + image-14

+ +### AccessToken & RefreshToken, HttpOnly 쿠키, Axios Interceptor + +![img_1](https://github.com/KDT1-FE/KDT5-Mini/assets/87072568/2300bb17-b7a3-477d-8e12-7518f7a6b563) + +- Authorization + + - 사용자가 로그인을 요청을 보내면 서버에서 accessToken은 res.body에 담아서 보내고, refreshToken은 Http Only 쿠키로 보낸다.

+ +- Athentication + + - 로그인 한 사용자가 요청을 보낼 때 header에 accessToken을 담아 보낸다. + - Axios interceptor를 이용하면 accessToken이 필요한 모든 요청들에 토큰을 담아 보낼 수 있다.

+ + ```js + export const customAxios = axios.create({ + baseURL: BASE_API_URL, + timeout: 5000, + }); + + customAxios.interceptors.request.use( + async (req) => { + const accessToken = getAccessTokenFromCookie(); + if (!accessToken) { + return req; + } + req.headers.Authorization = `Bearer ${accessToken}`; + return req; + }, + (error) => { + return Promise.reject(error); + }, + ); + ``` + + - 서버는 해당 accessToken을 검증하고 응답을 보낸다.

+ +- AccessToken이 만료 된 경우 + + - accessToken이 만료 된 경우 서버에서는 401 상태메세지를 보내고 이를 Axios interceptor를 통해 중간에서 새로운 accessToken을 요청하는 로직을 구현할 수 있다. + + ```ts + // 여러개의 요청이 밀렸을 경우 리프레시토큰 api가 여러번 실행되는 것을 막는 flag + let isRefreshing = false; + + // 만료된 토큰으로 인해 pending상태가 된 기존의 요청들을 배열에 담음, 새로운 토큰이 발행되면 이들 요청을 진행 + let refreshSubscribers: ((accessToken: string) => void)[] = []; + + customAxios.interceptors.response.use( + (response) => { + return response; + }, + async (error) => { + const status = error.response?.data.error.status; + + if (status === 401) { + if (!isRefreshing) { + isRefreshing = true; + + try { + const response = await axios( + `${BASE_API_URL}/v1/auth/refresh-token`, + { + withCredentials: true, + }, + ); + + if (response.status === 200) { + setAccessTokenToCookie(response.data.response.accessToken); + + const config = error.config; + config.headers.Authorization = `Bearer ${response.data.response.accessToken}`; + config.withCredentials = true; + + const retryOriginalRequest = new Promise((resolve) => { + resolve(axios(config)); + }); + + isRefreshing = false; + + refreshSubscribers.forEach((callback) => + callback(response.data.response.accessToken), + ); + refreshSubscribers = []; + + return retryOriginalRequest; + } + } catch (error) { + deleteAccessTokenFromCookie(); + isRefreshing = false; + return Promise.reject(error); + } + } else { + return new Promise((resolve) => { + refreshSubscribers.push((accessToken: string) => { + error.config.headers.Authorization = `Bearer ${accessToken}`; + resolve(axios(error.config)); + }); + }); + } + } else { + return Promise.reject(error); + } + }, + ); + ``` + +- HttpOnly 쿠키 + + - 401응답을 받은 클라이언트는 accessToken 재발급을 위해 "/v1/auth/refresh-token" GET요청을 보낸다. 이 때 refreshToken은 이미 쿠키에 담겨있다. + - HttpOnly 쿠키는 클라이언트측에서 자바스크립트로 접근 할 수 없으므로 XXS와 같은 공격을 무력화 시킨다. + - `withCredentials`를 `true`로 설정해야 한다. + - 클라이언트 배포 주소와 서버 배포 주소가 모두 SSL인증서를 필요로 한다. + - 서버에서는 무료 dns와 SSL인증서를 발급방아서 해결 + - 클라이언트는 vecel로 배포시 https로 배포가 되므로 서버와 같은 작업이 필요없다. + - 그러나 로컬 환경에서 테스트를 할 때는 HttpOnly 쿠키 사용이 불가능하므로 배포 환경과 로컬 환경에서의 api를 분리하여서 작업하였다. + - 해당 리포지토리는 로컬용 diff --git a/src/api/auth/checkEmail.ts b/src/api/auth/checkEmail.ts index a7e0d25b..7a3aa7b3 100644 --- a/src/api/auth/checkEmail.ts +++ b/src/api/auth/checkEmail.ts @@ -12,7 +12,6 @@ export const checkEmailAuth = async (data: { userEmail: string; userEmailAuth: string; }) => { - // 추후에 변경 /v1/auth/check-email-auth const response = await customAxios.post('/v1/auth/check-email-auth', data, { headers: { 'Content-Type': 'application/json' }, }); diff --git a/src/api/customAxios.ts b/src/api/customAxios.ts index e6290549..6a8da0e8 100644 --- a/src/api/customAxios.ts +++ b/src/api/customAxios.ts @@ -9,13 +9,13 @@ import getPayloadFromJWT from '@/utils/getPayloadFromJWT'; export const customAxios = axios.create({ baseURL: BASE_API_URL, - timeout: 7000, // 5초간 아무 응답이 없으면 취소 + timeout: 5000, }); customAxios.interceptors.request.use( async (req) => { const accessToken = getAccessTokenFromCookie(); - // accessToken이 없는 경우 === 로그아웃을 한 상태에서하는 요청들 : 회원가입, 이메일중복체크, 로그인), 그런데 있어도 되는듯 + if (!accessToken) { return req; } diff --git a/src/page/home/home.tsx b/src/page/home/home.tsx index 7a7955d2..310f4b1e 100644 --- a/src/page/home/home.tsx +++ b/src/page/home/home.tsx @@ -99,7 +99,6 @@ export default function Home() { const [userYearlySchedulesLoading, setUserYearlySchedulesLoading] = useState(false); - const [pendingLoading, setPendingLoading] = useState(false); useEffect(() => { const getUsersYearlySchedules = async () => { if (!accessToken) { @@ -124,18 +123,20 @@ export default function Home() { }); setSideMyschedule(sideMyScheduleData); - const events = listResponseData.map((item: ScheduleItem) => { - const adjustEndDate = dayjs(item.endDate) - .add(1, 'day') - .format('YYYY-MM-DD'); - return { - userEmail: item.userEmail, - title: item.userName, - start: item.startDate, - end: adjustEndDate, - color: DUTY_ANNUAL[item.scheduleType].color, - }; - }); + const events = listResponseData + .filter((item: mySchedule) => item.state === 'APPROVE') + .map((item: ScheduleItem) => { + const adjustEndDate = dayjs(item.endDate) + .add(1, 'day') + .format('YYYY-MM-DD'); + return { + userEmail: item.userEmail, + title: item.userName, + start: item.startDate, + end: adjustEndDate, + color: DUTY_ANNUAL[item.scheduleType].color, + }; + }); setEvents(events); } catch (error) { console.log(error); @@ -146,6 +147,8 @@ export default function Home() { getUsersYearlySchedules(); }, [year, accessToken, userEmail]); + const [pendingLoading, setPendingLoading] = useState(false); + useEffect(() => { const myPendingSchedule = async () => { if (!accessToken) { @@ -174,7 +177,7 @@ export default function Home() { } }; myPendingSchedule(); - }, [year, reRender, accessToken]); + }, [year, accessToken, reRender]); const handleSelect = (value: string) => { setScheduleInput({ @@ -221,7 +224,25 @@ export default function Home() { ]?.label } 신청 완료`, }); - setReRender((prev) => !prev); + + if (response.data.response.scheduleType === 'ANNUAL') { + setReRender((prev) => !prev); + } + + if (response.data.response.scheduleType === 'DUTY') { + const newPendingSchedule = { + id: response.data.response.id, + key: response.data.response.id, + scheduleType: response.data.response.scheduleType, + startDate: response.data.response.startDate, + endDate: response.data.response.endDate, + state: response.data.response.state, + }; + setMyPendingScheduleList((prev) => { + console.log(prev); + return [...prev, newPendingSchedule]; + }); + } } // eslint-disable-next-line @typescript-eslint/no-explicit-any } catch (error: any) { @@ -241,7 +262,7 @@ export default function Home() { }; const mySchedule = events.filter((event) => event.userEmail === userEmail); - console.log(events); + return ( <> {contextHolder} diff --git a/src/page/home/mySchedule.tsx b/src/page/home/mySchedule.tsx index 30a7424d..fb86bb23 100644 --- a/src/page/home/mySchedule.tsx +++ b/src/page/home/mySchedule.tsx @@ -55,7 +55,7 @@ export default function MySchedule({ if (scheduleType === 'ANNUAL') { setReRender((prev) => !prev); } - setMyPendingScheduleList((prev) => + setMyPendingScheduleList?.((prev) => prev.filter((item) => item.key !== key), ); } diff --git a/src/page/home/signin.tsx b/src/page/home/signin.tsx index d0495ae1..7a101997 100644 --- a/src/page/home/signin.tsx +++ b/src/page/home/signin.tsx @@ -103,7 +103,7 @@ export default function Signin({ ref={formRef} > diff --git a/src/page/myAccount/admin/approve.tsx b/src/page/myAccount/admin/approve.tsx index d2676143..4fab7dbe 100644 --- a/src/page/myAccount/admin/approve.tsx +++ b/src/page/myAccount/admin/approve.tsx @@ -127,8 +127,8 @@ export default function Approve() { key: 'startDate', align: 'center', sorter: (a, b) => - Number(a.startDate.replaceAll('-', '')) - - Number(b.startDate.replaceAll('-', '')), + Number(a.startDate.slice(0, 10).replaceAll('-', '')) - + Number(b.startDate.slice(0, 10).replaceAll('-', '')), render: (_, { startDate }) => <>{startDate.slice(0, 10)}, }, @@ -138,8 +138,8 @@ export default function Approve() { key: 'endDate', align: 'center', sorter: (a, b) => - Number(a.endDate.replaceAll('-', '')) - - Number(b.endDate.replaceAll('-', '')), + Number(a.endDate.slice(0, 10).replaceAll('-', '')) - + Number(b.endDate.slice(0, 10).replaceAll('-', '')), render: (_, { endDate }) => <>{endDate.slice(0, 10)}, }, diff --git a/src/page/myAccount/admin/promote.tsx b/src/page/myAccount/admin/promote.tsx index 55714165..567c0829 100644 --- a/src/page/myAccount/admin/promote.tsx +++ b/src/page/myAccount/admin/promote.tsx @@ -85,8 +85,9 @@ export default function Promote() { dataIndex: 'createAt', align: 'center', sorter: (a, b) => - Number(a.createAt.replaceAll('-', '')) - - Number(b.createAt.replaceAll('-', '')), + Number(a.createAt.slice(0, 10).replaceAll('-', '')) - + Number(b.createAt.slice(0, 10).replaceAll('-', '')), + render: (_, { createAt }) => <>{createAt.slice(0, 10)}, }, { title: 'Action', diff --git a/src/page/signup/signup.tsx b/src/page/signup/signup.tsx index a7dc8b73..89b6a3ca 100644 --- a/src/page/signup/signup.tsx +++ b/src/page/signup/signup.tsx @@ -10,7 +10,7 @@ import { Space, } from 'antd'; import { PlusOutlined } from '@ant-design/icons'; -import { Link, useNavigate } from 'react-router-dom'; +import { useNavigate } from 'react-router-dom'; import { EMAIL_REGEX, PASSWORD_REGEX, POSITIONS } from '@/data/constants'; import { RuleObject } from 'antd/es/form'; import { handleUpload } from '@/api/auth/cloudinary'; @@ -20,6 +20,7 @@ import { verificationEmail } from '@/api/auth/verification'; import { useSetRecoilState } from 'recoil'; import { AccessTokenAtom } from '@/recoil/AccessTokkenAtom'; import { setAccessTokenToCookie } from '@/utils/cookies'; +import ButtonGroup from 'antd/es/button/button-group'; interface valuseType { confirm_password: string; @@ -336,16 +337,13 @@ export default function Signup() { background: '#eee', height: '100vh', boxSizing: 'border-box', - padding: 40, + padding: 70, display: 'flex', justifyContent: 'center', }} > {contextHolder} - +
-
- - - - - - - - - { - if (!value) { - return Promise.reject('필수 진행 항목입니다.'); - } - return Promise.resolve(); - }, + + + + + + + + + { + if (!value) { + return Promise.reject('필수 진행 항목입니다.'); + } + return Promise.resolve(); }, - ]} - > - {isEmailCehck ? ( - verification ? ( - emailVerified ? ( - // 인증에 성공한 경우 - 재인증 버튼 표시 -
- -
- ) : ( - // 인증 번호 입력 창과 제출 버튼 -
- - - - - -
남은 시간: {timer}초
-
- ) + }, + ]} + > + {isEmailCehck ? ( + verification ? ( + emailVerified ? ( + // 인증에 성공한 경우 - 재인증 버튼 표시 +
+ +
) : ( - // 인증 번호 전송 버튼 - + // 인증 번호 입력 창과 제출 버튼 +
+ + + + + +
남은 시간: {timer}초
+
) ) : ( - // 이메일 중복 체크 버튼 + // 인증 번호 전송 버튼 - )} -
- - - - - - ({ - validator(_, value) { - if (!value || getFieldValue('userPassword') === value) { - return Promise.resolve(); - } - return Promise.reject( - new Error('비밀번호가 일치하지 않습니다.'), - ); - }, - }), - ]} - > - - - - - - + ) + ) : ( + // 이메일 중복 체크 버튼 + + )} +
+ + + + + + ({ + validator(_, value) { + if (!value || getFieldValue('userPassword') === value) { + return Promise.resolve(); + } + return Promise.reject( + new Error('비밀번호가 일치하지 않습니다.'), + ); + }, + }), + ]} + > + + + + + + +
-
- - - + + -
+
diff --git a/vercel.json b/vercel.json new file mode 100644 index 00000000..d8b552c4 --- /dev/null +++ b/vercel.json @@ -0,0 +1 @@ +{ "routes": [{ "src": "/[^.]+", "dest": "/", "status": 200 }] } From a74f9fd076ba751d4027caece81ec4bc952ee8fd Mon Sep 17 00:00:00 2001 From: howooking Date: Wed, 16 Aug 2023 15:55:39 +0900 Subject: [PATCH 3/3] =?UTF-8?q?=EC=B6=94=EA=B0=80=20=EC=88=98=EC=A0=95?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- README.md | 141 ++++- src/index.css | 16 + src/page/home/calendar.tsx | 59 +- src/page/home/home.tsx | 19 +- src/page/myAccount/myAccount.tsx | 89 ++- src/page/signup/EmailChecked/email.tsx | 185 ++++++ src/page/signup/EmailChecked/emailAuth.tsx | 84 +++ src/page/signup/EmailChecked/emailChecked.tsx | 53 ++ .../signup/EmailChecked/emailVerification.tsx | 30 + .../EmailValidation/emailValidation.tsx | 30 + .../PasswordValidation/passwordValidation.tsx | 92 +++ .../signup/ProfileImageUrl/profileImg.tsx | 59 ++ .../signup/SelectPostion/selectPosition.tsx | 35 ++ src/page/signup/signup.tsx | 555 +----------------- src/page/signup/signupForm.tsx | 230 ++++++++ src/recoil/EmailRecoil.ts | 31 + src/recoil/ImgUrlAtom.ts | 6 + 17 files changed, 1133 insertions(+), 581 deletions(-) create mode 100644 src/page/signup/EmailChecked/email.tsx create mode 100644 src/page/signup/EmailChecked/emailAuth.tsx create mode 100644 src/page/signup/EmailChecked/emailChecked.tsx create mode 100644 src/page/signup/EmailChecked/emailVerification.tsx create mode 100644 src/page/signup/EmailValidation/emailValidation.tsx create mode 100644 src/page/signup/PasswordValidation/passwordValidation.tsx create mode 100644 src/page/signup/ProfileImageUrl/profileImg.tsx create mode 100644 src/page/signup/SelectPostion/selectPosition.tsx create mode 100644 src/page/signup/signupForm.tsx create mode 100644 src/recoil/EmailRecoil.ts create mode 100644 src/recoil/ImgUrlAtom.ts diff --git a/README.md b/README.md index 15434b63..f8b1d46c 100644 --- a/README.md +++ b/README.md @@ -6,7 +6,9 @@ 패스트캠퍼스 백엔드 5기 3분과 팀을 이루어 협업을 진행하였습니다. -[결과물 보러가기](https://kdt-5-mini-team-11-eifz.vercel.app/) +[기존 결과물 보러가기](https://kdt-5-mini-team-11-eifz.vercel.app/) + +[프로젝트 끝난 후 추가 결과물 보러가기](https://team-project-vacation-management.vercel.app/) ## 11조 개쩌는팀 소개 @@ -255,3 +257,140 @@ - 클라이언트는 vecel로 배포시 https로 배포가 되므로 서버와 같은 작업이 필요없다. - 그러나 로컬 환경에서 테스트를 할 때는 HttpOnly 쿠키 사용이 불가능하므로 배포 환경과 로컬 환경에서의 api를 분리하여서 작업하였다. - 해당 리포지토리는 로컬용 + +### 외부 서비스 (cloudinary) + +- 프로필 이미지 자체를 서버에 저장해서 가져오는 형식이 아닌, 업로드한 프로필 이미지를 cloudinary라는 외부 서비스를 이용해서 관리하여, 업로드한 이미지의 url만 서버에 저장해서 사용했다. + - 프로젝트 기획 단계에서 프로필 이미지를 url형식으로 가져와서 사용하기로 했는데, cloudinary라는 외부 서비스를 같은 팀원인 정우님의 추천으로 사용하게 되었고, 초기 과정에서 cloudinary를 사용하기 위해서 주말 이틀을 꼬박 ant design upload ui에 cloudinary를 연결하기 위해 헤메었다. + - 생각과는 다르게 cloudinary 서비스를 사용하기 위해서, API KEY나 API SELECT등 cloudinary 홈페이지에 초기에 적혀있던 환경변수들은 딱히 필요가 없었고, 이 때문에 많은 착오를 겪었던거 같다. + - 외부 서비스이기 때문에 이미지 업로드하는데 있어서 그러한 환경 변수는 필수적으로 있어야한다는 생각이 있었고, 관련 글을 찾아보는데도 API KEY를 관리를 해야한다. 이런 글들이 있었기 때문이었다. 하지만 주말 내내 헤메다가 찾아낸건 다름아닌 cloudinary 영문 docs에 필요한 사항이 다 적혀있었다. + - 필요한 것은 자신의 cloudinary의 이름과 preset이게 끝이었다. + - 우여곡절 끝에 정말 주말 토,일을 다 할애해서 upload를 하면 url을 받아오는데 성공했다. + - 이렇게 회원가입 부분에서 프로필 이미지 업로드가 동작하게 되었고, 추후에 마이페이지 부분에서 프로필 이미지 수정하는 부분도 맡아서 정리하게 되었다. -by. 기훈 + +### 불필요한 리렌더링과 통신 + +- 기획 설계 단계에서의 헛점이 여실히 드러나는 부분이었다고 생각드게 불필요한 리렌더링이 일어나는 것과 서버에 요청하는 것이다. + +- 헤더에 사용자 정보에 있는 연차의 상태가 업데이트 될 때 마다 리렌더링 일어나게 하였으나, 연차와 상관없는 당직을 신청하거나 삭제할 때도 같은 상태가 업데이트가 되어서 연차와 당직을 구분하는 로직을 추가하였다. + +- 메인 페이지에서 switch를 토글하면 모든 일정과 사용자의 일정을 필터링해서 볼 수 있게 했는데, 초기에는 토글 할 때 마다 서버에 통신을 해서 모든 일정을 볼 때 1년치 데이터를 가져왔고, 사용자의 일정을 볼 때에도 1년치 데이터를 또 받아온 다음에 사용자 정보 데이터도 받아와서 필터링 하는 구조였었다. + 생각해보니 처음에 모든 일정 데이터를 1년 단위로 서버에서 받아왔고, 이 받아온 데이터에는 사용자의 일정도 포함되어있는 상태였다. + 서버에서 보내온 1년 치 데이터와 사용자 정보를 받아와서 모든 일정과 사용자 일정을 필터링하는 로직을 추가해서 불필요한 통신을 줄였다. + +수정 전 +switch토글과 month의 상태가 변할 때 마다 서버에 불필요하게 요청을 함. + +``` +const listResponse = await scheduleList(year); +const infoResponse = await getMyAccount(); +``` + +scheduleList의 response에 있는 userName과 getMyAccount response에 있는 userName이 같을 시에 필터링 되게 하였었음. + +``` +useEffect(() => { + const schedule = async () => { + // getAccessTokenFromCookie를 이용해서 쿠키에 저장된 accessToken을 가져옴 + const accessToken = getAccessTokenFromCookie(); + // 엑세스 토큰이 없으면 서버에 요청하지 않음 + if (!accessToken) { + return; + } + + setIsLoading(true); + + const listResponse = await scheduleList(year); + const infoResponse = await getMyAccount(); + + // 실제 응답 데이터 추출 + const listResponseData = listResponse.data.response; + const infoResponseData = infoResponse.data.response; + + // response data를 가져오는데 그 내부에 있는 response라는 배열 데이터를 각각의 요소를 + // 아래의 형태의 객체로 변환해서 events 변수에 저장, setEvents에 전달 + const events = listResponseData + .filter( + (item: ScheduleItem) => + (isAllChecked && item.state === 'APPROVE') || + (item.userName === infoResponseData.userName && + item.state === 'APPROVE'), + ) + .map((item: ScheduleItem) => { + return { + title: item.userName, + start: item.startDate, + end: item.endDate, + color: DUTY_ANNUAL[item.scheduleType].color, + }; + }); + setEvents(events); + setIsLoading(false); + }; + schedule(); + }, [isSignedin, year, month, isAllChecked]); + +``` + +수정 후 +scheduleList response로 오는 data스키마를 수정하여 userEmail을 확인 할 수 있게 하였고, +userEmail을 리코일을 통해서 전역으로 관리하여, response의 userEmail같으면 필터링 되게 하였다. +스위치 토글과 month의 상태값 변화를 수정 전과 달리 의존성 배열에서 삭제하였다. +scheduleList는 년 단위의 데이터를 받아오니 월 단위 변경에 대해서 의존성 배열에서 필요없다고 판단하였고, +스위치 토글 또한 처음 받아온 년 단위 데이터를 이용하면 되어서 의존성 배열에서 제거했다. + +``` +useEffect(() => { + const getUsersYearlySchedules = async () => { + if (!accessToken) { + return; + } + try { + setUserYearlySchedulesLoading(true); + const listResponse = await scheduleList(year); + const listResponseData = listResponse.data.response; + + const sideMyScheduleData = listResponseData + .filter((item: mySchedule) => item.userEmail === userEmail) + .map((item: mySchedule) => { + return { + id: item.id, + key: item.id, + scheduleType: item.scheduleType, + startDate: item.startDate, + endDate: item.endDate, + state: item.state, + }; + }); + setSideMyschedule(sideMyScheduleData); + + const events = listResponseData + .filter((item: mySchedule) => item.state === 'APPROVE') + .map((item: ScheduleItem) => { + const adjustEndDate = dayjs(item.endDate) + .add(1, 'day') + .format('YYYY-MM-DD'); + return { + userEmail: item.userEmail, + title: item.userName, + start: item.startDate, + end: adjustEndDate, + color: DUTY_ANNUAL[item.scheduleType].color, + }; + }); + setEvents(events); + } catch (error) { + console.log(error); + } finally { + setUserYearlySchedulesLoading(false); + } + }; + getUsersYearlySchedules(); + }, [year, accessToken, userEmail]); +``` + +## 프로젝트가 끝난 후 수정 + +- 프로필 이미지 수정 썸네일 출력 안되는 부분 수정 및 관련 스타일 수정 +- 체크박스를 사용해서 연차/당직 구분 +- signup 컴포넌트 가독성이 좋지 않은거 같아서 일부 컴포넌트 분리작업 하였음. diff --git a/src/index.css b/src/index.css index 7b16ef62..0e3e92b1 100644 --- a/src/index.css +++ b/src/index.css @@ -48,3 +48,19 @@ display: block; } } + +.ant-upload-wrapper .ant-upload-list { + width: 148px; +} + +.ant-upload-wrapper.ant-upload-picture-card-wrapper + .ant-upload.ant-upload-select { + margin: 0; + width: 150px; + height: 150px; +} + +.ant-layout * { + vertical-align: middle; + margin: 0; +} diff --git a/src/page/home/calendar.tsx b/src/page/home/calendar.tsx index 515f6b71..950ae603 100644 --- a/src/page/home/calendar.tsx +++ b/src/page/home/calendar.tsx @@ -2,9 +2,19 @@ import { useState, useRef, Dispatch } from 'react'; import FullCalendar from '@fullcalendar/react'; import dayGridPlugin from '@fullcalendar/daygrid'; import interactionPlugin from '@fullcalendar/interaction'; -import { Switch, Button, Space, Typography, Tooltip, Skeleton } from 'antd'; +import { + Switch, + Button, + Space, + Typography, + Tooltip, + Skeleton, + Checkbox, + message, +} from 'antd'; import { LeftOutlined, RightOutlined } from '@ant-design/icons'; import { ScheduleItem } from './home'; +import { DUTY_ANNUAL } from '@/data/constants'; interface propsType { mySchedule: ScheduleItem[]; @@ -62,8 +72,47 @@ export default function Calendar({ calendarApi?.today(); }; + const options = [ + { label: '연차', value: 'ANNUAL' }, + { label: '당직', value: 'DUTY' }, + ]; + + const [messageApi, contextHolder] = message.useMessage(); + const [checkedBox, setCheckedBox] = useState(['ANNUAL', 'DUTY']); + + // eslint-disable-next-line @typescript-eslint/no-explicit-any + const handleCheckboxChange = (check: any) => { + if (check.length === 0) { + messageApi.open({ + type: 'error', + content: '최소한 하나의 옵션은 선택되어야 합니다.', + }); + return; + } + setCheckedBox(check); + }; + + const getFilteredEvents = () => { + // 현재 보여줄 이벤트를 선택 + // 전체 일정인지, 나만의 일정인지 + const currentEvents = isAllChecked ? events : mySchedule; + + // checkedBox의 상태에 따라 이벤트를 필터링 + return currentEvents.filter((event) => { + if (checkedBox.includes('ANNUAL') && checkedBox.includes('DUTY')) { + return true; // 연차와 당직 모두 선택된 경우 모든 이벤트를 반환 + } else if (checkedBox.includes('ANNUAL')) { + return event.color === DUTY_ANNUAL.ANNUAL.color; + } else if (checkedBox.includes('DUTY')) { + return event.color === DUTY_ANNUAL.DUTY.color; + } + return false; + }); + }; + return ( <> + {contextHolder}
+ +
(!accessToken); - const setReRender = useSetRecoilState(ReRenderStateAtom); - const reRender = useRecoilValue(ReRenderStateAtom); + const [reRender, setReRender] = useRecoilState(ReRenderStateAtom); // 로그아웃을 하면 isModalOpen이 !accessToken의 상태를 바로 반영하지 않음 // 따라서 useEffect로 반영이 되도록함 useEffect(() => { @@ -99,6 +99,17 @@ export default function Home() { const [userYearlySchedulesLoading, setUserYearlySchedulesLoading] = useState(false); + /* const [shcduleTypeList, setShcduleTypeList] = useState< + { + id: number; + key: number; + scheduleType: 'ANNUAL' | 'DUTY'; + startDate: string; + endDate: string; + state: 'APPROVE'; + }[] + >([]); */ + useEffect(() => { const getUsersYearlySchedules = async () => { if (!accessToken) { @@ -123,6 +134,8 @@ export default function Home() { }); setSideMyschedule(sideMyScheduleData); + /* const scheduleCheckbox = listResponseData.filter((item: mySchedule) => item.state === 'APPROVE') */ + const events = listResponseData .filter((item: mySchedule) => item.state === 'APPROVE') .map((item: ScheduleItem) => { diff --git a/src/page/myAccount/myAccount.tsx b/src/page/myAccount/myAccount.tsx index 63a87666..7ca05e9b 100644 --- a/src/page/myAccount/myAccount.tsx +++ b/src/page/myAccount/myAccount.tsx @@ -14,17 +14,20 @@ import { EditOutlined, CloseCircleOutlined, PlusOutlined, + LoadingOutlined, } from '@ant-design/icons'; import defaultProfile from '@/assets/defaultProfile.png'; import { POSITIONS } from '@/data/constants'; import formatPhoneNumber from '@/utils/formatPhonenumber'; import { useEffect, useState } from 'react'; import { getMyAccount } from '@/api/myAccount/getMyAccount'; -import { useRecoilValue } from 'recoil'; +import { useRecoilValue, useSetRecoilState, useRecoilState } from 'recoil'; import { AccessTokenAtom } from '@/recoil/AccessTokkenAtom'; import { changeMyInfo } from '@/api/myAccount/changeMyInfo'; import PasswordChangeModal from '@/page/myAccount/passwordChangeModal'; import { handleUpload } from '@/api/auth/cloudinary'; +import { ReRenderStateAtom } from '@/recoil/ReRenderStateAtom'; +import { ImgUrlAtom } from '@/recoil/ImgUrlAtom'; interface MyAccountInfoType { phoneNumber: string; @@ -52,9 +55,13 @@ export default function MyAccount() { const [editPhoneNumberInput, setEditPhonNumberInput] = useState(''); const [isModalOpen, setIsModalOpen] = useState(false); const [editprofileThumbUrl, setEditProfileThumbUrl] = useState(false); + const setReRender = useSetRecoilState(ReRenderStateAtom); const accessToken = useRecoilValue(AccessTokenAtom); + const [uploadLoading, setUploadLoading] = useState(false); + const [imgUrl, setimgUrl] = useRecoilState(ImgUrlAtom); + useEffect(() => { const getData = async () => { if (!accessToken) { @@ -114,8 +121,8 @@ export default function MyAccount() { // eslint-disable-next-line @typescript-eslint/no-explicit-any const handleChangeProfileImg = async (file: any) => { const imageUrl = await getImageUrl(file); - console.log(imageUrl); try { + setUploadLoading(true); const response = await changeMyInfo({ profileThumbUrl: imageUrl, }); @@ -124,12 +131,14 @@ export default function MyAccount() { ...prev, profileThumbUrl: imageUrl, })); + setReRender((prev) => !prev); // 성공 messageApi.open({ type: 'success', content: '프로필 이미지를 수정하였습니다.', }); } + setimgUrl(''); // eslint-disable-next-line @typescript-eslint/no-explicit-any } catch (error: any) { messageApi.open({ @@ -138,6 +147,7 @@ export default function MyAccount() { }); } finally { setEditProfileThumbUrl(false); + setUploadLoading(false); } }; @@ -178,6 +188,33 @@ export default function MyAccount() { return imageUrl; }; + const props = { + showUploadList: false, + beforeUpload: (file: any) => { + const isImage = file.type.startsWith('image/'); + if (!isImage) { + message.error('이미지 파일만 업로드할 수 있습니다!'); + return false; + } + if (imgUrl) { + // 이미 이미지가 업로드되어 있다면 새로운 업로드 중지 + message.error('이미 사진이 업로드되었습니다!'); + return false; + } + const reader = new FileReader(); + reader.readAsDataURL(file); + reader.onload = () => setimgUrl(reader.result as string); // 미리보기를 위한 이미지 URL 설정 + return true; + }, + }; + + const uploadButton = ( +
+ {uploadLoading ? : } +
Upload
+
+ ); + return (
{contextHolder} @@ -268,6 +305,7 @@ export default function MyAccount() { { setEditProfileThumbUrl(true); + setimgUrl(''); }} style={{ marginLeft: 5, fontSize: 15 }} className="icons" @@ -284,31 +322,36 @@ export default function MyAccount() { width: '200px', height: '200px', display: 'flex', - alignItems: 'center', - paddingTop: '22px', }} > - - false} /* showUploadList={false} */ +
+ - - - - + + {imgUrl ? ( + avatar + ) : ( + uploadButton + )} + + + +
) : ( diff --git a/src/page/signup/EmailChecked/email.tsx b/src/page/signup/EmailChecked/email.tsx new file mode 100644 index 00000000..cafe26d5 --- /dev/null +++ b/src/page/signup/EmailChecked/email.tsx @@ -0,0 +1,185 @@ +import { useEffect, useState } from 'react'; +import { Form, Button, message, Tooltip, Spin, Space } from 'antd'; +import { SignupFormProps } from '../signupForm'; +import { verificationEmail } from '@/api/auth/verification'; +import EmailChecked from './emailChecked'; +import EmailVerification from './emailVerification'; +import EmailAuth from './emailAuth'; +import { + IsLoadingAtom, + IsEmailCheckAtom, + ResendAtom, + TimerAtom, + EmailVerifiedAtom, + VerificationAtom, +} from '@/recoil/EmailRecoil'; +import { useRecoilState, useSetRecoilState } from 'recoil'; +import { AxiosResponse } from 'axios'; +import { RedoOutlined, LoadingOutlined } from '@ant-design/icons'; + +export interface EmailProps extends SignupFormProps { + // eslint-disable-next-line @typescript-eslint/no-explicit-any + form: any; +} + +export interface EmailVerificationProps { + handleVerificationEmail: () => Promise | null>; +} + +export default function Email({ form, messageApi }: EmailProps) { + // 이메일 중복 체크 + const [isEmailCehck, setIsEmailCheck] = useRecoilState(IsEmailCheckAtom); + // 이메일 인증 번호 재발송 ui + const setReSend = useSetRecoilState(ResendAtom); + // 이메일 인증 과정 로딩 (중복체크, 인증번호 발송, 인증확인) + const [isLoading, setIsLoading] = useRecoilState(IsLoadingAtom); + // 이메일 인증번호 확인 + const [emailVerified, setEmailVerified] = useRecoilState(EmailVerifiedAtom); + // 이메일 인증 번호 발송 + const [verification, setVerification] = useRecoilState(VerificationAtom); + // 이메일 인증 번호 발송 후에 제출 가능 타이머 + const [timer, setTimer] = useRecoilState(TimerAtom); + + const [reLoading, setReLoading] = useState(false); + + // 이메일 인증번호 발송 + const handleVerificationEmail = async () => { + setIsLoading(true); + try { + const values = await form.validateFields(['userEmail']); + const userEmail = values.userEmail; + const response = await verificationEmail(userEmail); + + if (response.status === 200) { + message.success('인증번호를 발송했습니다.'); + // 인증번호 발생이 정상적으로 이루어졌으면 타이머가 180으로 상태값 변경 + setTimer(180); + } else { + message.warning('이메일을 다시 확인해주세요.'); + } + return response; + } catch (error) { + message.error('인증번호 발송에 오류가 발생했습니다.'); + return null; + } finally { + setIsLoading(false); + setVerification(true); + } + }; + + // 타이머 변경에 따른 리렌더링이 일어나게 + useEffect(() => { + // side effect를 멈추기 위해서 exactTimer라는 변수를 선언 + let exactTimer: string | number | NodeJS.Timeout | undefined; + // 타이머가 0보다 크면 timer의 상태값이 초당 -1씩 줄어들고 + // timer가 0초이고 verification state값이 true일 때, + // reSend state값이 true가 되어서 재발송 버튼을 클릭 할 수 있다. + if (timer > 0) { + exactTimer = setTimeout( + () => setTimer((prevTimer) => prevTimer - 1), + 1000, + ); + } else if (timer === 0 && verification) { + setReSend(true); + } + // 위의 동작들이 실행되면 타이머가 멈춘다. + return () => { + if (exactTimer) { + clearTimeout(exactTimer); + } + }; + }, [timer, verification]); + + // 이메일 인증을 다시 처음부터 하게 하는 재인증 기능 + const handleReAuthentication = () => { + if (isLoading) return; + setReLoading(true); + + setTimeout(() => { + setIsEmailCheck(false); + setVerification(false); + setEmailVerified(false); + setReSend(false); + setReLoading(false); + }, 500); + }; + + return ( + <> + { + if (!value) { + return Promise.reject('필수 진행 항목입니다.'); + } + return Promise.resolve(); + }, + }, + ]} + > + {isEmailCehck ? ( + verification ? ( + emailVerified ? ( + // 인증에 성공한 경우 - 재인증 버튼 표시 +
+ +
+ ) : ( + // 인증 번호 입력 창과 제출 버튼 +
+ + + {reLoading ? ( + } /> + ) : ( + + + + )} + +
남은 시간: {timer}초
+
+ ) + ) : ( + + + {reLoading ? ( + } /> + ) : ( + + + + )} + + ) + ) : ( + + )} +
+ + ); +} diff --git a/src/page/signup/EmailChecked/emailAuth.tsx b/src/page/signup/EmailChecked/emailAuth.tsx new file mode 100644 index 00000000..a8487f64 --- /dev/null +++ b/src/page/signup/EmailChecked/emailAuth.tsx @@ -0,0 +1,84 @@ +import { Button, Input, Space, message } from 'antd'; +import { useRecoilState, useSetRecoilState } from 'recoil'; +import { + IsLoadingAtom, + ResendAtom, + EmailVerifiedAtom, + TimerAtom, +} from '@/recoil/EmailRecoil'; +import { EmailVerificationProps } from './email'; +import { checkEmailAuth } from '@/api/auth/checkEmail'; + +interface EmailAuthProps extends EmailVerificationProps { + // eslint-disable-next-line @typescript-eslint/no-explicit-any + form: any; +} + +export default function EmailAuth({ + form, + handleVerificationEmail, +}: EmailAuthProps) { + const [isLoading, setIsLoading] = useRecoilState(IsLoadingAtom); + const [reSend, setReSend] = useRecoilState(ResendAtom); + const setEmailVerified = useSetRecoilState(EmailVerifiedAtom); + const setTimer = useSetRecoilState(TimerAtom); + + // 이메일 인증번호 발송 받은 번호를 인증 + const handleEmailAuth = async () => { + try { + const userEmailData = await form.validateFields(['userEmail']); + const userEmail = userEmailData.userEmail; + const emailAuthData = await form.validateFields(['emailAuth']); + const userEmailAuth = emailAuthData.emailAuth; + const data = { userEmail, userEmailAuth }; + const response = await checkEmailAuth(data); + + if (response.data.success) { + message.success('이메일 인증 성공!'); + setEmailVerified(true); + setTimer(0); + } else { + message.error('이메일 인증 실패!'); + } + } catch (error) { + message.error('서버 요청에 실패하였습니다.'); + } + }; + + const handleReSend = async () => { + if (reSend) { + setIsLoading(true); + try { + setReSend(false); + await handleVerificationEmail(); + } catch (error) { + console.error('이메일 인증 요청 중 오류 발생:', error); + } finally { + setIsLoading(false); + } + } + }; + return ( + <> + + + + + + + ); +} diff --git a/src/page/signup/EmailChecked/emailChecked.tsx b/src/page/signup/EmailChecked/emailChecked.tsx new file mode 100644 index 00000000..ea79ad5a --- /dev/null +++ b/src/page/signup/EmailChecked/emailChecked.tsx @@ -0,0 +1,53 @@ +import { Button } from 'antd'; +import { EmailProps } from './email'; +import { checkEmail } from '@/api/auth/checkEmail'; +import { useSetRecoilState, useRecoilState } from 'recoil'; +import { IsEmailCheckAtom, IsLoadingAtom } from '@/recoil/EmailRecoil'; + +export default function EmailChecked({ messageApi, form }: EmailProps) { + const [isLoading, setIsLoading] = useRecoilState(IsLoadingAtom); + const setIsEmailCheck = useSetRecoilState(IsEmailCheckAtom); + // 이메일 중복체크 + const handleCheckEmail = async () => { + setIsLoading(true); + try { + // userEmail form을 찾아서 + const values = await form.validateFields(['userEmail']); + // userEmail을 가져옴 + const userEmail = values.userEmail; + // checkEmail에 userEmail을 매개변수로 전달달 + const response = await checkEmail(userEmail); + + if (response.data.success) { + setIsEmailCheck(true); // 중복되지 않은 이메일일 경우 + messageApi.open({ + type: 'success', + content: '사용 가능한 이메일입니다.', + }); + } + // eslint-disable-next-line @typescript-eslint/no-explicit-any + } catch (error: any) { + messageApi.open({ + type: 'error', + content: + error.response?.data.error.message || + '이메일 중복체크에 실패하였습니다.', + }); + } finally { + setIsLoading(false); + } + }; + return ( + <> + {/* 이메일 중복 체크 버튼 */} + + + ); +} diff --git a/src/page/signup/EmailChecked/emailVerification.tsx b/src/page/signup/EmailChecked/emailVerification.tsx new file mode 100644 index 00000000..7d67bccd --- /dev/null +++ b/src/page/signup/EmailChecked/emailVerification.tsx @@ -0,0 +1,30 @@ +import { Button } from 'antd'; +import { useRecoilValue } from 'recoil'; +import { IsLoadingAtom } from '@/recoil/EmailRecoil'; +import { EmailVerificationProps } from './email'; + +interface props extends EmailVerificationProps { + // eslint-disable-next-line @typescript-eslint/no-explicit-any + reLoading: any; +} + +export default function EmailVerification({ + handleVerificationEmail, + reLoading, +}: props) { + const isLoading = useRecoilValue(IsLoadingAtom); + + return ( + <> + {/* 인증 번호 전송 버튼 */} + + + ); +} diff --git a/src/page/signup/EmailValidation/emailValidation.tsx b/src/page/signup/EmailValidation/emailValidation.tsx new file mode 100644 index 00000000..01e49ea5 --- /dev/null +++ b/src/page/signup/EmailValidation/emailValidation.tsx @@ -0,0 +1,30 @@ +import { EMAIL_REGEX } from '@/data/constants'; +import { Form, Input } from 'antd'; +import { RuleObject } from 'antd/es/form'; +import { useCallback } from 'react'; + +export default function EmailValidation() { + // 이메일 유효성 검사 + const validateEmail = useCallback((_: RuleObject, value: string) => { + if (!value || EMAIL_REGEX.test(value)) { + return Promise.resolve(); + } + return Promise.reject(new Error('올바른 형식의 메일을 입력해주세요.')); + }, []); + + return ( + <> + + + + + ); +} diff --git a/src/page/signup/PasswordValidation/passwordValidation.tsx b/src/page/signup/PasswordValidation/passwordValidation.tsx new file mode 100644 index 00000000..d1e7f3c1 --- /dev/null +++ b/src/page/signup/PasswordValidation/passwordValidation.tsx @@ -0,0 +1,92 @@ +import { PASSWORD_REGEX } from '@/data/constants'; +import { Form, Input } from 'antd'; +import { RuleObject } from 'antd/es/form'; +import { useCallback } from 'react'; + +export default function PasswordValidation() { + // 비밀번호 유효성 검사 + const validatePassword = useCallback((_: RuleObject, value: string) => { + const NUMBER_REGEX = /\d/; + const SPECIAL_REGEX = /[!@#$%^&*()-+=]/; + const ENGLISH_REGEX = /[a-zA-Z]/; + + if (!value) { + return Promise.reject(new Error('비밀번호를 입력해주세요.')); + } + + if (!NUMBER_REGEX.test(value) && !ENGLISH_REGEX.test(value)) { + return Promise.reject( + new Error('최소 하나의 숫자와 영어 대소문자가 포함되어야 합니다.'), + ); + } + + if (!SPECIAL_REGEX.test(value) && !ENGLISH_REGEX.test(value)) { + return Promise.reject( + new Error('최소 하나의 특수문자와 영어 대소문자가 포함되어야 합니다.'), + ); + } + + if (!NUMBER_REGEX.test(value) && !SPECIAL_REGEX.test(value)) { + return Promise.reject( + new Error('최소 하나의 숫자와 특수문자가 포함되어야 합니다.'), + ); + } + + if (!NUMBER_REGEX.test(value)) { + return Promise.reject(new Error('최소 하나의 숫자가 포함되어야 합니다.')); + } + + if (!SPECIAL_REGEX.test(value)) { + return Promise.reject( + new Error('최소 하나의 특수문자가 포함되어야 합니다.'), + ); + } + + if (!ENGLISH_REGEX.test(value)) { + return Promise.reject( + new Error('최소 하나의 영어 대소문자가 포함되어야 합니다.'), + ); + } + + if (!PASSWORD_REGEX.test(value)) { + return Promise.reject(new Error('비밀번호는 8~16자 입니다.')); + } + + return Promise.resolve(); + }, []); + return ( + <> + + + + + ({ + validator(_, value) { + if (!value || getFieldValue('userPassword') === value) { + return Promise.resolve(); + } + return Promise.reject(new Error('비밀번호가 일치하지 않습니다.')); + }, + }), + ]} + > + + + + ); +} diff --git a/src/page/signup/ProfileImageUrl/profileImg.tsx b/src/page/signup/ProfileImageUrl/profileImg.tsx new file mode 100644 index 00000000..a1f0926c --- /dev/null +++ b/src/page/signup/ProfileImageUrl/profileImg.tsx @@ -0,0 +1,59 @@ +import { message, Form, Upload } from 'antd'; +import { PlusOutlined } from '@ant-design/icons'; +import { ImgUrlAtom } from '@/recoil/ImgUrlAtom'; +import { useRecoilState } from 'recoil'; + +export default function ProfileImg() { + const [imgUrl, setimgUrl] = useRecoilState(ImgUrlAtom); + // eslint-disable-next-line @typescript-eslint/no-explicit-any + const normFile = (e: any) => { + if (Array.isArray(e)) { + return e; + } + return e?.fileList; + }; + + const props = { + showUploadList: false, + beforeUpload: (file: any) => { + const isImage = file.type.startsWith('image/'); + if (!isImage) { + message.error('이미지 파일만 업로드할 수 있습니다!'); + return false; + } + if (imgUrl) { + // 이미 이미지가 업로드되어 있다면 새로운 업로드 중지 + message.error('이미 사진이 업로드되었습니다!'); + return false; + } + const reader = new FileReader(); + reader.readAsDataURL(file); + reader.onload = () => setimgUrl(reader.result as string); // 미리보기를 위한 이미지 URL 설정 + return true; + }, + }; + + return ( + <> + + + {imgUrl ? ( + avatar + ) : ( +
+ +
Upload
+
+ )} +
+
+ + ); +} diff --git a/src/page/signup/SelectPostion/selectPosition.tsx b/src/page/signup/SelectPostion/selectPosition.tsx new file mode 100644 index 00000000..7d906c25 --- /dev/null +++ b/src/page/signup/SelectPostion/selectPosition.tsx @@ -0,0 +1,35 @@ +import { POSITIONS } from '@/data/constants'; +import { Form, Select } from 'antd'; + +export default function SelectPosition() { + const { Option } = Select; + // 직급 선택 동적으로 생성 + 매니저 선택 옵션이 생기지 않게 filter를 이용해서 제거 + // key가 MANAGER가 아니면 통과 + const selectedPositionOptions = Object.keys(POSITIONS) + .filter((key) => key !== 'MANAGER') + .map((key) => { + return ( + + ); + }); + return ( + <> + + + + + ); +} diff --git a/src/page/signup/signup.tsx b/src/page/signup/signup.tsx index 89b6a3ca..1568dffa 100644 --- a/src/page/signup/signup.tsx +++ b/src/page/signup/signup.tsx @@ -1,336 +1,8 @@ -import { useCallback, useState, useEffect } from 'react'; -import { - Button, - Input, - Form, - Select, - Card, - Upload, - message, - Space, -} from 'antd'; -import { PlusOutlined } from '@ant-design/icons'; -import { useNavigate } from 'react-router-dom'; -import { EMAIL_REGEX, PASSWORD_REGEX, POSITIONS } from '@/data/constants'; -import { RuleObject } from 'antd/es/form'; -import { handleUpload } from '@/api/auth/cloudinary'; -import { signup } from '@/api/auth/signup'; -import { checkEmail, checkEmailAuth } from '@/api/auth/checkEmail'; -import { verificationEmail } from '@/api/auth/verification'; -import { useSetRecoilState } from 'recoil'; -import { AccessTokenAtom } from '@/recoil/AccessTokkenAtom'; -import { setAccessTokenToCookie } from '@/utils/cookies'; -import ButtonGroup from 'antd/es/button/button-group'; - -interface valuseType { - confirm_password: string; - phone: string; - position: string; - // eslint-disable-next-line @typescript-eslint/no-explicit-any - profileThumbUrl: any; - userEmail: string; - userPassword: string; - userName: string; -} +import { Card, message } from 'antd'; +import SignupForm from './signupForm'; export default function Signup() { - // 이메일 중복 체크 - const [isEmailCehck, setIsEmailCheck] = useState(false); - // 이메일 인증 번호 발송 - const [verification, setVerification] = useState(false); - // 이메일 인증 번호 발송 후에 제출 가능 타이머 - const [timer, setTimer] = useState(0); - // 이메일 인증 번호 재발송 ui - const [reSend, setReSend] = useState(false); - // 이메일 인증 과정 로딩 (중복체크, 인증번호 발송, 인증확인) - const [isLoading, setIsLoading] = useState(false); - // 이메일 인증번호 확인 - const [emailVerified, setEmailVerified] = useState(false); - // 회원가입 통신 과정 로딩 ui - const [isSigningUp, setIsSigningUp] = useState(false); - // antd message(화면 상단에 뜨는 메세지)기능 const [messageApi, contextHolder] = message.useMessage(); - - const setAccessToken = useSetRecoilState(AccessTokenAtom); - const navigate = useNavigate(); - const { Option } = Select; - const [form] = Form.useForm(); - - const onFinish = async (values: valuseType) => { - // 클라우디너리로 전송한 이미지 url 가져옴 - const imageUrl = await getImageUrl(values); - - // 가져온 이미지 url만 profileThumbUrl에 전달하기 위해서 - // values객체를 전개연산자를 이용해서 newValues에 모든 속성을 복사하고, - // profileThumbUrl: imageUrl을 마지막에 사용해서 덮어 씌움 - const newValues = { - ...values, - profileThumbUrl: imageUrl, - }; - - setIsSigningUp(true); - try { - const response = await signup(newValues); - - // 로그인 성공 - if (response.status === 200) { - const { accessToken } = response.data.response; - - // 쿠키에 저장 - setAccessTokenToCookie(accessToken); - - // recoil에 저장 - setAccessToken(accessToken); - - // 안내메시지 - messageApi.open({ - type: 'success', - content: '회원가입이 완료되었습니다.', - }); - - // 회원가입이 성공한 경우 홈으로 이동 - setTimeout(() => { - navigate('/'); - }, 1000); - - return; - } - // eslint-disable-next-line @typescript-eslint/no-explicit-any - } catch (error: any) { - messageApi.open({ - type: 'error', - content: - error.response?.data.error.message || - '회원가입에 실패하였습니다. 관리자에게 문의하세요.', - }); - } finally { - setIsSigningUp(false); - } - }; - - // cloudinary를 이용해서 이미지url이 없으면 null, 있으면 이미지url을 반환 - const getImageUrl = async (values: valuseType) => { - let imageUrl = null; - - try { - if (values.profileThumbUrl && values.profileThumbUrl.length > 0) { - const response = await handleUpload( - values.profileThumbUrl[0].originFileObj, - ); - if (response?.status === 200) { - const data = response.data; - imageUrl = data.url; // 이미지 URL을 받아옴 - } else { - throw new Error('이미지 업로드에 실패하였습니다.'); - } - } - } catch (error) { - console.error('이미지 업로드중 오류 발생:', error); - imageUrl = null; // 오류가 발생했으므로 imageUrl을 null로 설정 - } - - return imageUrl; - }; - - // eslint-disable-next-line @typescript-eslint/no-explicit-any - const normFile = (e: any) => { - if (Array.isArray(e)) { - return e; - } - return e?.fileList; - }; - - // 이메일 유효성 검사 - const validateEmail = useCallback((_: RuleObject, value: string) => { - if (!value || EMAIL_REGEX.test(value)) { - return Promise.resolve(); - } - return Promise.reject(new Error('올바른 형식의 메일을 입력해주세요.')); - }, []); - - // 비밀번호 유효성 검사 - const validatePassword = useCallback((_: RuleObject, value: string) => { - const NUMBER_REGEX = /\d/; - const SPECIAL_REGEX = /[!@#$%^&*()-+=]/; - const ENGLISH_REGEX = /[a-zA-Z]/; - - if (!value) { - return Promise.reject(new Error('비밀번호를 입력해주세요.')); - } - - if (!NUMBER_REGEX.test(value) && !ENGLISH_REGEX.test(value)) { - return Promise.reject( - new Error('최소 하나의 숫자와 영어 대소문자가 포함되어야 합니다.'), - ); - } - - if (!SPECIAL_REGEX.test(value) && !ENGLISH_REGEX.test(value)) { - return Promise.reject( - new Error('최소 하나의 특수문자와 영어 대소문자가 포함되어야 합니다.'), - ); - } - - if (!NUMBER_REGEX.test(value) && !SPECIAL_REGEX.test(value)) { - return Promise.reject( - new Error('최소 하나의 숫자와 특수문자가 포함되어야 합니다.'), - ); - } - - if (!NUMBER_REGEX.test(value)) { - return Promise.reject(new Error('최소 하나의 숫자가 포함되어야 합니다.')); - } - - if (!SPECIAL_REGEX.test(value)) { - return Promise.reject( - new Error('최소 하나의 특수문자가 포함되어야 합니다.'), - ); - } - - if (!ENGLISH_REGEX.test(value)) { - return Promise.reject( - new Error('최소 하나의 영어 대소문자가 포함되어야 합니다.'), - ); - } - - if (!PASSWORD_REGEX.test(value)) { - return Promise.reject(new Error('비밀번호는 8~16자 입니다.')); - } - - return Promise.resolve(); - }, []); - - // 직급 선택 동적으로 생성 + 매니저 선택 옵션이 생기지 않게 filter를 이용해서 제거 - // key가 MANAGER가 아니면 통과 - const selectedPositionOptions = Object.keys(POSITIONS) - .filter((key) => key !== 'MANAGER') - .map((key) => { - return ( - - ); - }); - - // 이메일 중복체크 - const handleCheckEmail = async () => { - setIsLoading(true); - try { - // userEmail form을 찾아서 - const values = await form.validateFields(['userEmail']); - // userEmail을 가져옴 - const userEmail = values.userEmail; - // checkEmail에 userEmail을 매개변수로 전달달 - const response = await checkEmail(userEmail); - - if (response.data.success) { - setIsEmailCheck(true); // 중복되지 않은 이메일일 경우 - message.success('사용 가능한 이메일입니다.'); - } - // eslint-disable-next-line @typescript-eslint/no-explicit-any - } catch (error: any) { - messageApi.open({ - type: 'error', - content: - error.response?.data.error.message || - '이메일 중복체크에 실패하였습니다. 관리자에게 문의하세요.', - }); - } finally { - setIsLoading(false); - } - }; - - // 이메일 인증번호 발송 - const handleVerificationEmail = async () => { - setIsLoading(true); - try { - const values = await form.validateFields(['userEmail']); - const userEmail = values.userEmail; - const response = await verificationEmail(userEmail); - - if (response.status === 200) { - message.success('인증번호를 발송했습니다.'); - // 인증번호 발생이 정상적으로 이루어졌으면 타이머가 180으로 상태값 변경 - setTimer(180); - } else { - message.warning('이메일을 다시 확인해주세요.'); - } - return response; - } catch (error) { - message.error('인증번호 발송에 오류가 발생했습니다.'); - return null; - } finally { - setIsLoading(false); - setVerification(true); - } - }; - - const handleReSend = async () => { - if (reSend) { - setIsLoading(true); - try { - setReSend(false); - await handleVerificationEmail(); - } catch (error) { - console.error('이메일 인증 요청 중 오류 발생:', error); - } finally { - setIsLoading(false); - } - } - }; - - // 타이머 변경에 따른 리렌더링이 일어나게 - useEffect(() => { - // side effect를 멈추기 위해서 exactTimer라는 변수를 선언 - let exactTimer: string | number | NodeJS.Timeout | undefined; - // 타이머가 0보다 크면 timer의 상태값이 초당 -1씩 줄어들고 - // timer가 0초이고 verification state값이 true일 때, - // reSend state값이 true가 되어서 재발송 버튼을 클릭 할 수 있다. - if (timer > 0) { - exactTimer = setTimeout( - () => setTimer((prevTimer) => prevTimer - 1), - 1000, - ); - } else if (timer === 0 && verification) { - setReSend(true); - } - // 위의 동작들이 실행되면 타이머가 멈춘다. - return () => { - if (exactTimer) { - clearTimeout(exactTimer); - } - }; - }, [timer, verification]); - - // 이메일 인증번호 발송 받은 번호를 인증 - const handleEmailAuth = async () => { - try { - const userEmailData = await form.validateFields(['userEmail']); - const userEmail = userEmailData.userEmail; - const emailAuthData = await form.validateFields(['emailAuth']); - const userEmailAuth = emailAuthData.emailAuth; - const data = { userEmail, userEmailAuth }; - const response = await checkEmailAuth(data); - - if (response.data.success) { - message.success('이메일 인증 성공!'); - setEmailVerified(true); - setTimer(0); - } else { - message.error('이메일 인증 실패!'); - } - } catch (error) { - message.error('서버 요청에 실패하였습니다.'); - } - }; - - // 이메일 인증을 다시 처음부터 하게 하는 재인증 기능 - const handleReAuthentication = () => { - setIsEmailCheck(false); - setVerification(false); - setEmailVerified(false); - setReSend(false); - }; - return (
{contextHolder} -
- - - - - - - - - { - if (!value) { - return Promise.reject('필수 진행 항목입니다.'); - } - return Promise.resolve(); - }, - }, - ]} - > - {isEmailCehck ? ( - verification ? ( - emailVerified ? ( - // 인증에 성공한 경우 - 재인증 버튼 표시 -
- -
- ) : ( - // 인증 번호 입력 창과 제출 버튼 -
- - - - - -
남은 시간: {timer}초
-
- ) - ) : ( - // 인증 번호 전송 버튼 - - ) - ) : ( - // 이메일 중복 체크 버튼 - - )} -
- - - - - - ({ - validator(_, value) { - if (!value || getFieldValue('userPassword') === value) { - return Promise.resolve(); - } - return Promise.reject( - new Error('비밀번호가 일치하지 않습니다.'), - ); - }, - }), - ]} - > - - - - - - -
- - - - - false}> - - - -
- - - - - - +
); diff --git a/src/page/signup/signupForm.tsx b/src/page/signup/signupForm.tsx new file mode 100644 index 00000000..1fb7e651 --- /dev/null +++ b/src/page/signup/signupForm.tsx @@ -0,0 +1,230 @@ +import { useState } from 'react'; +import { Button, Input, Form } from 'antd'; +import { useNavigate } from 'react-router-dom'; +import { handleUpload } from '@/api/auth/cloudinary'; +import { signup } from '@/api/auth/signup'; +import { useSetRecoilState } from 'recoil'; +import { AccessTokenAtom } from '@/recoil/AccessTokkenAtom'; +import { setAccessTokenToCookie } from '@/utils/cookies'; +import ButtonGroup from 'antd/es/button/button-group'; +import Email from './EmailChecked/email'; +import EmailValidation from './EmailValidation/emailValidation'; +import PasswordValidation from './PasswordValidation/passwordValidation'; +import ProfileImg from './ProfileImageUrl/profileImg'; +import SelectPosition from './SelectPostion/selectPosition'; +import { + IsEmailCheckAtom, + ResendAtom, + EmailVerifiedAtom, + VerificationAtom, +} from '@/recoil/EmailRecoil'; + +interface valuseType { + confirm_password: string; + phone: string; + position: string; + // eslint-disable-next-line @typescript-eslint/no-explicit-any + profileThumbUrl: any; + userEmail: string; + userPassword: string; + userName: string; +} + +export interface SignupFormProps { + // eslint-disable-next-line @typescript-eslint/no-explicit-any + messageApi: any; +} + +export default function SignupForm({ messageApi }: SignupFormProps) { + // 회원가입 통신 과정 로딩 ui + const [isSigningUp, setIsSigningUp] = useState(false); + + const [form] = Form.useForm(); + + const setAccessToken = useSetRecoilState(AccessTokenAtom); + const navigate = useNavigate(); + + const setIsEmailCheck = useSetRecoilState(IsEmailCheckAtom); + const setReSend = useSetRecoilState(ResendAtom); + const setEmailVerified = useSetRecoilState(EmailVerifiedAtom); + const setVerification = useSetRecoilState(VerificationAtom); + + const onFinish = async (values: valuseType) => { + // 클라우디너리로 전송한 이미지 url 가져옴 + const imageUrl = await getImageUrl(values); + + // 가져온 이미지 url만 profileThumbUrl에 전달하기 위해서 + // values객체를 전개연산자를 이용해서 newValues에 모든 속성을 복사하고, + // profileThumbUrl: imageUrl을 마지막에 사용해서 덮어 씌움 + const newValues = { + ...values, + profileThumbUrl: imageUrl, + }; + + setIsSigningUp(true); + try { + const response = await signup(newValues); + + // 로그인 성공 + if (response.status === 200) { + const { accessToken } = response.data.response; + + // 쿠키에 저장 + setAccessTokenToCookie(accessToken); + + // recoil에 저장 + setAccessToken(accessToken); + + // 안내메시지 + messageApi.open({ + type: 'success', + content: '회원가입이 완료되었습니다.', + }); + + // 회원가입이 성공한 경우 홈으로 이동 + setTimeout(() => { + navigate('/'); + }, 1000); + + return; + } + // eslint-disable-next-line @typescript-eslint/no-explicit-any + } catch (error: any) { + messageApi.open({ + type: 'error', + content: + error.response?.data.error.message || + '회원가입에 실패하였습니다. 관리자에게 문의하세요.', + }); + } finally { + setIsSigningUp(false); + } + }; + + // cloudinary를 이용해서 이미지url이 없으면 null, 있으면 이미지url을 반환 + const getImageUrl = async (values: valuseType) => { + let imageUrl = null; + + try { + if (values.profileThumbUrl && values.profileThumbUrl.length > 0) { + const response = await handleUpload( + values.profileThumbUrl[0].originFileObj, + ); + if (response?.status === 200) { + const data = response.data; + imageUrl = data.url; // 이미지 URL을 받아옴 + } else { + throw new Error('이미지 업로드에 실패하였습니다.'); + } + } + } catch (error) { + console.error('이미지 업로드중 오류 발생:', error); + imageUrl = null; // 오류가 발생했으므로 imageUrl을 null로 설정 + } + + return imageUrl; + }; + + const goHomeAndResetForm = () => { + // 홈으로 이동 + navigate('/'); + setIsEmailCheck(false); + setEmailVerified(false); + setVerification(false); + setReSend(false); + + // Form의 값을 초기화 + form.resetFields(); + }; + + return ( + <> +
+ + + + + + + + + + + + + + +
+ + + +
+ + + + + + + + ); +} diff --git a/src/recoil/EmailRecoil.ts b/src/recoil/EmailRecoil.ts new file mode 100644 index 00000000..84c61162 --- /dev/null +++ b/src/recoil/EmailRecoil.ts @@ -0,0 +1,31 @@ +import { atom } from 'recoil'; + +export const IsEmailCheckAtom = atom({ + key: 'IsEmailCheckAtom', + default: false, +}); + +export const IsLoadingAtom = atom({ + key: 'IsLoadingAtom', + default: false, +}); + +export const ResendAtom = atom({ + key: 'ResendAtom', + default: false, +}); + +export const EmailVerifiedAtom = atom({ + key: 'EmailVerifiedAtom', + default: false, +}); + +export const TimerAtom = atom({ + key: 'TimerAtom', + default: 0, +}); + +export const VerificationAtom = atom({ + key: 'VerificationAtom ', + default: false, +}); diff --git a/src/recoil/ImgUrlAtom.ts b/src/recoil/ImgUrlAtom.ts new file mode 100644 index 00000000..9dd717bd --- /dev/null +++ b/src/recoil/ImgUrlAtom.ts @@ -0,0 +1,6 @@ +import { atom } from 'recoil'; + +export const ImgUrlAtom = atom({ + key: 'ImgUrlAtom', + default: '', +});