diff --git a/.gitmodules b/.gitmodules
deleted file mode 100644
index 9b4f4025..00000000
--- a/.gitmodules
+++ /dev/null
@@ -1,3 +0,0 @@
-[submodule "frontend/packages/crawlab-ui"]
- path = frontend/packages/crawlab-ui
- url = git@github.com:crawlab-team/crawlab-ui.git
diff --git a/frontend/.Dockerfile.swp b/frontend/.Dockerfile.swp
new file mode 100644
index 00000000..7dc4d5c4
Binary files /dev/null and b/frontend/.Dockerfile.swp differ
diff --git a/frontend/.dockerignore b/frontend/.dockerignore
deleted file mode 100644
index 9bc955c7..00000000
--- a/frontend/.dockerignore
+++ /dev/null
@@ -1,33 +0,0 @@
-# Ignore the .npmrc file
-.npmrc
-
-# Ignore node_modules
-node_modules
-
-# local env files
-.env.local
-.env.*.local
-
-# Editor directories and files
-.idea
-.vscode
-*.suo
-*.ntvs*
-*.njsproj
-*.sln
-*.sw?
-
-# Log files
-npm-debug.log*
-yarn-debug.log*
-yarn-error.log*
-pnpm-debug.log*
-
-# Ignore stats.html
-stats.html
-
-# Ignore .turbo
-.turbo
-
-# Ignore tmp directory
-tmp/
diff --git a/frontend/.editorconfig b/frontend/.editorconfig
deleted file mode 100644
index a3425fe4..00000000
--- a/frontend/.editorconfig
+++ /dev/null
@@ -1,40 +0,0 @@
-root = true
-
-[*]
-tab_width = 2
-# charset = utf-8
-# end_of_line = lf
-# indent_size = 4
-# indent_style = space
-# insert_final_newline = true
-# max_line_length = 120
-# tab_width = 4
-# trim_trailing_whitespace = true
-
-[*.ts]
-indent_size = 2
-
-[*.scss]
-indent_size = 2
-
-[{*.ats, *.ts}]
-# indent_size = 2
-# tab_width = 2
-
-[{*.js, *.cjs}]
-# indent_size = 2
-# tab_width = 2
-
-[{*.sht, *.html, *.shtm, *.shtml, *.htm, *.ng}]
-# indent_size = 2
-# tab_width = 2
-
-[{.analysis_options, *.yml, *.yaml}]
-# indent_size = 2
-
-[{.babelrc, .prettierrc, .stylelintrc, .eslintrc, jest.config, *.json, *.jsb3, *.jsb2, *.bowerrc}]
-# indent_size = 2
-
-[vue.config.js]
-indent_size = 2
-tab_width = 2
diff --git a/frontend/.env b/frontend/.env
deleted file mode 100644
index f0fa7c0a..00000000
--- a/frontend/.env
+++ /dev/null
@@ -1 +0,0 @@
-VITE_APP_API_BASE_URL=
diff --git a/frontend/.npmrc b/frontend/.npmrc
deleted file mode 100644
index 38f11c64..00000000
--- a/frontend/.npmrc
+++ /dev/null
@@ -1 +0,0 @@
-registry=https://registry.npmjs.org
diff --git a/frontend/Dockerfile b/frontend/Dockerfile
index 84cefded..69c50de0 100644
--- a/frontend/Dockerfile
+++ b/frontend/Dockerfile
@@ -12,7 +12,7 @@ WORKDIR /app
RUN npm install -g pnpm
# Copy application code
-ADD . .
+ADD crawlab-ui/* .
# Install project dependencies
RUN pnpm install
diff --git a/frontend/apps/crawlab/package.json b/frontend/apps/crawlab/package.json
deleted file mode 100644
index bf020617..00000000
--- a/frontend/apps/crawlab/package.json
+++ /dev/null
@@ -1,31 +0,0 @@
-{
- "name": "crawlab",
- "version": "0.7.0-rc7",
- "scripts": {
- "dev": "pnpm run serve",
- "serve": "vite",
- "preview": "vite preview",
- "build": "vite build",
- "build:analyze": "vite build --mode analyze",
- "build:serve": "serve dist",
- "test": "jest"
- },
- "dependencies": {
- "crawlab-ui": "workspace:crawlab-ui"
- },
- "devDependencies": {
- "@types/node": "^20.14.8",
- "@vitejs/plugin-vue": "^5.0.5",
- "@vue/compiler-sfc": "^3.2.45",
- "pnpm": "^9",
- "rollup-plugin-external-globals": "^0.10.0",
- "rollup-plugin-visualizer": "^5.12.0",
- "sass": "^1.56.1",
- "typescript": "^5",
- "vite": "^5.4.6",
- "vite-aliases": "^0.11.7",
- "vite-plugin-dynamic-import": "^1.5.0",
- "vite-plugin-externalize-deps": "^0.8.0",
- "vue-tsc": "^2.0.22"
- }
-}
diff --git a/frontend/apps/crawlab/pnpm-lock.yaml b/frontend/apps/crawlab/pnpm-lock.yaml
deleted file mode 100644
index e977a6c7..00000000
--- a/frontend/apps/crawlab/pnpm-lock.yaml
+++ /dev/null
@@ -1,4511 +0,0 @@
-lockfileVersion: '9.0'
-
-settings:
- autoInstallPeers: true
- excludeLinksFromLockfile: false
-
-importers:
-
- .:
- dependencies:
- '@element-plus/icons-vue':
- specifier: ^2.3.1
- version: 2.3.1(vue@3.5.12(typescript@https://registry.npmmirror.com/typescript/-/typescript-4.9.3.tgz))
- '@fortawesome/fontawesome-common-types':
- specifier: ^6.5.2
- version: 6.6.0
- '@fortawesome/fontawesome-svg-core':
- specifier: ^6.5.2
- version: 6.6.0
- '@fortawesome/free-brands-svg-icons':
- specifier: ^6.5.2
- version: 6.6.0
- '@fortawesome/free-regular-svg-icons':
- specifier: ^6.5.2
- version: 6.6.0
- '@fortawesome/free-solid-svg-icons':
- specifier: ^6.5.2
- version: 6.6.0
- '@fortawesome/vue-fontawesome':
- specifier: ^3.0.8
- version: 3.0.8(@fortawesome/fontawesome-svg-core@6.6.0)(vue@3.5.12(typescript@https://registry.npmmirror.com/typescript/-/typescript-4.9.3.tgz))
- '@popperjs/core':
- specifier: ^2.11.8
- version: 2.11.8
- '@types/getos':
- specifier: ^3.0.4
- version: 3.0.4
- '@types/humanize-duration':
- specifier: ^3.27.4
- version: 3.27.4
- '@types/javascript-time-ago':
- specifier: ^2.0.8
- version: 2.0.8
- '@types/md5':
- specifier: ^2.3.5
- version: 2.3.5
- '@types/pinyin':
- specifier: ^2.10.2
- version: 2.10.2
- '@types/showdown':
- specifier: ^2.0.6
- version: 2.0.6
- '@types/url-join':
- specifier: ^4.0.3
- version: 4.0.3
- async-validator:
- specifier: ^4.2.5
- version: 4.2.5
- atom-material-icons:
- specifier: ^3.0.0
- version: 3.0.0
- axios:
- specifier: ^1.7.2
- version: 1.7.7
- chart.js:
- specifier: ^4.4.3
- version: 4.4.5
- chartjs-adapter-date-fns:
- specifier: ^3.0.0
- version: 3.0.0(chart.js@4.4.5)(date-fns@4.1.0)
- core-js:
- specifier: ^3.37.1
- version: 3.38.1
- crawlab-ui:
- specifier: 0.7.0-rc3
- version: 0.7.0-rc3(dlwfpvqjxhvrevoazxvsg7egqu)
- crawlab-vue3-dropzone:
- specifier: 3.0.3
- version: 3.0.3(vue@3.5.12(typescript@https://registry.npmmirror.com/typescript/-/typescript-4.9.3.tgz))
- cron-parser:
- specifier: ^4.9.0
- version: 4.9.0
- cronstrue:
- specifier: ^2.50.0
- version: 2.50.0
- dayjs:
- specifier: ^1.11.11
- version: 1.11.13
- element-plus:
- specifier: ^2.7.6
- version: 2.8.5(vue@3.5.12(typescript@https://registry.npmmirror.com/typescript/-/typescript-4.9.3.tgz))
- font-awesome:
- specifier: ^4.7.0
- version: 4.7.0
- humanize-duration:
- specifier: ^3.32.1
- version: 3.32.1
- javascript-time-ago:
- specifier: ^2.5.10
- version: 2.5.11
- md5:
- specifier: ^2.3.0
- version: 2.3.0
- monaco-editor:
- specifier: ^0.50.0
- version: 0.50.0
- normalize.css:
- specifier: ^8.0.1
- version: 8.0.1
- os-name:
- specifier: ^5.0.1
- version: 5.0.1
- pinyin:
- specifier: ^2.10.2
- version: 2.11.2
- point-cluster:
- specifier: ^3.1.8
- version: 3.1.8
- showdown:
- specifier: ^2.1.0
- version: 2.1.0
- three:
- specifier: ^0.166.0
- version: 0.166.1
- url-join:
- specifier: ^4.0.1
- version: 4.0.1
- vue:
- specifier: ^3.4
- version: 3.5.12(typescript@https://registry.npmmirror.com/typescript/-/typescript-4.9.3.tgz)
- vue-clipboard3:
- specifier: ^2.0.0
- version: 2.0.0
- vue-i18n:
- specifier: 9.13.1
- version: 9.13.1(vue@3.5.12(typescript@https://registry.npmmirror.com/typescript/-/typescript-4.9.3.tgz))
- vue-router:
- specifier: 4.4.0
- version: 4.4.0(vue@3.5.12(typescript@https://registry.npmmirror.com/typescript/-/typescript-4.9.3.tgz))
- vuex:
- specifier: ^4.1.0
- version: 4.1.0(vue@3.5.12(typescript@https://registry.npmmirror.com/typescript/-/typescript-4.9.3.tgz))
- devDependencies:
- '@types/node':
- specifier: ^18.11.10
- version: https://registry.npmmirror.com/@types/node/-/node-18.11.10.tgz
- '@vitejs/plugin-vue':
- specifier: ^3.2.0
- version: https://registry.npmmirror.com/@vitejs/plugin-vue/-/plugin-vue-3.2.0.tgz(vite@https://registry.npmmirror.com/vite/-/vite-3.2.4.tgz(@types/node@https://registry.npmmirror.com/@types/node/-/node-18.11.10.tgz)(sass@1.80.1))(vue@3.5.12(typescript@https://registry.npmmirror.com/typescript/-/typescript-4.9.3.tgz))
- '@vue/compiler-sfc':
- specifier: ^3.2.45
- version: https://registry.npmmirror.com/@vue/compiler-sfc/-/compiler-sfc-3.2.45.tgz
- rollup-plugin-external-globals:
- specifier: ^0.8.0
- version: 0.8.0(rollup@3.26.2)
- rollup-plugin-visualizer:
- specifier: ^5.9.2
- version: 5.9.2(rollup@3.26.2)
- typescript:
- specifier: ^4.6.4
- version: https://registry.npmmirror.com/typescript/-/typescript-4.9.3.tgz
- vite:
- specifier: ^3.2.4
- version: https://registry.npmmirror.com/vite/-/vite-3.2.4.tgz(@types/node@https://registry.npmmirror.com/@types/node/-/node-18.11.10.tgz)(sass@1.80.1)
- vite-aliases:
- specifier: ^0.9.7
- version: https://registry.npmmirror.com/vite-aliases/-/vite-aliases-0.9.7.tgz(@types/node@https://registry.npmmirror.com/@types/node/-/node-18.11.10.tgz)(sass@1.80.1)
- vite-plugin-dynamic-import:
- specifier: ^1.2.4
- version: https://registry.npmmirror.com/vite-plugin-dynamic-import/-/vite-plugin-dynamic-import-1.2.4.tgz
- vite-plugin-externalize-deps:
- specifier: ^0.7.0
- version: 0.7.0(vite@https://registry.npmmirror.com/vite/-/vite-3.2.4.tgz(@types/node@https://registry.npmmirror.com/@types/node/-/node-18.11.10.tgz)(sass@1.80.1))
- vue-tsc:
- specifier: ^1.0.9
- version: https://registry.npmmirror.com/vue-tsc/-/vue-tsc-1.0.11.tgz(typescript@https://registry.npmmirror.com/typescript/-/typescript-4.9.3.tgz)
-
-packages:
-
- '@ampproject/remapping@2.3.0':
- resolution: {integrity: sha512-30iZtAPgz+LTIYoeivqYo853f02jBYSd5uGnGpkFV0M3xOt9aN73erkgYAmZU43x4VfqcnLxW9Kpg3R5LC4YYw==}
- engines: {node: '>=6.0.0'}
-
- '@arr/every@1.0.1':
- resolution: {integrity: sha512-UQFQ6SgyJ6LX42W8rHCs8KVc0JS0tzVL9ct4XYedJukskYVWTo49tNiMEK9C2HTyarbNiT/RVIRSY82vH+6sTg==}
- engines: {node: '>=4'}
-
- '@babel/helper-string-parser@7.25.7':
- resolution: {integrity: sha512-CbkjYdsJNHFk8uqpEkpCvRs3YRp9tY6FmFY7wLMSYuGYkrdUi7r2lc4/wqsvlHoMznX3WJ9IP8giGPq68T/Y6g==}
- engines: {node: '>=6.9.0'}
-
- '@babel/helper-string-parser@https://registry.npmmirror.com/@babel/helper-string-parser/-/helper-string-parser-7.19.4.tgz':
- resolution: {integrity: sha512-nHtDoQcuqFmwYNYPz3Rah5ph2p8PFeFCsZk9A/48dPc/rGocJ5J3hAAZ7pb76VWX3fZKu+uEr/FhH5jLx7umrw==, tarball: https://registry.npmmirror.com/@babel/helper-string-parser/-/helper-string-parser-7.19.4.tgz}
- version: 7.19.4
- engines: {node: '>=6.9.0'}
-
- '@babel/helper-validator-identifier@7.25.7':
- resolution: {integrity: sha512-AM6TzwYqGChO45oiuPqwL2t20/HdMC1rTPAesnBCgPCSF1x3oN9MVUwQV2iyz4xqWrctwK5RNC8LV22kaQCNYg==}
- engines: {node: '>=6.9.0'}
-
- '@babel/helper-validator-identifier@https://registry.npmmirror.com/@babel/helper-validator-identifier/-/helper-validator-identifier-7.19.1.tgz':
- resolution: {integrity: sha512-awrNfaMtnHUr653GgGEs++LlAvW6w+DcPrOliSMXWCKo597CwL5Acf/wWdNkf/tfEQE3mjkeD1YOVZOUV/od1w==, tarball: https://registry.npmmirror.com/@babel/helper-validator-identifier/-/helper-validator-identifier-7.19.1.tgz}
- version: 7.19.1
- engines: {node: '>=6.9.0'}
-
- '@babel/parser@7.25.8':
- resolution: {integrity: sha512-HcttkxzdPucv3nNFmfOOMfFf64KgdJVqm1KaCm25dPGMLElo9nsLvXeJECQg8UzPuBGLyTSA0ZzqCtDSzKTEoQ==}
- engines: {node: '>=6.0.0'}
- hasBin: true
-
- '@babel/parser@https://registry.npmmirror.com/@babel/parser/-/parser-7.20.5.tgz':
- resolution: {integrity: sha512-r27t/cy/m9uKLXQNWWebeCUHgnAZq0CpG1OwKRxzJMP1vpSU4bSIK2hq+/cp0bQxetkXx38n09rNu8jVkcK/zA==, tarball: https://registry.npmmirror.com/@babel/parser/-/parser-7.20.5.tgz}
- version: 7.20.5
- engines: {node: '>=6.0.0'}
- hasBin: true
-
- '@babel/types@7.25.8':
- resolution: {integrity: sha512-JWtuCu8VQsMladxVz/P4HzHUGCAwpuqacmowgXFs5XjxIgKuNjnLokQzuVjlTvIzODaDmpjT3oxcC48vyk9EWg==}
- engines: {node: '>=6.9.0'}
-
- '@babel/types@https://registry.npmmirror.com/@babel/types/-/types-7.20.5.tgz':
- resolution: {integrity: sha512-c9fst/h2/dcF7H+MJKZ2T0KjEQ8hY/BNnDk/H3XY8C4Aw/eWQXWn/lWntHF9ooUBnGmEvbfGrTgLWc+um0YDUg==, tarball: https://registry.npmmirror.com/@babel/types/-/types-7.20.5.tgz}
- version: 7.20.5
- engines: {node: '>=6.9.0'}
-
- '@codemirror/autocomplete@6.18.1':
- resolution: {integrity: sha512-iWHdj/B1ethnHRTwZj+C1obmmuCzquH29EbcKr0qIjA9NfDeBDJ7vs+WOHsFeLeflE4o+dHfYndJloMKHUkWUA==}
- peerDependencies:
- '@codemirror/language': ^6.0.0
- '@codemirror/state': ^6.0.0
- '@codemirror/view': ^6.0.0
- '@lezer/common': ^1.0.0
-
- '@codemirror/commands@6.7.0':
- resolution: {integrity: sha512-+cduIZ2KbesDhbykV02K25A5xIVrquSPz4UxxYBemRlAT2aW8dhwUgLDwej7q/RJUHKk4nALYcR1puecDvbdqw==}
-
- '@codemirror/lang-json@6.0.1':
- resolution: {integrity: sha512-+T1flHdgpqDDlJZ2Lkil/rLiRy684WMLc74xUnjJH48GQdfJo/pudlTRreZmKwzP8/tGdKf83wlbAdOCzlJOGQ==}
-
- '@codemirror/language@6.10.3':
- resolution: {integrity: sha512-kDqEU5sCP55Oabl6E7m5N+vZRoc0iWqgDVhEKifcHzPzjqCegcO4amfrYVL9PmPZpl4G0yjkpTpUO/Ui8CzO8A==}
-
- '@codemirror/lint@6.8.2':
- resolution: {integrity: sha512-PDFG5DjHxSEjOXk9TQYYVjZDqlZTFaDBfhQixHnQOEVDDNHUbEh/hstAjcQJaA6FQdZTD1hquXTK0rVBLADR1g==}
-
- '@codemirror/search@6.5.6':
- resolution: {integrity: sha512-rpMgcsh7o0GuCDUXKPvww+muLA1pDJaFrpq/CCHtpQJYz8xopu4D1hPcKRoDD0YlF8gZaqTNIRa4VRBWyhyy7Q==}
-
- '@codemirror/state@6.4.1':
- resolution: {integrity: sha512-QkEyUiLhsJoZkbumGZlswmAhA7CBU02Wrz7zvH4SrcifbsqwlXShVXg65f3v/ts57W3dqyamEriMhij1Z3Zz4A==}
-
- '@codemirror/view@6.34.1':
- resolution: {integrity: sha512-t1zK/l9UiRqwUNPm+pdIT0qzJlzuVckbTEMVNFhfWkGiBQClstzg+78vedCvLSX0xJEZ6lwZbPpnljL7L6iwMQ==}
-
- '@ctrl/tinycolor@3.4.1':
- resolution: {integrity: sha512-ej5oVy6lykXsvieQtqZxCOaLT+xD4+QNarq78cIYISHmZXshCvROLudpQN3lfL8G0NL7plMSSK+zlyvCaIJ4Iw==}
- engines: {node: '>=10'}
-
- '@element-plus/icons-vue@2.3.1':
- resolution: {integrity: sha512-XxVUZv48RZAd87ucGS48jPf6pKu0yV5UCg9f4FFwtrYxXOwWuVJo6wOvSLKEoMQKjv8GsX/mhP6UsC1lRwbUWg==}
- version: 2.3.1
- peerDependencies:
- vue: ^3.2.0
-
- '@esbuild/android-arm@0.15.16':
- resolution: {integrity: sha512-nyB6CH++2mSgx3GbnrJsZSxzne5K0HMyNIWafDHqYy7IwxFc4fd/CgHVZXr8Eh+Q3KbIAcAe3vGyqIPhGblvMQ==}
- engines: {node: '>=12'}
- cpu: [arm]
- os: [android]
-
- '@esbuild/linux-loong64@0.15.16':
- resolution: {integrity: sha512-SDLfP1uoB0HZ14CdVYgagllgrG7Mdxhkt4jDJOKl/MldKrkQ6vDJMZKl2+5XsEY/Lzz37fjgLQoJBGuAw/x8kQ==}
- engines: {node: '>=12'}
- cpu: [loong64]
- os: [linux]
-
- '@floating-ui/core@1.6.8':
- resolution: {integrity: sha512-7XJ9cPU+yI2QeLS+FCSlqNFZJq8arvswefkZrYI1yQBbftw6FyrZOxYSh+9S7z7TpeWlRt9zJ5IhM1WIL334jA==}
-
- '@floating-ui/dom@1.6.11':
- resolution: {integrity: sha512-qkMCxSR24v2vGkhYDo/UzxfJN3D4syqSjyuTFz6C7XcpU1pASPRieNI0Kj5VP3/503mOfYiGY891ugBX1GlABQ==}
-
- '@floating-ui/utils@0.2.8':
- resolution: {integrity: sha512-kym7SodPp8/wloecOpcmSnWJsK7M0E5Wg8UcFA+uO4B9s5d0ywXOEro/8HM9x0rW+TljRzul/14UYz3TleT3ig==}
-
- '@fortawesome/fontawesome-common-types@6.6.0':
- resolution: {integrity: sha512-xyX0X9mc0kyz9plIyryrRbl7ngsA9jz77mCZJsUkLl+ZKs0KWObgaEBoSgQiYWAsSmjz/yjl0F++Got0Mdp4Rw==}
- engines: {node: '>=6'}
-
- '@fortawesome/fontawesome-svg-core@6.6.0':
- resolution: {integrity: sha512-KHwPkCk6oRT4HADE7smhfsKudt9N/9lm6EJ5BVg0tD1yPA5hht837fB87F8pn15D8JfTqQOjhKTktwmLMiD7Kg==}
- engines: {node: '>=6'}
-
- '@fortawesome/free-brands-svg-icons@6.6.0':
- resolution: {integrity: sha512-1MPD8lMNW/earme4OQi1IFHtmHUwAKgghXlNwWi9GO7QkTfD+IIaYpIai4m2YJEzqfEji3jFHX1DZI5pbY/biQ==}
- engines: {node: '>=6'}
-
- '@fortawesome/free-regular-svg-icons@6.6.0':
- resolution: {integrity: sha512-Yv9hDzL4aI73BEwSEh20clrY8q/uLxawaQ98lekBx6t9dQKDHcDzzV1p2YtBGTtolYtNqcWdniOnhzB+JPnQEQ==}
- engines: {node: '>=6'}
-
- '@fortawesome/free-solid-svg-icons@6.6.0':
- resolution: {integrity: sha512-IYv/2skhEDFc2WGUcqvFJkeK39Q+HyPf5GHUrT/l2pKbtgEIv1al1TKd6qStR5OIwQdN1GZP54ci3y4mroJWjA==}
- engines: {node: '>=6'}
-
- '@fortawesome/vue-fontawesome@3.0.8':
- resolution: {integrity: sha512-yyHHAj4G8pQIDfaIsMvQpwKMboIZtcHTUvPqXjOHyldh1O1vZfH4W03VDPv5RvI9P6DLTzJQlmVgj9wCf7c2Fw==}
- version: 3.0.8
- peerDependencies:
- '@fortawesome/fontawesome-svg-core': ~1 || ~6
- vue: '>= 3.0.0 < 4'
-
- '@intlify/core-base@9.13.1':
- resolution: {integrity: sha512-+bcQRkJO9pcX8d0gel9ZNfrzU22sZFSA0WVhfXrf5jdJOS24a+Bp8pozuS9sBI9Hk/tGz83pgKfmqcn/Ci7/8w==}
- engines: {node: '>= 16'}
-
- '@intlify/message-compiler@9.13.1':
- resolution: {integrity: sha512-SKsVa4ajYGBVm7sHMXd5qX70O2XXjm55zdZB3VeMFCvQyvLew/dLvq3MqnaIsTMF1VkkOb9Ttr6tHcMlyPDL9w==}
- engines: {node: '>= 16'}
-
- '@intlify/shared@9.13.1':
- resolution: {integrity: sha512-u3b6BKGhE6j/JeRU6C/RL2FgyJfy6LakbtfeVF8fJXURpZZTzfh3e05J0bu0XPw447Q6/WUp3C4ajv4TMS4YsQ==}
- engines: {node: '>= 16'}
-
- '@jridgewell/gen-mapping@0.3.5':
- resolution: {integrity: sha512-IzL8ZoEDIBRWEzlCcRhOaCupYyN5gdIK+Q6fbFdPDg6HqX6jpkItn7DFIpW9LQzXG6Df9sA7+OKnq0qlz/GaQg==}
- engines: {node: '>=6.0.0'}
-
- '@jridgewell/resolve-uri@3.1.2':
- resolution: {integrity: sha512-bRISgCIjP20/tbWSPWMEi54QVPRZExkuD9lJL+UIxUKtwVJA8wW1Trb1jMs1RFXo1CBTNZ/5hpC9QvmKWdopKw==}
- engines: {node: '>=6.0.0'}
-
- '@jridgewell/set-array@1.2.1':
- resolution: {integrity: sha512-R8gLRTZeyp03ymzP/6Lil/28tGeGEzhx1q2k703KGWRAI1VdvPIXdG70VJc2pAMw3NA6JKL5hhFu1sJX0Mnn/A==}
- engines: {node: '>=6.0.0'}
-
- '@jridgewell/sourcemap-codec@1.4.15':
- resolution: {integrity: sha512-eF2rxCRulEKXHTRiDrDy6erMYWqNw4LPdQ8UQA4huuxaQsVeRPFl2oM8oDGxMFhJUWZf9McpLtJasDDZb/Bpeg==}
-
- '@jridgewell/sourcemap-codec@1.5.0':
- resolution: {integrity: sha512-gv3ZRaISU3fjPAgNsriBRqGWQL6quFx04YMPW/zD8XMLsU32mhCCbfbO6KZFLjvYpCZ8zyDEgqsgf+PwPaM7GQ==}
-
- '@jridgewell/trace-mapping@0.3.25':
- resolution: {integrity: sha512-vNk6aEwybGtawWmy/PzwnGDOjCkLWSD2wqvjGGAgOAwCGWySYXfYoxt00IJkTF+8Lb57DwOb3Aa0o9CApepiYQ==}
-
- '@jsep-plugin/assignment@1.2.1':
- resolution: {integrity: sha512-gaHqbubTi29aZpVbBlECRpmdia+L5/lh2BwtIJTmtxdbecEyyX/ejAOg7eQDGNvGOUmPY7Z2Yxdy9ioyH/VJeA==}
- engines: {node: '>= 10.16.0'}
- peerDependencies:
- jsep: ^0.4.0||^1.0.0
-
- '@jsep-plugin/regex@1.0.3':
- resolution: {integrity: sha512-XfZgry4DwEZvSFtS/6Y+R48D7qJYJK6R9/yJFyUFHCIUMEEHuJ4X95TDgJp5QkmzfLYvapMPzskV5HpIDrREug==}
- engines: {node: '>= 10.16.0'}
- peerDependencies:
- jsep: ^0.4.0||^1.0.0
-
- '@jsonquerylang/jsonquery@3.1.1':
- resolution: {integrity: sha512-P6Qo5egd3W8TBpqQsqaZtZ9lPO7oXBM21QdkYamCAYZHv9VCPXiI8NeIuSoXdoe5zKVZPUWmqaI14uacJLmcNw==}
- hasBin: true
-
- '@kurkle/color@0.3.2':
- resolution: {integrity: sha512-fuscdXJ9G1qb7W8VdHi+IwRqij3lBkosAm4ydQtEmbY58OzHXqQhvlxqEkoz0yssNVn38bcpRWgA9PP+OGoisw==}
-
- '@lexical/clipboard@0.16.1':
- resolution: {integrity: sha512-0dWs/SwKS5KPpuf6fUVVt9vSCl6HAqcDGhSITw/okv0rrIlXTUT6WhVsMJtXfFxTyVvwMeOecJHvQH3i/jRQtA==}
-
- '@lexical/code@0.16.1':
- resolution: {integrity: sha512-pOC28rRZ2XkmI2nIJm50DbKaCJtk5D0o7r6nORYp4i0z+lxt5Sf2m82DL9ksUHJRqKy87pwJDpoWvJ2SAI0ohw==}
-
- '@lexical/history@0.16.1':
- resolution: {integrity: sha512-WQhScx0TJeKSQAnEkRpIaWdUXqirrNrom2MxbBUc/32zEUMm9FzV7nRGknvUabEFUo7vZq6xTZpOExQJqHInQA==}
-
- '@lexical/html@0.16.1':
- resolution: {integrity: sha512-vbtAdCvQ3PaAqa5mFmtmrvbiAvjCu1iXBAJ0bsHqFXCF2Sba5LwHVe8dUAOTpfEZEMbiHfjul6b5fj4vNPGF2A==}
-
- '@lexical/link@0.16.1':
- resolution: {integrity: sha512-zG36gEnEqbIe6tK/MhXi7wn/XMY/zdivnPcOY5WyC3derkEezeLSSIFsC1u5UNeK5pbpNMSy4LDpLhi1Ww4Y5w==}
-
- '@lexical/list@0.16.1':
- resolution: {integrity: sha512-i9YhLAh5N6YO9dP+R1SIL9WEdCKeTiQQYVUzj84vDvX5DIBxMPUjTmMn3LXu9T+QO3h1s2L/vJusZASrl45eAw==}
-
- '@lexical/markdown@0.16.1':
- resolution: {integrity: sha512-0sBLttMvfQO/hVaIqpHdvDowpgV2CoRuWo2CNwvRLZPPWvPVjL4Nkb73wmi8zAZsAOTbX2aw+g4m/+k5oJqNig==}
-
- '@lexical/rich-text@0.16.1':
- resolution: {integrity: sha512-4uEVXJur7tdSbqbmsToCW4YVm0AMh4y9LK077Yq2O9hSuA5dqpI8UbTDnxZN2D7RfahNvwlqp8eZKFB1yeiJGQ==}
-
- '@lexical/selection@0.16.1':
- resolution: {integrity: sha512-+nK3RvXtyQvQDq7AZ46JpphmM33pwuulwiRfeXR5T9iFQTtgWOEjsAi/KKX7vGm70BxACfiSxy5QCOgBWFwVJg==}
-
- '@lexical/table@0.16.1':
- resolution: {integrity: sha512-GWb0/MM1sVXpi1p2HWWOBldZXASMQ4c6WRNYnRmq7J/aB5N66HqQgJGKp3m66Kz4k1JjhmZfPs7F018qIBhnFQ==}
-
- '@lexical/text@0.16.1':
- resolution: {integrity: sha512-Os/nKQegORTrKKN6vL3/FMVszyzyqaotlisPynvTaHTUC+yY4uyjM2hlF93i5a2ixxyiPLF9bDroxUP96TMPXg==}
-
- '@lexical/utils@0.16.1':
- resolution: {integrity: sha512-BVyJxDQi/rIxFTDjf2zE7rMDKSuEaeJ4dybHRa/hRERt85gavGByQawSLeQlTjLaYLVsy+x7wCcqh2fNhlLf0g==}
-
- '@lezer/common@1.2.3':
- resolution: {integrity: sha512-w7ojc8ejBqr2REPsWxJjrMFsA/ysDCFICn8zEOR9mrqzOu2amhITYuLD8ag6XZf0CFXDrhKqw7+tW8cX66NaDA==}
-
- '@lezer/highlight@1.2.1':
- resolution: {integrity: sha512-Z5duk4RN/3zuVO7Jq0pGLJ3qynpxUVsh7IbUbGj88+uV2ApSAn6kWg2au3iJb+0Zi7kKtqffIESgNcRXWZWmSA==}
-
- '@lezer/json@1.0.2':
- resolution: {integrity: sha512-xHT2P4S5eeCYECyKNPhr4cbEL9tc8w83SPwRC373o9uEdrvGKTZoJVAGxpOsZckMlEh9W23Pc72ew918RWQOBQ==}
-
- '@lezer/lr@1.4.2':
- resolution: {integrity: sha512-pu0K1jCIdnQ12aWNaAVU5bzi7Bd1w54J3ECgANPmYLtQKP0HBj2cE/5coBD66MT10xbtIuUr7tg0Shbsvk0mDA==}
-
- '@mapbox/node-pre-gyp@1.0.10':
- resolution: {integrity: sha512-4ySo4CjzStuprMwk35H5pPbkymjv1SF3jGLj6rAHp/xT/RF7TL7bd9CTm1xDY49K2qF7jmR/g7k+SkLETP6opA==}
- hasBin: true
-
- '@nodelib/fs.scandir@https://registry.npmmirror.com/@nodelib/fs.scandir/-/fs.scandir-2.1.5.tgz':
- resolution: {integrity: sha512-vq24Bq3ym5HEQm2NKCr3yXDwjc7vTsEThRDnkp2DK9p1uqLR+DHurm/NOTo0KG7HYHU7eppKZj3MyqYuMBf62g==, tarball: https://registry.npmmirror.com/@nodelib/fs.scandir/-/fs.scandir-2.1.5.tgz}
- version: 2.1.5
- engines: {node: '>= 8'}
-
- '@nodelib/fs.stat@https://registry.npmmirror.com/@nodelib/fs.stat/-/fs.stat-2.0.5.tgz':
- resolution: {integrity: sha512-RkhPPp2zrqDAQA/2jNhnztcPAlv64XdhIp7a7454A5ovI7Bukxgt7MX7udwAu3zg1DcpPU0rz3VV1SeaqvY4+A==, tarball: https://registry.npmmirror.com/@nodelib/fs.stat/-/fs.stat-2.0.5.tgz}
- version: 2.0.5
- engines: {node: '>= 8'}
-
- '@nodelib/fs.walk@https://registry.npmmirror.com/@nodelib/fs.walk/-/fs.walk-1.2.8.tgz':
- resolution: {integrity: sha512-oGB+UxlgWcgQkgwo8GcEGwemoTFt3FIO9ababBmaGwXIoBKZ+GTy0pP185beGg7Llih/NSHSV2XAs1lnznocSg==, tarball: https://registry.npmmirror.com/@nodelib/fs.walk/-/fs.walk-1.2.8.tgz}
- version: 1.2.8
- engines: {node: '>= 8'}
-
- '@parcel/watcher-android-arm64@2.4.1':
- resolution: {integrity: sha512-LOi/WTbbh3aTn2RYddrO8pnapixAziFl6SMxHM69r3tvdSm94JtCenaKgk1GRg5FJ5wpMCpHeW+7yqPlvZv7kg==}
- engines: {node: '>= 10.0.0'}
- cpu: [arm64]
- os: [android]
-
- '@parcel/watcher-darwin-arm64@2.4.1':
- resolution: {integrity: sha512-ln41eihm5YXIY043vBrrHfn94SIBlqOWmoROhsMVTSXGh0QahKGy77tfEywQ7v3NywyxBBkGIfrWRHm0hsKtzA==}
- engines: {node: '>= 10.0.0'}
- cpu: [arm64]
- os: [darwin]
-
- '@parcel/watcher-darwin-x64@2.4.1':
- resolution: {integrity: sha512-yrw81BRLjjtHyDu7J61oPuSoeYWR3lDElcPGJyOvIXmor6DEo7/G2u1o7I38cwlcoBHQFULqF6nesIX3tsEXMg==}
- engines: {node: '>= 10.0.0'}
- cpu: [x64]
- os: [darwin]
-
- '@parcel/watcher-freebsd-x64@2.4.1':
- resolution: {integrity: sha512-TJa3Pex/gX3CWIx/Co8k+ykNdDCLx+TuZj3f3h7eOjgpdKM+Mnix37RYsYU4LHhiYJz3DK5nFCCra81p6g050w==}
- engines: {node: '>= 10.0.0'}
- cpu: [x64]
- os: [freebsd]
-
- '@parcel/watcher-linux-arm-glibc@2.4.1':
- resolution: {integrity: sha512-4rVYDlsMEYfa537BRXxJ5UF4ddNwnr2/1O4MHM5PjI9cvV2qymvhwZSFgXqbS8YoTk5i/JR0L0JDs69BUn45YA==}
- engines: {node: '>= 10.0.0'}
- cpu: [arm]
- os: [linux]
-
- '@parcel/watcher-linux-arm64-glibc@2.4.1':
- resolution: {integrity: sha512-BJ7mH985OADVLpbrzCLgrJ3TOpiZggE9FMblfO65PlOCdG++xJpKUJ0Aol74ZUIYfb8WsRlUdgrZxKkz3zXWYA==}
- engines: {node: '>= 10.0.0'}
- cpu: [arm64]
- os: [linux]
-
- '@parcel/watcher-linux-arm64-musl@2.4.1':
- resolution: {integrity: sha512-p4Xb7JGq3MLgAfYhslU2SjoV9G0kI0Xry0kuxeG/41UfpjHGOhv7UoUDAz/jb1u2elbhazy4rRBL8PegPJFBhA==}
- engines: {node: '>= 10.0.0'}
- cpu: [arm64]
- os: [linux]
-
- '@parcel/watcher-linux-x64-glibc@2.4.1':
- resolution: {integrity: sha512-s9O3fByZ/2pyYDPoLM6zt92yu6P4E39a03zvO0qCHOTjxmt3GHRMLuRZEWhWLASTMSrrnVNWdVI/+pUElJBBBg==}
- engines: {node: '>= 10.0.0'}
- cpu: [x64]
- os: [linux]
-
- '@parcel/watcher-linux-x64-musl@2.4.1':
- resolution: {integrity: sha512-L2nZTYR1myLNST0O632g0Dx9LyMNHrn6TOt76sYxWLdff3cB22/GZX2UPtJnaqQPdCRoszoY5rcOj4oMTtp5fQ==}
- engines: {node: '>= 10.0.0'}
- cpu: [x64]
- os: [linux]
-
- '@parcel/watcher-win32-arm64@2.4.1':
- resolution: {integrity: sha512-Uq2BPp5GWhrq/lcuItCHoqxjULU1QYEcyjSO5jqqOK8RNFDBQnenMMx4gAl3v8GiWa59E9+uDM7yZ6LxwUIfRg==}
- engines: {node: '>= 10.0.0'}
- cpu: [arm64]
- os: [win32]
-
- '@parcel/watcher-win32-ia32@2.4.1':
- resolution: {integrity: sha512-maNRit5QQV2kgHFSYwftmPBxiuK5u4DXjbXx7q6eKjq5dsLXZ4FJiVvlcw35QXzk0KrUecJmuVFbj4uV9oYrcw==}
- engines: {node: '>= 10.0.0'}
- cpu: [ia32]
- os: [win32]
-
- '@parcel/watcher-win32-x64@2.4.1':
- resolution: {integrity: sha512-+DvS92F9ezicfswqrvIRM2njcYJbd5mb9CUgtrHCHmvn7pPPa+nMDRu1o1bYYz/l5IB2NVGNJWiH7h1E58IF2A==}
- engines: {node: '>= 10.0.0'}
- cpu: [x64]
- os: [win32]
-
- '@parcel/watcher@2.4.1':
- resolution: {integrity: sha512-HNjmfLQEVRZmHRET336f20H/8kOozUGwk7yajvsonjNxbj2wBTK1WsQuHkD5yYh9RxFGL2EyDHryOihOwUoKDA==}
- engines: {node: '>= 10.0.0'}
-
- '@polka/url@0.5.0':
- resolution: {integrity: sha512-oZLYFEAzUKyi3SKnXvj32ZCEGH6RDnao7COuCVhDydMS9NrCSVXhM79VaKyP5+Zc33m0QXEd2DN3UkU7OsHcfw==}
-
- '@polka/url@1.0.0-next.21':
- resolution: {integrity: sha512-a5Sab1C4/icpTZVzZc5Ghpz88yQtGOyNqYXcZgOssB2uuAr+wF/MvN6bgtW32q7HHrvBki+BsZ0OuNv6EV3K9g==}
-
- '@popperjs/core@2.11.8':
- resolution: {integrity: sha512-P1st0aksCrn9sGZhp8GMYwBnQsbvAWsZAX44oXNNvLHGqAOcoVxmjZiohstwQ7SqKnbR47akdNi+uleWD8+g6A==}
-
- '@replit/codemirror-indentation-markers@6.5.3':
- resolution: {integrity: sha512-hL5Sfvw3C1vgg7GolLe/uxX5T3tmgOA3ZzqlMv47zjU1ON51pzNWiVbS22oh6crYhtVhv8b3gdXwoYp++2ilHw==}
- peerDependencies:
- '@codemirror/language': ^6.0.0
- '@codemirror/state': ^6.0.0
- '@codemirror/view': ^6.0.0
-
- '@rollup/pluginutils@5.0.2':
- resolution: {integrity: sha512-pTd9rIsP92h+B6wWwFbW8RkZv4hiR/xKsqre4SIuAOaOEQRxi0lqLke9k2/7WegC85GgUs9pjmOjCUi3In4vwA==}
- engines: {node: '>=14.0.0'}
- peerDependencies:
- rollup: ^1.20.0||^2.0.0||^3.0.0
- peerDependenciesMeta:
- rollup:
- optional: true
-
- '@sphinxxxx/color-conversion@2.2.2':
- resolution: {integrity: sha512-XExJS3cLqgrmNBIP3bBw6+1oQ1ksGjFh0+oClDKFYpCCqx/hlqwWO5KO/S63fzUo67SxI9dMrF0y5T/Ey7h8Zw==}
-
- '@sxzz/popperjs-es@2.11.7':
- resolution: {integrity: sha512-Ccy0NlLkzr0Ex2FKvh2X+OyERHXJ88XJ1MXtsI9y9fGexlaXaVTPzBCRBwIxFkORuOb+uBqeu+RqnpgYTEZRUQ==}
-
- '@trysound/sax@0.2.0':
- resolution: {integrity: sha512-L7z9BgrNEcYyUYtF+HaEfiS5ebkh9jXqbszz7pC0hRBPaatV0XjSD3+eHrpqFemQfgwiFF0QPIarnIihIDn7OA==}
- engines: {node: '>=10.13.0'}
-
- '@tweenjs/tween.js@23.1.3':
- resolution: {integrity: sha512-vJmvvwFxYuGnF2axRtPYocag6Clbb5YS7kLL+SO/TeVFzHqDIWrNKYtcsPMibjDx9O+bu+psAy9NKfWklassUA==}
-
- '@types/estree@1.0.0':
- resolution: {integrity: sha512-WulqXMDUTYAXCjZnk6JtIHPigp55cVtDgDrO2gHRwhyJto21+1zbVCtOYB2L1F9w4qCQ0rOGWBnBe0FNTiEJIQ==}
-
- '@types/estree@1.0.6':
- resolution: {integrity: sha512-AYnb1nQyY49te+VRAVgmzfcgjYS91mY5P0TKUDCLEM+gNnA+3T6rWITXRLYCpahpqSQbN5cE+gHpnPyXjHWxcw==}
-
- '@types/getos@3.0.4':
- resolution: {integrity: sha512-OsNVsuGEBd5QoeXTgjPmA89a465hjQithLMImsiJlIi0aYP8axKWtWrsmEMfm50w6EL2oRzCz+cLkZmp9/LfZw==}
-
- '@types/humanize-duration@3.27.4':
- resolution: {integrity: sha512-yaf7kan2Sq0goxpbcwTQ+8E9RP6HutFBPv74T/IA/ojcHKhuKVlk2YFYyHhWZeLvZPzzLE3aatuQB4h0iqyyUA==}
-
- '@types/javascript-time-ago@2.0.8':
- resolution: {integrity: sha512-X77q3xUzWVn0qohgurKE1G5NiXZjee8VbLqaukW/HXVkz7bdCFJgOPQ3JVB4IkrDhMS4CviFTmpZuNVg0i2QFA==}
-
- '@types/lodash-es@4.17.12':
- resolution: {integrity: sha512-0NgftHUcV4v34VhXm8QBSftKVXtbkBG3ViCjs6+eJ5a6y6Mi/jiFGPc1sC7QK+9BFhWrURE3EOggmWaSxL9OzQ==}
-
- '@types/lodash@4.17.10':
- resolution: {integrity: sha512-YpS0zzoduEhuOWjAotS6A5AVCva7X4lVlYLF0FYHAY9sdraBfnatttHItlWeZdGhuEkf+OzMNg2ZYAx8t+52uQ==}
-
- '@types/md5@2.3.5':
- resolution: {integrity: sha512-/i42wjYNgE6wf0j2bcTX6kuowmdL/6PE4IVitMpm2eYKBUuYCprdcWVK+xEF0gcV6ufMCRhtxmReGfc6hIK7Jw==}
-
- '@types/node@https://registry.npmmirror.com/@types/node/-/node-18.11.10.tgz':
- resolution: {integrity: sha512-juG3RWMBOqcOuXC643OAdSA525V44cVgGV6dUDuiFtss+8Fk5x1hI93Rsld43VeJVIeqlP9I7Fn9/qaVqoEAuQ==, tarball: https://registry.npmmirror.com/@types/node/-/node-18.11.10.tgz}
- version: 18.11.10
-
- '@types/pinyin@2.10.2':
- resolution: {integrity: sha512-jLzlRkaLRLg+lgYPjOuP3HX2cozUkhXls5GTXopsKuKJ9lDGlIAb88OoIztH6TbNUsoJnl/7e/kjaumA5IKKJg==}
-
- '@types/showdown@2.0.6':
- resolution: {integrity: sha512-pTvD/0CIeqe4x23+YJWlX2gArHa8G0J0Oh6GKaVXV7TAeickpkkZiNOgFcFcmLQ5lB/K0qBJL1FtRYltBfbGCQ==}
-
- '@types/stats.js@0.17.3':
- resolution: {integrity: sha512-pXNfAD3KHOdif9EQXZ9deK82HVNaXP5ZIF5RP2QG6OQFNTaY2YIetfrE9t528vEreGQvEPRDDc8muaoYeK0SxQ==}
-
- '@types/three@0.166.0':
- resolution: {integrity: sha512-FHMnpcdhdbdOOIYbfkTkUVpYMW53odxbTRwd0/xJpYnTzEsjnVnondGAvHZb4z06UW0vo6WPVuvH0/9qrxKx7g==}
-
- '@types/url-join@4.0.3':
- resolution: {integrity: sha512-3l1qMm3wqO0iyC5gkADzT95UVW7C/XXcdvUcShOideKF0ddgVRErEQQJXBd2kvQm+aSgqhBGHGB38TgMeT57Ww==}
-
- '@types/web-bluetooth@0.0.16':
- resolution: {integrity: sha512-oh8q2Zc32S6gd/j50GowEjKLoOVOwHP/bWVjKJInBwQqdOYMdPrf1oVlelTlyfFK3CKxL1uahMDAr+vy8T7yMQ==}
-
- '@types/webxr@0.5.20':
- resolution: {integrity: sha512-JGpU6qiIJQKUuVSKx1GtQnHJGxRjtfGIhzO2ilq43VZZS//f1h1Sgexbdk+Lq+7569a6EYhOWrUpIruR/1Enmg==}
-
- '@vitejs/plugin-vue@https://registry.npmmirror.com/@vitejs/plugin-vue/-/plugin-vue-3.2.0.tgz':
- resolution: {integrity: sha512-E0tnaL4fr+qkdCNxJ+Xd0yM31UwMkQje76fsDVBBUCoGOUPexu2VDUYHL8P4CwV+zMvWw6nlRw19OnRKmYAJpw==, tarball: https://registry.npmmirror.com/@vitejs/plugin-vue/-/plugin-vue-3.2.0.tgz}
- version: 3.2.0
- engines: {node: ^14.18.0 || >=16.0.0}
- peerDependencies:
- vite: ^3.0.0
- vue: ^3.2.25
-
- '@volar/language-core@https://registry.npmmirror.com/@volar/language-core/-/language-core-1.0.11.tgz':
- resolution: {integrity: sha512-YwUYKxIyDc+Fq3kQ6BGGfkrKCG5JzE2Yr6vMxrxEXW2rg/gsq3JgMk/4sI8ybRsaTirhCB4V8+AIVYsvcRxgig==, tarball: https://registry.npmmirror.com/@volar/language-core/-/language-core-1.0.11.tgz}
- version: 1.0.11
-
- '@volar/source-map@https://registry.npmmirror.com/@volar/source-map/-/source-map-1.0.11.tgz':
- resolution: {integrity: sha512-tkuV9MD+OuiZfHA0qZXrPdW6F7TvnpnuTan6Qe7UGUs9+sflezlMJdjaYdGgQObfP+06pcT1E3xdkOoi08ZyyQ==, tarball: https://registry.npmmirror.com/@volar/source-map/-/source-map-1.0.11.tgz}
- version: 1.0.11
-
- '@volar/typescript@https://registry.npmmirror.com/@volar/typescript/-/typescript-1.0.11.tgz':
- resolution: {integrity: sha512-mq7wDDAs0Eb43jev2FxbowuiwWqvL3kb+tar1we8VQbdabpyQ5dmbWPwo/IglevMmW3SKo1Et+6rqAeZpXNnPQ==, tarball: https://registry.npmmirror.com/@volar/typescript/-/typescript-1.0.11.tgz}
- version: 1.0.11
-
- '@volar/vue-language-core@https://registry.npmmirror.com/@volar/vue-language-core/-/vue-language-core-1.0.11.tgz':
- resolution: {integrity: sha512-A3ODs0/ua7BcpSSnE7KtO8bzWsYsbOJRyW2Q/2uktxlfHj8srln3JdgK/mNlIgfnWtACbE5K+EfMJOgJKv864A==, tarball: https://registry.npmmirror.com/@volar/vue-language-core/-/vue-language-core-1.0.11.tgz}
- version: 1.0.11
-
- '@volar/vue-typescript@https://registry.npmmirror.com/@volar/vue-typescript/-/vue-typescript-1.0.11.tgz':
- resolution: {integrity: sha512-jlnFPvBcTyPiAbGlgjhKK7fp3Q+Z7Z5eU1NTbTSS0lQC8Gog3sh2UxLAFG5Voe1gHIxasoOEPXzMR0CWF4bKbA==, tarball: https://registry.npmmirror.com/@volar/vue-typescript/-/vue-typescript-1.0.11.tgz}
- version: 1.0.11
-
- '@vue/compiler-core@3.5.12':
- resolution: {integrity: sha512-ISyBTRMmMYagUxhcpyEH0hpXRd/KqDU4ymofPgl2XAkY9ZhQ+h0ovEZJIiPop13UmR/54oA2cgMDjgroRelaEw==}
-
- '@vue/compiler-core@https://registry.npmmirror.com/@vue/compiler-core/-/compiler-core-3.2.45.tgz':
- resolution: {integrity: sha512-rcMj7H+PYe5wBV3iYeUgbCglC+pbpN8hBLTJvRiK2eKQiWqu+fG9F+8sW99JdL4LQi7Re178UOxn09puSXvn4A==, tarball: https://registry.npmmirror.com/@vue/compiler-core/-/compiler-core-3.2.45.tgz}
- version: 3.2.45
-
- '@vue/compiler-dom@3.5.12':
- resolution: {integrity: sha512-9G6PbJ03uwxLHKQ3P42cMTi85lDRvGLB2rSGOiQqtXELat6uI4n8cNz9yjfVHRPIu+MsK6TE418Giruvgptckg==}
-
- '@vue/compiler-dom@https://registry.npmmirror.com/@vue/compiler-dom/-/compiler-dom-3.2.45.tgz':
- resolution: {integrity: sha512-tyYeUEuKqqZO137WrZkpwfPCdiiIeXYCcJ8L4gWz9vqaxzIQRccTSwSWZ/Axx5YR2z+LvpUbmPNXxuBU45lyRw==, tarball: https://registry.npmmirror.com/@vue/compiler-dom/-/compiler-dom-3.2.45.tgz}
- version: 3.2.45
-
- '@vue/compiler-sfc@3.5.12':
- resolution: {integrity: sha512-2k973OGo2JuAa5+ZlekuQJtitI5CgLMOwgl94BzMCsKZCX/xiqzJYzapl4opFogKHqwJk34vfsaKpfEhd1k5nw==}
-
- '@vue/compiler-sfc@https://registry.npmmirror.com/@vue/compiler-sfc/-/compiler-sfc-3.2.45.tgz':
- resolution: {integrity: sha512-1jXDuWah1ggsnSAOGsec8cFjT/K6TMZ0sPL3o3d84Ft2AYZi2jWJgRMjw4iaK0rBfA89L5gw427H4n1RZQBu6Q==, tarball: https://registry.npmmirror.com/@vue/compiler-sfc/-/compiler-sfc-3.2.45.tgz}
- version: 3.2.45
-
- '@vue/compiler-ssr@3.5.12':
- resolution: {integrity: sha512-eLwc7v6bfGBSM7wZOGPmRavSWzNFF6+PdRhE+VFJhNCgHiF8AM7ccoqcv5kBXA2eWUfigD7byekvf/JsOfKvPA==}
-
- '@vue/compiler-ssr@https://registry.npmmirror.com/@vue/compiler-ssr/-/compiler-ssr-3.2.45.tgz':
- resolution: {integrity: sha512-6BRaggEGqhWht3lt24CrIbQSRD5O07MTmd+LjAn5fJj568+R9eUD2F7wMQJjX859seSlrYog7sUtrZSd7feqrQ==, tarball: https://registry.npmmirror.com/@vue/compiler-ssr/-/compiler-ssr-3.2.45.tgz}
- version: 3.2.45
-
- '@vue/devtools-api@6.4.5':
- resolution: {integrity: sha512-JD5fcdIuFxU4fQyXUu3w2KpAJHzTVdN+p4iOX2lMWSHMOoQdMAcpFLZzm9Z/2nmsoZ1a96QEhZ26e50xLBsgOQ==}
-
- '@vue/devtools-api@6.6.4':
- resolution: {integrity: sha512-sGhTPMuXqZ1rVOk32RylztWkfXTRhuS7vgAKv0zjqk8gbsHkJ7xfFf+jbySxt7tWObEJwyKaHMikV/WGDiQm8g==}
-
- '@vue/reactivity-transform@https://registry.npmmirror.com/@vue/reactivity-transform/-/reactivity-transform-3.2.45.tgz':
- resolution: {integrity: sha512-BHVmzYAvM7vcU5WmuYqXpwaBHjsS8T63jlKGWVtHxAHIoMIlmaMyurUSEs1Zcg46M4AYT5MtB1U274/2aNzjJQ==, tarball: https://registry.npmmirror.com/@vue/reactivity-transform/-/reactivity-transform-3.2.45.tgz}
- version: 3.2.45
-
- '@vue/reactivity@3.5.12':
- resolution: {integrity: sha512-UzaN3Da7xnJXdz4Okb/BGbAaomRHc3RdoWqTzlvd9+WBR5m3J39J1fGcHes7U3za0ruYn/iYy/a1euhMEHvTAg==}
-
- '@vue/reactivity@https://registry.npmmirror.com/@vue/reactivity/-/reactivity-3.2.45.tgz':
- resolution: {integrity: sha512-PRvhCcQcyEVohW0P8iQ7HDcIOXRjZfAsOds3N99X/Dzewy8TVhTCT4uXpAHfoKjVTJRA0O0K+6QNkDIZAxNi3A==, tarball: https://registry.npmmirror.com/@vue/reactivity/-/reactivity-3.2.45.tgz}
- version: 3.2.45
-
- '@vue/runtime-core@3.5.12':
- resolution: {integrity: sha512-hrMUYV6tpocr3TL3Ad8DqxOdpDe4zuQY4HPY3X/VRh+L2myQO8MFXPAMarIOSGNu0bFAjh1yBkMPXZBqCk62Uw==}
-
- '@vue/runtime-dom@3.5.12':
- resolution: {integrity: sha512-q8VFxR9A2MRfBr6/55Q3umyoN7ya836FzRXajPB6/Vvuv0zOPL+qltd9rIMzG/DbRLAIlREmnLsplEF/kotXKA==}
-
- '@vue/server-renderer@3.5.12':
- resolution: {integrity: sha512-I3QoeDDeEPZm8yR28JtY+rk880Oqmj43hreIBVTicisFTx/Dl7JpG72g/X7YF8hnQD3IFhkky5i2bPonwrTVPg==}
- version: 3.5.12
- peerDependencies:
- vue: 3.5.12
-
- '@vue/shared@3.5.12':
- resolution: {integrity: sha512-L2RPSAwUFbgZH20etwrXyVyCBu9OxRSi8T/38QsvnkJyvq2LufW2lDCOzm7t/U9C1mkhJGWYfCuFBCmIuNivrg==}
-
- '@vue/shared@https://registry.npmmirror.com/@vue/shared/-/shared-3.2.45.tgz':
- resolution: {integrity: sha512-Ewzq5Yhimg7pSztDV+RH1UDKBzmtqieXQlpTVm2AwraoRL/Rks96mvd8Vgi7Lj+h+TH8dv7mXD3FRZR3TUvbSg==, tarball: https://registry.npmmirror.com/@vue/shared/-/shared-3.2.45.tgz}
- version: 3.2.45
-
- '@vueuse/core@9.13.0':
- resolution: {integrity: sha512-pujnclbeHWxxPRqXWmdkKV5OX4Wk4YeK7wusHqRwU0Q7EFusHoqNA/aPhB6KCh9hEqJkLAJo7bb0Lh9b+OIVzw==}
- version: 9.13.0
-
- '@vueuse/metadata@9.13.0':
- resolution: {integrity: sha512-gdU7TKNAUVlXXLbaF+ZCfte8BjRJQWPCa2J55+7/h+yDtzw3vOoGQDRXzI6pyKyo6bXFT5/QoPE4hAknExjRLQ==}
-
- '@vueuse/shared@9.13.0':
- resolution: {integrity: sha512-UrnhU+Cnufu4S6JLCPZnkWh0WwZGUp72ktOF2DFptMlOs3TOdVv8xJN53zhHGARmVOsz5KqOls09+J1NR6sBKw==}
- version: 9.13.0
-
- abbrev@1.1.1:
- resolution: {integrity: sha512-nne9/IiQ/hzIhY6pdDnbBtz7DjPTKrY00P/zvPSm5pOFkl6xuGrGnXn/VtTNNfNtAfZ9/1RtehkszU9qcTii0Q==}
-
- accepts@1.3.8:
- resolution: {integrity: sha512-PYAthTa2m2VKxuvSD3DPC/Gy+U+sOA1LAuT8mkmRuvw+NACSaeXEQ+NHcVF7rONl6qcaxV3Uuemwawk+7+SJLw==}
- engines: {node: '>= 0.6'}
-
- acorn-dynamic-import@4.0.0:
- resolution: {integrity: sha512-d3OEjQV4ROpoflsnUA8HozoIR504TFxNivYEUi6uwz0IYhBkTDXGuWlNdMtybRt3nqVx/L6XqMt0FxkXuWKZhw==}
- deprecated: This is probably built in to whatever tool you're using. If you still need it... idk
- peerDependencies:
- acorn: ^6.0.0
-
- acorn-jsx@5.3.2:
- resolution: {integrity: sha512-rq9s+JNhf0IChjtDXxllJ7g41oZk5SlXtp0LHwyA5cejwn7vKmKp4pPri6YEePv2PU65sAsegbXtIinmDFDXgQ==}
- peerDependencies:
- acorn: ^6.0.0 || ^7.0.0 || ^8.0.0
-
- acorn@6.4.2:
- resolution: {integrity: sha512-XtGIhXwF8YM8bJhGxG5kXgjkEuNGLTkoYqVE+KMR+aspr4KGYmKYg7yUe3KghyQ9yheNwLnjmzh/7+gfDBmHCQ==}
- engines: {node: '>=0.4.0'}
- hasBin: true
-
- acorn@8.13.0:
- resolution: {integrity: sha512-8zSiw54Oxrdym50NlZ9sUusyO1Z1ZchgRLWRaK6c86XJFClyCgFKetdowBg5bKxyp/u+CDBJG4Mpp0m3HLZl9w==}
- engines: {node: '>=0.4.0'}
- hasBin: true
-
- agent-base@6.0.2:
- resolution: {integrity: sha512-RZNwNclF7+MS/8bDg70amg32dyeZGZxiDuQmZxKLAlQjr3jGyLx+4Kkk58UO7D2QdgFIQCovuSuZESne6RG6XQ==}
- engines: {node: '>= 6.0.0'}
-
- ajv@8.17.1:
- resolution: {integrity: sha512-B/gBuNg5SiMTrPkC+A2+cW0RszwxYmn6VYxB/inlBStS5nx6xHIt/ehKRhIMhqusl7a8LjQoZnjCs5vhwxOQ1g==}
-
- ansi-regex@5.0.1:
- resolution: {integrity: sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==}
- engines: {node: '>=8'}
-
- ansi-styles@3.2.1:
- resolution: {integrity: sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==}
- engines: {node: '>=4'}
-
- ansi-styles@4.3.0:
- resolution: {integrity: sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==}
- engines: {node: '>=8'}
-
- anymatch@https://registry.npmmirror.com/anymatch/-/anymatch-3.1.3.tgz:
- resolution: {integrity: sha512-KMReFUr0B4t+D+OBkjR3KYqvocp2XaSzO55UcB6mgQMd3KbcE+mWTyvVV7D/zsdEbNnV6acZUutkiHQXvTr1Rw==, tarball: https://registry.npmmirror.com/anymatch/-/anymatch-3.1.3.tgz}
- version: 3.1.3
- engines: {node: '>= 8'}
-
- aproba@2.0.0:
- resolution: {integrity: sha512-lYe4Gx7QT+MKGbDsA+Z+he/Wtef0BiwDOlK/XkBrdfsh9J/jPPXbX0tE9x9cl27Tmu5gg3QUbUrQYa/y+KOHPQ==}
-
- are-we-there-yet@2.0.0:
- resolution: {integrity: sha512-Ci/qENmwHnsYo9xKIcUJN5LeDKdJ6R1Z1j9V/J5wyq8nh/mYPEpIKJbBZXtZjG04HiK7zV/p6Vs9952MrMeUIw==}
- engines: {node: '>=10'}
- deprecated: This package is no longer supported.
-
- argparse@1.0.10:
- resolution: {integrity: sha512-o5Roy6tNG4SL/FOkCAN6RzjiakZS25RLYFrcMttJqbdd8BWrnA+fGz57iN5Pb06pvBGvl5gQ0B48dJlslXvoTg==}
-
- aria-query@5.3.2:
- resolution: {integrity: sha512-COROpnaoap1E2F000S62r6A60uHZnmlvomhfyT2DlTcrY1OrBKn2UhH7qn5wTC9zMvD0AY7csdPSNwKP+7WiQw==}
- engines: {node: '>= 0.4'}
-
- array-bounds@1.0.1:
- resolution: {integrity: sha512-8wdW3ZGk6UjMPJx/glyEt0sLzzwAE1bhToPsO1W2pbpR2gULyxe3BjSiuJFheP50T/GgODVPz2fuMUmIywt8cQ==}
-
- array-normalize@1.1.4:
- resolution: {integrity: sha512-fCp0wKFLjvSPmCn4F5Tiw4M3lpMZoHlCjfcs7nNzuj3vqQQ1/a8cgB9DXcpDSn18c+coLnaW7rqfcYCvKbyJXg==}
-
- array-timsort@https://registry.npmmirror.com/array-timsort/-/array-timsort-1.0.3.tgz:
- resolution: {integrity: sha512-/+3GRL7dDAGEfM6TseQk/U+mi18TU2Ms9I3UlLdUMhz2hbvGNTKdj9xniwXfUqgYhHxRx0+8UnKkvlNwVU+cWQ==, tarball: https://registry.npmmirror.com/array-timsort/-/array-timsort-1.0.3.tgz}
- version: 1.0.3
-
- async-validator@4.2.5:
- resolution: {integrity: sha512-7HhHjtERjqlNbZtqNqy2rckN/SpOOlmDliet+lP7k+eKZEjPk3DgyeU9lIXLdeLz0uBbbVp+9Qdow9wJWgwwfg==}
-
- asynckit@0.4.0:
- resolution: {integrity: sha512-Oei9OH4tRh0YqU3GxhX79dM/mwVgvbZJaSNaRk+bshkj0S5cfHcgYakreBjrHwatXKbz+IoIdYLxrKim2MjW0Q==}
-
- atom-material-icons@3.0.0:
- resolution: {integrity: sha512-qtMR5KxQh82iLIFOdrQZViq21GTNT+/mccrVAg41hIlrhpwHAEjCeGj91Iyas9GL28ld0D0hagHv6ipGWlLZHw==}
-
- attr-accept@2.2.2:
- resolution: {integrity: sha512-7prDjvt9HmqiZ0cl5CRjtS84sEyhsHP2coDkaZKRKVfCDo9s7iw7ChVmar78Gu9pC4SoR/28wFu/G5JJhTnqEg==}
- engines: {node: '>=4'}
-
- axios@1.7.7:
- resolution: {integrity: sha512-S4kL7XrjgBmvdGut0sN3yJxqYzrDOnivkBiN0OFs6hLiUam3UPvswUo0kqGyhqUZGEOytHyumEdXsAkgCOUf3Q==}
-
- axobject-query@4.1.0:
- resolution: {integrity: sha512-qIj0G9wZbMGNLjLmg1PT6v2mE9AH2zlnADJD/2tC6E00hgmhUOfEB6greHPAfLRSufHqROIUTkw6E+M3lH0PTQ==}
- engines: {node: '>= 0.4'}
-
- balanced-match@1.0.2:
- resolution: {integrity: sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==}
-
- balanced-match@https://registry.npmmirror.com/balanced-match/-/balanced-match-1.0.2.tgz:
- resolution: {integrity: sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==, tarball: https://registry.npmmirror.com/balanced-match/-/balanced-match-1.0.2.tgz}
- version: 1.0.2
-
- binary-extensions@https://registry.npmmirror.com/binary-extensions/-/binary-extensions-2.2.0.tgz:
- resolution: {integrity: sha512-jDctJ/IVQbZoJykoeHbhXpOlNBqGNcwXJKJog42E5HDPUwQTSdjCHdihjj0DlnheQ7blbT6dHOafNAiS8ooQKA==, tarball: https://registry.npmmirror.com/binary-extensions/-/binary-extensions-2.2.0.tgz}
- version: 2.2.0
- engines: {node: '>=8'}
-
- binary-search-bounds@2.0.5:
- resolution: {integrity: sha512-H0ea4Fd3lS1+sTEB2TgcLoK21lLhwEJzlQv3IN47pJS976Gx4zoWe0ak3q+uYh60ppQxg9F16Ri4tS1sfD4+jA==}
-
- boolbase@1.0.0:
- resolution: {integrity: sha512-JZOSA7Mo9sNGB8+UjSgzdLtokWAky1zbztM3WRLCbZ70/3cTANmQmOdR7y2g+J0e2WXywy1yS468tY+IruqEww==}
-
- brace-expansion@1.1.11:
- resolution: {integrity: sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==}
-
- brace-expansion@https://registry.npmmirror.com/brace-expansion/-/brace-expansion-2.0.1.tgz:
- resolution: {integrity: sha512-XnAIvQ8eM+kC6aULx6wuQiwVsnzsi9d3WxzV3FpWTGA19F621kwdbsAcFKXgKUHZWsy+mY6iL1sHTxWEFCytDA==, tarball: https://registry.npmmirror.com/brace-expansion/-/brace-expansion-2.0.1.tgz}
- version: 2.0.1
-
- braces@3.0.2:
- resolution: {integrity: sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==}
- engines: {node: '>=8'}
-
- braces@https://registry.npmmirror.com/braces/-/braces-3.0.2.tgz:
- resolution: {integrity: sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==, tarball: https://registry.npmmirror.com/braces/-/braces-3.0.2.tgz}
- version: 3.0.2
- engines: {node: '>=8'}
-
- buble@0.19.8:
- resolution: {integrity: sha512-IoGZzrUTY5fKXVkgGHw3QeXFMUNBFv+9l8a4QJKG1JhG3nCMHTdEX1DCOg8568E2Q9qvAQIiSokv6Jsgx8p2cA==}
- hasBin: true
-
- bubleify@1.2.1:
- resolution: {integrity: sha512-vp3NHmaQVoKaKWvi15FTMinPNjfp+47+/kFJ9ifezdMF/CBLArCxDVUh+FQE3qRxCRj1qyjJqilTBHHqlM8MaQ==}
- engines: {node: '>=4.0.0'}
-
- bytes@3.0.0:
- resolution: {integrity: sha512-pMhOfFDPiv9t5jjIXkHosWmkSyQbvsgEVNkz0ERHbuLh2T/7j4Mqqpz523Fe8MVY89KC6Sh/QfS2sM+SjgFDcw==}
- engines: {node: '>= 0.8'}
-
- chalk@2.4.2:
- resolution: {integrity: sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==}
- engines: {node: '>=4'}
-
- charenc@0.0.2:
- resolution: {integrity: sha512-yrLQ/yVUFXkzg7EDQsPieE/53+0RlaWTs+wBrvW36cyilJ2SaDWfl4Yj7MtLTXleV9uEKefbAGUPv2/iWSooRA==}
-
- chart.js@4.4.5:
- resolution: {integrity: sha512-CVVjg1RYTJV9OCC8WeJPMx8gsV8K6WIyIEQUE3ui4AR9Hfgls9URri6Ja3hyMVBbTF8Q2KFa19PE815gWcWhng==}
- engines: {pnpm: '>=8'}
-
- chartjs-adapter-date-fns@3.0.0:
- resolution: {integrity: sha512-Rs3iEB3Q5pJ973J93OBTpnP7qoGwvq3nUnoMdtxO+9aoJof7UFcRbWcIDteXuYd1fgAvct/32T9qaLyLuZVwCg==}
- peerDependencies:
- chart.js: '>=2.8.0'
- date-fns: '>=2.0.0'
-
- chokidar@4.0.1:
- resolution: {integrity: sha512-n8enUVCED/KVRQlab1hr3MVpcVMvxtZjmEa956u+4YijlmQED223XMSYj2tLuKvr4jcCTzNNMpQDUer72MMmzA==}
- engines: {node: '>= 14.16.0'}
-
- chokidar@https://registry.npmmirror.com/chokidar/-/chokidar-3.5.3.tgz:
- resolution: {integrity: sha512-Dr3sfKRP6oTcjf2JmUmFJfeVMvXBdegxB0iVQ5eb2V10uFJUCAS8OByZdVAyVb8xXNz3GjjTgj9kLWsZTqE6kw==, tarball: https://registry.npmmirror.com/chokidar/-/chokidar-3.5.3.tgz}
- version: 3.5.3
- engines: {node: '>= 8.10.0'}
-
- chownr@2.0.0:
- resolution: {integrity: sha512-bIomtDF5KGpdogkLd9VspvFzk9KfpyyGlS8YFVZl7TGPBHL5snIOnxeshwVgPteQ9b4Eydl+pVbIyE1DcvCWgQ==}
- engines: {node: '>=10'}
-
- clamp@1.0.1:
- resolution: {integrity: sha512-kgMuFyE78OC6Dyu3Dy7vcx4uy97EIbVxJB/B0eJ3bUNAkwdNcxYzgKltnyADiYwsR7SEqkkUPsEUT//OVS6XMA==}
-
- clipboard@2.0.11:
- resolution: {integrity: sha512-C+0bbOqkezLIsmWSvlsXS0Q0bmkugu7jcfMIACB+RDEntIzQIkdr148we28AfSloQLRdZlYL/QYyrq05j/3Faw==}
-
- cliui@8.0.1:
- resolution: {integrity: sha512-BSeNnyus75C4//NQ9gQt1/csTXyo/8Sb+afLAkzAptFuMsod9HFokGNudZpi/oQV73hnVK+sR+5PVRMd+Dr7YQ==}
- engines: {node: '>=12'}
-
- code-red@1.0.4:
- resolution: {integrity: sha512-7qJWqItLA8/VPVlKJlFXU+NBlo/qyfs39aJcuMT/2ere32ZqvF5OSxgdM5xOfJJ7O429gg2HM47y8v9P+9wrNw==}
-
- codemirror-wrapped-line-indent@1.0.8:
- resolution: {integrity: sha512-5UwuHCz4oAZuvot1DbfFxSxJacTESdNGa/KpJD7HfpVpDAJdgB1vV9OG4b4pkJqPWuOfIpFLTQEKS85kTpV+XA==}
- peerDependencies:
- '@codemirror/language': ^6.9.0
- '@codemirror/state': ^6.2.1
- '@codemirror/view': ^6.17.1
-
- color-convert@1.9.3:
- resolution: {integrity: sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==}
-
- color-convert@2.0.1:
- resolution: {integrity: sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==}
- engines: {node: '>=7.0.0'}
-
- color-name@1.1.3:
- resolution: {integrity: sha512-72fSenhMw2HZMTVHeCA9KCmpEIbzWiQsjN+BHcBbS9vr1mtt+vJjPdksIBNUmKAW8TFUDPJK5SUU3QhE9NEXDw==}
-
- color-name@1.1.4:
- resolution: {integrity: sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==}
-
- color-support@1.1.3:
- resolution: {integrity: sha512-qiBjkpbMLO/HL68y+lh4q0/O1MZFj2RX6X/KmMa3+gJD3z+WwI1ZzDHysvqHGS3mP6mznPckpXmw1nI9cJjyRg==}
- hasBin: true
-
- combined-stream@1.0.8:
- resolution: {integrity: sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==}
- engines: {node: '>= 0.8'}
-
- commander@1.1.1:
- resolution: {integrity: sha512-71Rod2AhcH3JhkBikVpNd0pA+fWsmAaVoti6OR38T76chA7vE3pSerS0Jor4wDw+tOueD2zLVvFOw5H0Rcj7rA==}
- engines: {node: '>= 0.6.x'}
-
- commander@7.2.0:
- resolution: {integrity: sha512-QrWXB+ZQSVPmIWIhtEO9H+gwHaMGYiF5ChvoJ+K9ZGHG/sVsa6yiesAD1GC/x46sET00Xlwo1u49RVVVzvcSkw==}
- engines: {node: '>= 10'}
-
- commander@9.5.0:
- resolution: {integrity: sha512-KRs7WVDKg86PWiuAqhDrAQnTXZKraVcCc6vFdL14qrZ/DcWwuRo7VoiYXalXO7S5GKpqYiVEwCbgFDfxNHKJBQ==}
- engines: {node: ^12.20.0 || >=14}
-
- comment-json@https://registry.npmmirror.com/comment-json/-/comment-json-4.2.3.tgz:
- resolution: {integrity: sha512-SsxdiOf064DWoZLH799Ata6u7iV658A11PlWtZATDlXPpKGJnbJZ5Z24ybixAi+LUUqJ/GKowAejtC5GFUG7Tw==, tarball: https://registry.npmmirror.com/comment-json/-/comment-json-4.2.3.tgz}
- version: 4.2.3
- engines: {node: '>= 6'}
-
- compressible@2.0.18:
- resolution: {integrity: sha512-AF3r7P5dWxL8MxyITRMlORQNaOA2IkAFaTr4k7BUumjPtRpGDTZpl0Pb1XCO6JeDCBdp126Cgs9sMxqSjgYyRg==}
- engines: {node: '>= 0.6'}
-
- compression@1.7.4:
- resolution: {integrity: sha512-jaSIDzP9pZVS4ZfQ+TzvtiWhdpFhE2RDHz8QJkpX9SIpLq88VueF5jJw6t+6CUQcAoA6t+x89MLrWAqpfDE8iQ==}
- engines: {node: '>= 0.8.0'}
-
- concat-map@0.0.1:
- resolution: {integrity: sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg==}
-
- consola@https://registry.npmmirror.com/consola/-/consola-2.15.3.tgz:
- resolution: {integrity: sha512-9vAdYbHj6x2fLKC4+oPH0kFzY/orMZyG2Aj+kNylHxKGJ/Ed4dpNyAQYwJOdqO4zdM7XpVHmyejQDcQHrnuXbw==, tarball: https://registry.npmmirror.com/consola/-/consola-2.15.3.tgz}
- version: 2.15.3
-
- console-clear@1.1.1:
- resolution: {integrity: sha512-pMD+MVR538ipqkG5JXeOEbKWS5um1H4LUUccUQG68qpeqBYbzYy79Gh55jkd2TtPdRfUaLWdv6LPP//5Zt0aPQ==}
- engines: {node: '>=4'}
-
- console-control-strings@1.1.0:
- resolution: {integrity: sha512-ty/fTekppD2fIwRvnZAVdeOiGd1c7YXEixbgJTNzqcxJWKQnjJ/V1bNEEE6hygpM3WjwHFUVK6HTjWSzV4a8sQ==}
-
- core-js@3.38.1:
- resolution: {integrity: sha512-OP35aUorbU3Zvlx7pjsFdu1rGNnD4pgw/CWoYzRY3t2EzoVT7shKHY1dlAy3f41cGIO7ZDPQimhGFTlEYkG/Hw==}
-
- core-util-is@https://registry.npmmirror.com/core-util-is/-/core-util-is-1.0.3.tgz:
- resolution: {integrity: sha512-ZQBvi1DcpJ4GDqanjucZ2Hj3wEO5pZDS89BWbkcrvdxksJorwUDDZamX9ldFkp9aw2lmBDLgkObEA4DWNJ9FYQ==, tarball: https://registry.npmmirror.com/core-util-is/-/core-util-is-1.0.3.tgz}
- version: 1.0.3
-
- crawlab-ui@0.7.0-rc3:
- resolution: {integrity: sha512-WF7DLjrXa0J3uY/u7iTtZyqbhCIHZaW2K5dWHMj28GCaT69uvPRR32AfZPWag5sarqXdOmKNSOt5NidAPKqKpQ==}
- peerDependencies:
- '@element-plus/icons-vue': ^2.3.1
- '@fortawesome/fontawesome-common-types': ^6.5.2
- '@fortawesome/fontawesome-svg-core': ^6.5.2
- '@fortawesome/free-brands-svg-icons': ^6.5.2
- '@fortawesome/free-regular-svg-icons': ^6.5.2
- '@fortawesome/free-solid-svg-icons': ^6.5.2
- '@fortawesome/vue-fontawesome': ^3.0.8
- '@lexical/code': ^0.16.1
- '@lexical/history': ^0.16.1
- '@lexical/html': ^0.16.1
- '@lexical/link': ^0.16.1
- '@lexical/list': ^0.16.1
- '@lexical/markdown': ^0.16.1
- '@lexical/rich-text': ^0.16.1
- '@lexical/selection': ^0.16.1
- '@lexical/table': ^0.16.1
- '@lexical/text': ^0.16.1
- '@lexical/utils': ^0.16.1
- '@popperjs/core': ^2.11.8
- '@types/getos': ^3.0.4
- '@types/humanize-duration': ^3.27.4
- '@types/javascript-time-ago': ^2.0.8
- '@types/lodash': ^4.17.6
- '@types/md5': ^2.3.5
- '@types/pinyin': ^2.10.2
- '@types/showdown': ^2.0.6
- '@types/three': ^0.166.0
- '@types/url-join': ^4.0.3
- async-validator: ^4.2.5
- atom-material-icons: ^3.0.0
- axios: ^1.7.2
- chart.js: ^4.4.3
- chartjs-adapter-date-fns: ^3.0.0
- clipboard: ^2.0.11
- core-js: ^3.37.1
- crawlab-vue3-dropzone: 3.0.3
- cron-parser: ^4.9.0
- cronstrue: ^2.50.0
- dayjs: ^1.11.11
- element-plus: ^2.7.6
- font-awesome: ^4.7.0
- humanize-duration: ^3.32.1
- javascript-time-ago: ^2.5.10
- lexical: ^0.16.1
- lodash: ^4.17.21
- md5: ^2.3.0
- mitt: ^3.0.1
- monaco-editor: ^0.50.0
- normalize.css: ^8.0.1
- os-name: ^5.0.1
- pinyin: ^2.10.2
- point-cluster: ^3.1.8
- prismjs: ^1.29.0
- showdown: ^2.1.0
- three: ^0.166.1
- tiny-invariant: ^1.3.3
- url-join: ^4.0.1
- url-regex: ^5.0.0
- vite-svg-loader: ^5.1.0
- vue: ^3.4
- vue-chartjs: ^5.3.1
- vue-clipboard3: ^2.0.0
- vue-i18n: 9.13.1
- vue-router: ^4.4.0
- vuex: ^4.1.0
-
- crawlab-vue3-dropzone@3.0.3:
- resolution: {integrity: sha512-s/GHSTteFqJcMk14Xkp1imwEyrOCa+uwwH1eCsTAxVzaYocpiDpbDjQ3u3vrMyws9InsqqgtmAJhy2vGHEsk/Q==}
- version: 3.0.3
- peerDependencies:
- vue: '>=3.2.45'
-
- crelt@1.0.6:
- resolution: {integrity: sha512-VQ2MBenTq1fWZUH9DJNGti7kKv6EeAuYr3cLwxUWhIu1baTaXh4Ib5W2CqHVqib4/MqbYGJqiL3Zb8GJZr3l4g==}
-
- cron-parser@4.9.0:
- resolution: {integrity: sha512-p0SaNjrHOnQeR8/VnfGbmg9te2kfyYSQ7Sc/j/6DtPL3JQvKxmjO9TSjNFpujqV3vEYYBvNNvXSxzyksBWAx1Q==}
- engines: {node: '>=12.0.0'}
-
- cronstrue@2.50.0:
- resolution: {integrity: sha512-ULYhWIonJzlScCCQrPUG5uMXzXxSixty4djud9SS37DoNxDdkeRocxzHuAo4ImRBUK+mAuU5X9TSwEDccnnuPg==}
- hasBin: true
-
- cross-spawn@7.0.3:
- resolution: {integrity: sha512-iRDPJKUPVEND7dHPO8rkbOnPpyDygcDFtWjpeWNCgy8WP2rXcxXL8TskReQl6OrB2G7+UJrags1q15Fudc7G6w==}
- engines: {node: '>= 8'}
-
- crypt@0.0.2:
- resolution: {integrity: sha512-mCxBlsHFYh9C+HVpiEacem8FEBnMXgU9gy4zmNC+SXAZNB/1idgp/aulFJ4FgCi7GPEVbfyng092GqL2k2rmow==}
-
- css-select@5.1.0:
- resolution: {integrity: sha512-nwoRF1rvRRnnCqqY7updORDsuqKzqYJ28+oSMaJMMgOauh3fvwHqMS7EZpIPqK8GL+g9mKxF1vP/ZjSeNjEVHg==}
-
- css-tree@2.2.1:
- resolution: {integrity: sha512-OA0mILzGc1kCOCSJerOeqDxDQ4HOh+G8NbOJFOTgOCzpw7fCBubk0fEyxp8AgOL/jvLgYA/uV0cMbe43ElF1JA==}
- engines: {node: ^10 || ^12.20.0 || ^14.13.0 || >=15.0.0, npm: '>=7.0.0'}
-
- css-tree@2.3.1:
- resolution: {integrity: sha512-6Fv1DV/TYw//QF5IzQdqsNDjx/wc8TrMBZsqjL9eW01tWb7R7k/mq+/VXfJCl7SoD5emsJop9cOByJZfs8hYIw==}
- engines: {node: ^10 || ^12.20.0 || ^14.13.0 || >=15.0.0}
-
- css-what@6.1.0:
- resolution: {integrity: sha512-HTUrgRJ7r4dsZKU6GjmpfRK1O76h97Z8MfS1G0FozR+oF2kG6Vfe8JE6zwrkbxigziPHinCJ+gCPjA9EaBDtRw==}
- engines: {node: '>= 6'}
-
- csso@5.0.5:
- resolution: {integrity: sha512-0LrrStPOdJj+SPCCrGhzryycLjwcgUSHBtxNA8aIDxf0GLsRh1cKYhB00Gd1lDOS4yGH69+SNn13+TWbVHETFQ==}
- engines: {node: ^10 || ^12.20.0 || ^14.13.0 || >=15.0.0, npm: '>=7.0.0'}
-
- csstype@3.1.3:
- resolution: {integrity: sha512-M1uQkMl8rQK/szD0LNhtqxIPLpimGm8sOBwU7lLnCpSbTyY3yeU1Vc7l4KT5zT4s/yOxHH5O7tIuuLOCnLADRw==}
-
- date-fns@4.1.0:
- resolution: {integrity: sha512-Ukq0owbQXxa/U3EGtsdVBkR1w7KOQ5gIBqdH2hkvknzZPYvBxb/aa6E8L7tmjFtkwZBu3UXBbjIgPo/Ez4xaNg==}
-
- dayjs@1.11.13:
- resolution: {integrity: sha512-oaMBel6gjolK862uaPQOVTA7q3TZhuSvuMQAAglQDOWYO9A91IrAOUJEyKVlqJlHE0vq5p5UXxzdPfMH/x6xNg==}
-
- de-indent@https://registry.npmmirror.com/de-indent/-/de-indent-1.0.2.tgz:
- resolution: {integrity: sha512-e/1zu3xH5MQryN2zdVaF0OrdNLUbvWxzMbi+iNA6Bky7l1RoP8a2fIbRocyHclXt/arDrrR6lL3TqFD9pMQTsg==, tarball: https://registry.npmmirror.com/de-indent/-/de-indent-1.0.2.tgz}
- version: 1.0.2
-
- debug@2.6.9:
- resolution: {integrity: sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==}
- peerDependencies:
- supports-color: '*'
- peerDependenciesMeta:
- supports-color:
- optional: true
-
- debug@4.3.4:
- resolution: {integrity: sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ==}
- engines: {node: '>=6.0'}
- peerDependencies:
- supports-color: '*'
- peerDependenciesMeta:
- supports-color:
- optional: true
-
- define-lazy-prop@2.0.0:
- resolution: {integrity: sha512-Ds09qNh8yw3khSjiJjiUInaGX9xlqZDY7JVryGxdxV7NPeuqQfplOpQ66yJFZut3jLa5zOwkXw1g9EI2uKh4Og==}
- engines: {node: '>=8'}
-
- defined@1.0.1:
- resolution: {integrity: sha512-hsBd2qSVCRE+5PmNdHt1uzyrFu5d3RwmFDKzyNZMFq/EwDNJF7Ee5+D5oEKF0hU6LhtoUF1macFvOe4AskQC1Q==}
-
- delayed-stream@1.0.0:
- resolution: {integrity: sha512-ZySD7Nf91aLB0RxL4KGrKHBXl7Eds1DAmEdcoVawXnLD7SDhpNgtuII2aAkg7a7QS41jxPSZ17p4VdGnMHk3MQ==}
- engines: {node: '>=0.4.0'}
-
- delegate@3.2.0:
- resolution: {integrity: sha512-IofjkYBZaZivn0V8nnsMJGBr4jVLxHDheKSW88PyxS5QC4Vo9ZbZVvhzlSxY87fVq3STR6r+4cGepyHkcWOQSw==}
-
- delegates@1.0.0:
- resolution: {integrity: sha512-bd2L678uiWATM6m5Z1VzNCErI3jiGzt6HGY8OVICs40JQq/HALfbyNJmp0UDakEY4pMMaN0Ly5om/B1VI/+xfQ==}
-
- detect-libc@1.0.3:
- resolution: {integrity: sha512-pGjwhsmsp4kL2RTz08wcOlGN83otlqHeD/Z5T8GXZB+/YcpQ/dgo+lbU8ZsGxV0HIvqqxo9l7mqYwyYMD9bKDg==}
- engines: {node: '>=0.10'}
- hasBin: true
-
- detect-libc@2.0.1:
- resolution: {integrity: sha512-463v3ZeIrcWtdgIg6vI6XUncguvr2TnGl4SzDXinkt9mSLpBJKXT3mW6xT3VQdDN11+WVs29pgvivTc4Lp8v+w==}
- engines: {node: '>=8'}
-
- diff-sequences@29.6.3:
- resolution: {integrity: sha512-EjePK1srD3P08o2j4f0ExnylqRs5B9tJjcp9t1krH2qRi8CCdsYfwe9JgSLurFBWwq4uOlipzfk5fHNvwFKr8Q==}
- engines: {node: ^14.15.0 || ^16.10.0 || >=18.0.0}
-
- dom-serializer@2.0.0:
- resolution: {integrity: sha512-wIkAryiqt/nV5EQKqQpo3SToSOV9J0DnbJqwK7Wv/Trc92zIAYZ4FlMu+JPFW1DfGFt81ZTCGgDEabffXeLyJg==}
-
- domelementtype@2.3.0:
- resolution: {integrity: sha512-OLETBj6w0OsagBwdXnPdN0cnMfF9opN69co+7ZrbfPGrdpPVNBUj02spi6B1N7wChLQiPn4CSH/zJvXw56gmHw==}
-
- domhandler@5.0.3:
- resolution: {integrity: sha512-cgwlv/1iFQiFnU96XXgROh8xTeetsnJiDsTc7TYCLFd9+/WNkIqPTxiM/8pSd8VIrhXGTf1Ny1q1hquVqDJB5w==}
- engines: {node: '>= 4'}
-
- domutils@3.1.0:
- resolution: {integrity: sha512-H78uMmQtI2AhgDJjWeQmHwJJ2bLPD3GMmO7Zja/ZZh84wkm+4ut+IUnUdRa8uCGX88DiVx1j6FRe1XfxEgjEZA==}
-
- dtype@2.0.0:
- resolution: {integrity: sha512-s2YVcLKdFGS0hpFqJaTwscsyt0E8nNFdmo73Ocd81xNPj4URI4rj6D60A+vFMIw7BXWlb4yRkEwfBqcZzPGiZg==}
- engines: {node: '>= 0.8.0'}
-
- element-plus@2.8.5:
- resolution: {integrity: sha512-Px+kPbRTVvn5oa5+9saa7QEOnUweKXm0JVI7yJHzKF/doQGixwcFMsQEF2+3Fy62EA/7dRRKVuhsNGGZYNk3cw==}
- version: 2.8.5
- peerDependencies:
- vue: ^3.2.0
-
- emoji-regex@8.0.0:
- resolution: {integrity: sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==}
-
- entities@4.5.0:
- resolution: {integrity: sha512-V0hjH4dGPh9Ao5p0MoRY6BVqtwCjhz6vI5LT8AJ55H+4g9/4vbHx1I54fS0XuclLhDHArPQCiMjDxjaL8fPxhw==}
- engines: {node: '>=0.12'}
-
- esbuild-android-64@0.15.16:
- resolution: {integrity: sha512-Vwkv/sT0zMSgPSVO3Jlt1pUbnZuOgtOQJkJkyyJFAlLe7BiT8e9ESzo0zQSx4c3wW4T6kGChmKDPMbWTgtliQA==}
- engines: {node: '>=12'}
- cpu: [x64]
- os: [android]
-
- esbuild-android-arm64@0.15.16:
- resolution: {integrity: sha512-lqfKuofMExL5niNV3gnhMUYacSXfsvzTa/58sDlBET/hCOG99Zmeh+lz6kvdgvGOsImeo6J9SW21rFCogNPLxg==}
- engines: {node: '>=12'}
- cpu: [arm64]
- os: [android]
-
- esbuild-darwin-64@0.15.16:
- resolution: {integrity: sha512-wo2VWk/n/9V2TmqUZ/KpzRjCEcr00n7yahEdmtzlrfQ3lfMCf3Wa+0sqHAbjk3C6CKkR3WKK/whkMq5Gj4Da9g==}
- engines: {node: '>=12'}
- cpu: [x64]
- os: [darwin]
-
- esbuild-darwin-arm64@0.15.16:
- resolution: {integrity: sha512-fMXaUr5ou0M4WnewBKsspMtX++C1yIa3nJ5R2LSbLCfJT3uFdcRoU/NZjoM4kOMKyOD9Sa/2vlgN8G07K3SJnw==}
- engines: {node: '>=12'}
- cpu: [arm64]
- os: [darwin]
-
- esbuild-freebsd-64@0.15.16:
- resolution: {integrity: sha512-UzIc0xlRx5x9kRuMr+E3+hlSOxa/aRqfuMfiYBXu2jJ8Mzej4lGL7+o6F5hzhLqWfWm1GWHNakIdlqg1ayaTNQ==}
- engines: {node: '>=12'}
- cpu: [x64]
- os: [freebsd]
-
- esbuild-freebsd-arm64@0.15.16:
- resolution: {integrity: sha512-8xyiYuGc0DLZphFQIiYaLHlfoP+hAN9RHbE+Ibh8EUcDNHAqbQgUrQg7pE7Bo00rXmQ5Ap6KFgcR0b4ALZls1g==}
- engines: {node: '>=12'}
- cpu: [arm64]
- os: [freebsd]
-
- esbuild-linux-32@0.15.16:
- resolution: {integrity: sha512-iGijUTV+0kIMyUVoynK0v+32Oi8yyp0xwMzX69GX+5+AniNy/C/AL1MjFTsozRp/3xQPl7jVux/PLe2ds10/2w==}
- engines: {node: '>=12'}
- cpu: [ia32]
- os: [linux]
-
- esbuild-linux-64@0.15.16:
- resolution: {integrity: sha512-tuSOjXdLw7VzaUj89fIdAaQT7zFGbKBcz4YxbWrOiXkwscYgE7HtTxUavreBbnRkGxKwr9iT/gmeJWNm4djy/g==}
- engines: {node: '>=12'}
- cpu: [x64]
- os: [linux]
-
- esbuild-linux-arm64@0.15.16:
- resolution: {integrity: sha512-mPYksnfHnemNrvjrDhZyixL/AfbJN0Xn9S34ZOHYdh6/jJcNd8iTsv3JwJoEvTJqjMggjMhGUPJAdjnFBHoH8A==}
- engines: {node: '>=12'}
- cpu: [arm64]
- os: [linux]
-
- esbuild-linux-arm@0.15.16:
- resolution: {integrity: sha512-XKcrxCEXDTOuoRj5l12tJnkvuxXBMKwEC5j0JISw3ziLf0j4zIwXbKbTmUrKFWbo6ZgvNpa7Y5dnbsjVvH39bQ==}
- engines: {node: '>=12'}
- cpu: [arm]
- os: [linux]
-
- esbuild-linux-mips64le@0.15.16:
- resolution: {integrity: sha512-kSJO2PXaxfm0pWY39+YX+QtpFqyyrcp0ZeI8QPTrcFVQoWEPiPVtOfTZeS3ZKedfH+Ga38c4DSzmKMQJocQv6A==}
- engines: {node: '>=12'}
- cpu: [mips64el]
- os: [linux]
-
- esbuild-linux-ppc64le@0.15.16:
- resolution: {integrity: sha512-NimPikwkBY0yGABw6SlhKrtT35sU4O23xkhlrTT/O6lSxv3Pm5iSc6OYaqVAHWkLdVf31bF4UDVFO+D990WpAA==}
- engines: {node: '>=12'}
- cpu: [ppc64]
- os: [linux]
-
- esbuild-linux-riscv64@0.15.16:
- resolution: {integrity: sha512-ty2YUHZlwFOwp7pR+J87M4CVrXJIf5ZZtU/umpxgVJBXvWjhziSLEQxvl30SYfUPq0nzeWKBGw5i/DieiHeKfw==}
- engines: {node: '>=12'}
- cpu: [riscv64]
- os: [linux]
-
- esbuild-linux-s390x@0.15.16:
- resolution: {integrity: sha512-VkZaGssvPDQtx4fvVdZ9czezmyWyzpQhEbSNsHZZN0BHvxRLOYAQ7sjay8nMQwYswP6O2KlZluRMNPYefFRs+w==}
- engines: {node: '>=12'}
- cpu: [s390x]
- os: [linux]
-
- esbuild-netbsd-64@0.15.16:
- resolution: {integrity: sha512-ElQ9rhdY51et6MJTWrCPbqOd/YuPowD7Cxx3ee8wlmXQQVW7UvQI6nSprJ9uVFQISqSF5e5EWpwWqXZsECLvXg==}
- engines: {node: '>=12'}
- cpu: [x64]
- os: [netbsd]
-
- esbuild-openbsd-64@0.15.16:
- resolution: {integrity: sha512-KgxMHyxMCT+NdLQE1zVJEsLSt2QQBAvJfmUGDmgEq8Fvjrf6vSKB00dVHUEDKcJwMID6CdgCpvYNt999tIYhqA==}
- engines: {node: '>=12'}
- cpu: [x64]
- os: [openbsd]
-
- esbuild-sunos-64@0.15.16:
- resolution: {integrity: sha512-exSAx8Phj7QylXHlMfIyEfNrmqnLxFqLxdQF6MBHPdHAjT7fsKaX6XIJn+aQEFiOcE4X8e7VvdMCJ+WDZxjSRQ==}
- engines: {node: '>=12'}
- cpu: [x64]
- os: [sunos]
-
- esbuild-windows-32@0.15.16:
- resolution: {integrity: sha512-zQgWpY5pUCSTOwqKQ6/vOCJfRssTvxFuEkpB4f2VUGPBpdddZfdj8hbZuFRdZRPIVHvN7juGcpgCA/XCF37mAQ==}
- engines: {node: '>=12'}
- cpu: [ia32]
- os: [win32]
-
- esbuild-windows-64@0.15.16:
- resolution: {integrity: sha512-HjW1hHRLSncnM3MBCP7iquatHVJq9l0S2xxsHHj4yzf4nm9TU4Z7k4NkeMlD/dHQ4jPlQQhwcMvwbJiOefSuZw==}
- engines: {node: '>=12'}
- cpu: [x64]
- os: [win32]
-
- esbuild-windows-arm64@0.15.16:
- resolution: {integrity: sha512-oCcUKrJaMn04Vxy9Ekd8x23O8LoU01+4NOkQ2iBToKgnGj5eo1vU9i27NQZ9qC8NFZgnQQZg5oZWAejmbsppNA==}
- engines: {node: '>=12'}
- cpu: [arm64]
- os: [win32]
-
- esbuild@https://registry.npmmirror.com/esbuild/-/esbuild-0.15.16.tgz:
- resolution: {integrity: sha512-o6iS9zxdHrrojjlj6pNGC2NAg86ECZqIETswTM5KmJitq+R1YmahhWtMumeQp9lHqJaROGnsBi2RLawGnfo5ZQ==, tarball: https://registry.npmmirror.com/esbuild/-/esbuild-0.15.16.tgz}
- version: 0.15.16
- engines: {node: '>=12'}
- hasBin: true
-
- escalade@3.1.1:
- resolution: {integrity: sha512-k0er2gUkLf8O0zKJiAhmkTnJlTvINGv7ygDNPbeIsX/TJjGJZHuh9B2UxbsaEkmlEo9MfhrSzmhIlhRlI2GXnw==}
- engines: {node: '>=6'}
-
- escape-html@1.0.3:
- resolution: {integrity: sha512-NiSupZ4OeuGwr68lGIeym/ksIZMJodUGOSCZ/FSnTxcrekbvqrgdUxlJOMpijaKZVjAJrWrGs/6Jy8OMuyj9ow==}
-
- escape-string-regexp@1.0.5:
- resolution: {integrity: sha512-vbRorB5FUQWvla16U8R/qgaFIya2qGzwDrNmCZuYKrbdSUMG6I1ZCGQRefkRVhuOkIGVne7BQ35DSfo1qvJqFg==}
- engines: {node: '>=0.8.0'}
-
- esprima@4.0.1:
- resolution: {integrity: sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A==}
- engines: {node: '>=4'}
- hasBin: true
-
- esprima@https://registry.npmmirror.com/esprima/-/esprima-4.0.1.tgz:
- resolution: {integrity: sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A==, tarball: https://registry.npmmirror.com/esprima/-/esprima-4.0.1.tgz}
- version: 4.0.1
- engines: {node: '>=4'}
- hasBin: true
-
- estree-walker@2.0.2:
- resolution: {integrity: sha512-Rfkk/Mp/DL7JVje3u18FxFujQlTNR2q6QfMSMB7AvCBx91NGj/ba3kCfza0f6dVDbw7YlRf/nDrn7pQrCCyQ/w==}
-
- estree-walker@3.0.3:
- resolution: {integrity: sha512-7RUKfXgSMMkzt6ZuXmqapOurLGPPfgj6l9uRZ7lRGolvk0y2yocc35LdcxKC5PQZdn2DMqioAQ2NoWcrTKmm6g==}
-
- estree-walker@https://registry.npmmirror.com/estree-walker/-/estree-walker-2.0.2.tgz:
- resolution: {integrity: sha512-Rfkk/Mp/DL7JVje3u18FxFujQlTNR2q6QfMSMB7AvCBx91NGj/ba3kCfza0f6dVDbw7YlRf/nDrn7pQrCCyQ/w==, tarball: https://registry.npmmirror.com/estree-walker/-/estree-walker-2.0.2.tgz}
- version: 2.0.2
-
- execa@5.1.1:
- resolution: {integrity: sha512-8uSpZZocAZRBAPIEINJj3Lo9HyGitllczc27Eh5YYojjMFMn8yHMDMaUHE2Jqfq05D/wucwI4JGURyXt1vchyg==}
- engines: {node: '>=10'}
-
- fast-deep-equal@3.1.3:
- resolution: {integrity: sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q==}
-
- fast-glob@https://registry.npmmirror.com/fast-glob/-/fast-glob-3.2.12.tgz:
- resolution: {integrity: sha512-DVj4CQIYYow0BlaelwK1pHl5n5cRSJfM60UA0zK891sVInoPri2Ekj7+e1CT3/3qxXenpI+nBBmQAcJPJgaj4w==, tarball: https://registry.npmmirror.com/fast-glob/-/fast-glob-3.2.12.tgz}
- version: 3.2.12
- engines: {node: '>=8.6.0'}
-
- fast-uri@3.0.3:
- resolution: {integrity: sha512-aLrHthzCjH5He4Z2H9YZ+v6Ujb9ocRuW6ZzkJQOrTxleEijANq4v1TsaPaVG1PZcuurEzrLcWRyYBYXD5cEiaw==}
-
- fastq@https://registry.npmmirror.com/fastq/-/fastq-1.14.0.tgz:
- resolution: {integrity: sha512-eR2D+V9/ExcbF9ls441yIuN6TI2ED1Y2ZcA5BmMtJsOkWOFRJQ0Jt0g1UwqXJJVAb+V+umH5Dfr8oh4EVP7VVg==, tarball: https://registry.npmmirror.com/fastq/-/fastq-1.14.0.tgz}
- version: 1.14.0
-
- fflate@0.8.2:
- resolution: {integrity: sha512-cPJU47OaAoCbg0pBvzsgpTPhmhqI5eJjh/JIu8tPj5q+T7iLvW/JAYUqmE7KOB4R1ZyEhzBaIQpQpardBF5z8A==}
-
- file-selector@0.2.4:
- resolution: {integrity: sha512-ZDsQNbrv6qRi1YTDOEWzf5J2KjZ9KMI1Q2SGeTkCJmNNW25Jg4TW4UMcmoqcg4WrAyKRcpBXdbWRxkfrOzVRbA==}
- engines: {node: '>= 10'}
-
- fill-range@7.0.1:
- resolution: {integrity: sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ==}
- engines: {node: '>=8'}
-
- fill-range@https://registry.npmmirror.com/fill-range/-/fill-range-7.0.1.tgz:
- resolution: {integrity: sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ==, tarball: https://registry.npmmirror.com/fill-range/-/fill-range-7.0.1.tgz}
- version: 7.0.1
- engines: {node: '>=8'}
-
- flatten-vertex-data@1.0.2:
- resolution: {integrity: sha512-BvCBFK2NZqerFTdMDgqfHBwxYWnxeCkwONsw6PvBMcUXqo8U/KDWwmXhqx1x2kLIg7DqIsJfOaJFOmlua3Lxuw==}
-
- follow-redirects@1.15.9:
- resolution: {integrity: sha512-gew4GsXizNgdoRyqmyfMHyAmXsZDk6mHkSxZFCzW9gwlbtOW44CDtYavM+y+72qD/Vq2l550kMF52DT8fOLJqQ==}
- engines: {node: '>=4.0'}
- peerDependencies:
- debug: '*'
- peerDependenciesMeta:
- debug:
- optional: true
-
- font-awesome@4.7.0:
- resolution: {integrity: sha512-U6kGnykA/6bFmg1M/oT9EkFeIYv7JlX3bozwQJWiiLz6L0w3F5vBVPxHlwyX/vtNq1ckcpRKOB9f2Qal/VtFpg==}
- engines: {node: '>=0.10.3'}
-
- form-data@4.0.1:
- resolution: {integrity: sha512-tzN8e4TX8+kkxGPK8D5u0FNmjPUjw3lwC9lSLxxoB/+GtsJG91CO8bSWy73APlgAZzZbXEYZJuxjkHH2w+Ezhw==}
- engines: {node: '>= 6'}
-
- fs-minipass@2.1.0:
- resolution: {integrity: sha512-V/JgOLFCS+R6Vcq0slCuaeWEdNC3ouDlJMNIsacH2VtALiu9mV4LPrHc5cDl8k5aw6J8jwgWWpiTo5RYhmIzvg==}
- engines: {node: '>= 8'}
-
- fs.realpath@1.0.0:
- resolution: {integrity: sha512-OO0pH2lK6a0hZnAdau5ItzHPI6pUlvI7jMVnxUQRtw4owF2wk8lOSabtGDCTP4Ggrg2MbGnWO9X8K1t4+fGMDw==}
-
- fsevents@2.3.2:
- resolution: {integrity: sha512-xiqMQR4xAeHTuB9uWm+fFRcIOgKBMiOBP+eXiyT7jsgVCq1bkVygt00oASowB7EdtpOHaaPgKt812P9ab+DDKA==}
- engines: {node: ^8.16.0 || ^10.6.0 || >=11.0.0}
- os: [darwin]
-
- function-bind@https://registry.npmmirror.com/function-bind/-/function-bind-1.1.1.tgz:
- resolution: {integrity: sha512-yIovAzMX49sF8Yl58fSCWJ5svSLuaibPxXQJFLmBObTuCr0Mf1KiPopGM9NiFjiYBCbfaa2Fh6breQ6ANVTI0A==, tarball: https://registry.npmmirror.com/function-bind/-/function-bind-1.1.1.tgz}
- version: 1.1.1
-
- gauge@3.0.2:
- resolution: {integrity: sha512-+5J6MS/5XksCuXq++uFRsnUd7Ovu1XenbeuIuNRJxYWjgQbPuFhT14lAvsWfqfAmnwluf1OwMjz39HjfLPci0Q==}
- engines: {node: '>=10'}
- deprecated: This package is no longer supported.
-
- get-caller-file@2.0.5:
- resolution: {integrity: sha512-DyFP3BM/3YHTQOCUL/w0OZHR0lpKeGrxotcHWcqNEdnltqFwXVfhEBQ94eIo34AfQpo0rGki4cyIiftY06h2Fg==}
- engines: {node: 6.* || 8.* || >= 10.*}
-
- get-port@3.2.0:
- resolution: {integrity: sha512-x5UJKlgeUiNT8nyo/AcnwLnZuZNcSjSw0kogRB+Whd1fjjFq4B1hySFxSFWWSn4mIBzg3sRNUDFYc4g5gjPoLg==}
- engines: {node: '>=4'}
-
- get-stream@6.0.1:
- resolution: {integrity: sha512-ts6Wi+2j3jQjqi70w5AlN8DFnkSwC+MqmxEzdEALB2qXZYV3X/b1CTfgPLGJNMeAWxdPfU8FO1ms3NUfaHCPYg==}
- engines: {node: '>=10'}
-
- glob-parent@https://registry.npmmirror.com/glob-parent/-/glob-parent-5.1.2.tgz:
- resolution: {integrity: sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==, tarball: https://registry.npmmirror.com/glob-parent/-/glob-parent-5.1.2.tgz}
- version: 5.1.2
- engines: {node: '>= 6'}
-
- glob@7.2.3:
- resolution: {integrity: sha512-nFR0zLpU2YCaRxwoCJvL6UvCH2JFyFVIvwTLsIf21AuHlMskA1hhTdk+LlYJtOlYt9v6dvszD2BGRqBL+iQK9Q==}
- deprecated: Glob versions prior to v9 are no longer supported
-
- good-listener@1.2.2:
- resolution: {integrity: sha512-goW1b+d9q/HIwbVYZzZ6SsTr4IgE+WA44A0GmPIQstuOrgsFcT7VEJ48nmr9GaRtNu0XTKacFLGnBPAM6Afouw==}
-
- has-flag@3.0.0:
- resolution: {integrity: sha512-sKJf1+ceQBr4SMkvQnBDNDtf4TXpVhVGateu0t918bl30FnbE2m4vNLX+VWe/dpjlb+HugGYzW7uQXH98HPEYw==}
- engines: {node: '>=4'}
-
- has-own-prop@https://registry.npmmirror.com/has-own-prop/-/has-own-prop-2.0.0.tgz:
- resolution: {integrity: sha512-Pq0h+hvsVm6dDEa8x82GnLSYHOzNDt7f0ddFa3FqcQlgzEiptPqL+XrOJNavjOzSYiYWIrgeVYYgGlLmnxwilQ==, tarball: https://registry.npmmirror.com/has-own-prop/-/has-own-prop-2.0.0.tgz}
- version: 2.0.0
- engines: {node: '>=8'}
-
- has-unicode@2.0.1:
- resolution: {integrity: sha512-8Rf9Y83NBReMnx0gFzA8JImQACstCYWUplepDa9xprwwtmgEZUF0h/i5xSA625zB/I37EtrswSST6OXxwaaIJQ==}
-
- has@https://registry.npmmirror.com/has/-/has-1.0.3.tgz:
- resolution: {integrity: sha512-f2dvO0VU6Oej7RkWJGrehjbzMAjFp5/VKPp5tTpWIV4JHHZK1/BxbFRtf/siA2SWTe09caDmVtYYzWEIbBS4zw==, tarball: https://registry.npmmirror.com/has/-/has-1.0.3.tgz}
- version: 1.0.3
- engines: {node: '>= 0.4.0'}
-
- he@https://registry.npmmirror.com/he/-/he-1.2.0.tgz:
- resolution: {integrity: sha512-F/1DnUGPopORZi0ni+CvrCgHQ5FyEAHRLSApuYWMmrbSwoN2Mn/7k+Gl38gJnR7yyDZk6WLXwiGod1JOWNDKGw==, tarball: https://registry.npmmirror.com/he/-/he-1.2.0.tgz}
- version: 1.2.0
- hasBin: true
-
- https-proxy-agent@5.0.1:
- resolution: {integrity: sha512-dFcAjpTQFgoLMzC2VwU+C/CbS7uRL0lWmxDITmqm7C+7F0Odmj6s9l6alZc6AELXhrnggM2CeWSXHGOdX2YtwA==}
- engines: {node: '>= 6'}
-
- human-signals@2.1.0:
- resolution: {integrity: sha512-B4FFZ6q/T2jhhksgkbEW3HBvWIfDW85snkQgawt07S7J5QXTk6BkNV+0yAeZrM5QpMAdYlocGoljn0sJ/WQkFw==}
- engines: {node: '>=10.17.0'}
-
- humanize-duration@3.32.1:
- resolution: {integrity: sha512-inh5wue5XdfObhu/IGEMiA1nUXigSGcaKNemcbLRKa7jXYGDZXr3LoT9pTIzq2hPEbld7w/qv9h+ikWGz8fL1g==}
-
- immutable-json-patch@6.0.1:
- resolution: {integrity: sha512-BHL/cXMjwFZlTOffiWNdY8ZTvNyYLrutCnWxrcKPHr5FqpAb6vsO6WWSPnVSys3+DruFN6lhHJJPHi8uELQL5g==}
-
- immutable@4.3.7:
- resolution: {integrity: sha512-1hqclzwYwjRDFLjcFxOM5AYkkG0rpFPpr1RLPMEuGczoS7YA8gLhy8SWXYRAA/XwfEHpfo3cw5JGioS32fnMRw==}
-
- inflight@1.0.6:
- resolution: {integrity: sha512-k92I/b08q4wvFscXCLvqfsHCrjrF7yiXsQuIVvVE7N82W3+aqpzuUdBbfhWcy/FZR3/4IgflMgKLOsvPDrGCJA==}
- deprecated: This module is not supported, and leaks memory. Do not use it. Check out lru-cache if you want a good and tested way to coalesce async requests by a key value, which is much more comprehensive and powerful.
-
- inherits@2.0.4:
- resolution: {integrity: sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==}
-
- ip-regex@4.3.0:
- resolution: {integrity: sha512-B9ZWJxHHOHUhUjCPrMpLD4xEq35bUTClHM1S6CBU5ixQnkZmwipwgc96vAd7AAGM9TGHvJR+Uss+/Ak6UphK+Q==}
- engines: {node: '>=8'}
-
- is-binary-path@https://registry.npmmirror.com/is-binary-path/-/is-binary-path-2.1.0.tgz:
- resolution: {integrity: sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==, tarball: https://registry.npmmirror.com/is-binary-path/-/is-binary-path-2.1.0.tgz}
- version: 2.1.0
- engines: {node: '>=8'}
-
- is-buffer@1.1.6:
- resolution: {integrity: sha512-NcdALwpXkTm5Zvvbk7owOUSvVvBKDgKP5/ewfXEznmQFfs4ZRmanOeKBTjRVjka3QFoN6XJ+9F3USqfHqTaU5w==}
-
- is-core-module@https://registry.npmmirror.com/is-core-module/-/is-core-module-2.11.0.tgz:
- resolution: {integrity: sha512-RRjxlvLDkD1YJwDbroBHMb+cukurkDWNyHx7D3oNB5x9rb5ogcksMC5wHCadcXoo67gVr/+3GFySh3134zi6rw==, tarball: https://registry.npmmirror.com/is-core-module/-/is-core-module-2.11.0.tgz}
- version: 2.11.0
-
- is-docker@2.2.1:
- resolution: {integrity: sha512-F+i2BKsFrH66iaUFc0woD8sLy8getkwTwtOBjvs56Cx4CgJDeKQeqfz8wAYiSb8JOprWhHH5p77PbmYCvvUuXQ==}
- engines: {node: '>=8'}
- hasBin: true
-
- is-extglob@2.1.1:
- resolution: {integrity: sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ==}
- engines: {node: '>=0.10.0'}
-
- is-extglob@https://registry.npmmirror.com/is-extglob/-/is-extglob-2.1.1.tgz:
- resolution: {integrity: sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ==, tarball: https://registry.npmmirror.com/is-extglob/-/is-extglob-2.1.1.tgz}
- version: 2.1.1
- engines: {node: '>=0.10.0'}
-
- is-fullwidth-code-point@3.0.0:
- resolution: {integrity: sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==}
- engines: {node: '>=8'}
-
- is-glob@4.0.3:
- resolution: {integrity: sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==}
- engines: {node: '>=0.10.0'}
-
- is-glob@https://registry.npmmirror.com/is-glob/-/is-glob-4.0.3.tgz:
- resolution: {integrity: sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==, tarball: https://registry.npmmirror.com/is-glob/-/is-glob-4.0.3.tgz}
- version: 4.0.3
- engines: {node: '>=0.10.0'}
-
- is-number@7.0.0:
- resolution: {integrity: sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==}
- engines: {node: '>=0.12.0'}
-
- is-number@https://registry.npmmirror.com/is-number/-/is-number-7.0.0.tgz:
- resolution: {integrity: sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==, tarball: https://registry.npmmirror.com/is-number/-/is-number-7.0.0.tgz}
- version: 7.0.0
- engines: {node: '>=0.12.0'}
-
- is-obj@1.0.1:
- resolution: {integrity: sha512-l4RyHgRqGN4Y3+9JHVrNqO+tN0rV5My76uW5/nuO4K1b6vw5G8d/cmFjP9tRfEsdhZNt0IFdZuK/c2Vr4Nb+Qg==}
- engines: {node: '>=0.10.0'}
-
- is-reference@3.0.1:
- resolution: {integrity: sha512-baJJdQLiYaJdvFbJqXrcGv3WU3QCzBlUcI5QhbesIm6/xPsvmO+2CDoi/GMOFBQEQm+PXkwOPrp9KK5ozZsp2w==}
-
- is-stream@2.0.1:
- resolution: {integrity: sha512-hFoiJiTl63nn+kstHGBtewWSKnQLpyb155KHheA1l39uvtO9nWIop1p3udqPcUd/xbF1VLMO4n7OI6p7RbngDg==}
- engines: {node: '>=8'}
-
- is-wsl@2.2.0:
- resolution: {integrity: sha512-fKzAra0rGJUUBwGBgNkHZuToZcn+TtXHpeCgmkMJMMYx1sQDYaCSyjJBSCa2nH1DGm7s3n1oBnohoVTBaN7Lww==}
- engines: {node: '>=8'}
-
- isexe@2.0.0:
- resolution: {integrity: sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw==}
-
- javascript-time-ago@2.5.11:
- resolution: {integrity: sha512-Zeyf5R7oM1fSMW9zsU3YgAYwE0bimEeF54Udn2ixGd8PUwu+z1Yc5t4Y8YScJDMHD6uCx6giLt3VJR5K4CMwbg==}
-
- jmespath@0.16.0:
- resolution: {integrity: sha512-9FzQjJ7MATs1tSpnco1K6ayiYE3figslrXA72G2HQ/n76RzvYlofyi5QM+iX4YRs/pu3yzxlVQSST23+dMDknw==}
- engines: {node: '>= 0.6.0'}
-
- js-yaml@3.14.1:
- resolution: {integrity: sha512-okMH7OXXJ7YrN9Ok3/SXrnu4iX9yOk+25nqX4imS2npuvTYDmo/QEZoqwZkYaIDk3jVvBOTOIEgEhaLOynBS9g==}
- hasBin: true
-
- jsep@1.3.9:
- resolution: {integrity: sha512-i1rBX5N7VPl0eYb6+mHNp52sEuaS2Wi8CDYx1X5sn9naevL78+265XJqy1qENEk7mRKwS06NHpUqiBwR7qeodw==}
- engines: {node: '>= 10.16.0'}
-
- jsesc@0.5.0:
- resolution: {integrity: sha512-uZz5UnB7u4T9LvwmFqXii7pZSouaRPorGs5who1Ip7VO0wxanFvBL7GkM6dTHlgX+jhBApRetaWpnDabOeTcnA==}
- hasBin: true
-
- json-editor-vue@0.17.2:
- resolution: {integrity: sha512-P38IvEwQJe4IHFpZe1YG2CWOK6pCjeHmPu2tLs01IiV0kz8QBoHYC7N5ZORlGTL4tQFVOosmnI1gjXztywHtig==}
- version: 0.17.2
- peerDependencies:
- '@vue/composition-api': '>=1'
- vue: 2||3
- peerDependenciesMeta:
- '@vue/composition-api':
- optional: true
-
- json-schema-traverse@1.0.0:
- resolution: {integrity: sha512-NM8/P9n3XjXhIZn1lLhkFaACTOURQXjWhV4BA/RnOv8xvgqtqpAX9IO4mRQxSx1Rlo4tqzeqb0sOlruaOy3dug==}
-
- json-source-map@0.6.1:
- resolution: {integrity: sha512-1QoztHPsMQqhDq0hlXY5ZqcEdUzxQEIxgFkKl4WUp2pgShObl+9ovi4kRh2TfvAfxAoHOJ9vIMEqk3k4iex7tg==}
-
- jsonpath-plus@10.0.1:
- resolution: {integrity: sha512-30DeH2QD4nL1IpDLPIFz09G5XyLvh+oNMUI2Zxf4tbrlsVHs0e3VPnwpOnSTFb4yM0dfQK2WGKLsSaAS8V62rw==}
- engines: {node: '>=18.0.0'}
- hasBin: true
-
- jsonrepair@3.8.1:
- resolution: {integrity: sha512-5wnjaO53EJOhfLFY92nvBz2B9gqF9ql/D4HKUb1WOSBaqtVcAifFfmurblnhCJn/ySqKFA8U3n7nhGMAu/hEjQ==}
- hasBin: true
-
- keypress@0.1.0:
- resolution: {integrity: sha512-x0yf9PL/nx9Nw9oLL8ZVErFAk85/lslwEP7Vz7s5SI1ODXZIgit3C5qyWjw4DxOuO/3Hb4866SQh28a1V1d+WA==}
-
- kleur@3.0.3:
- resolution: {integrity: sha512-eTIzlVOSUR+JxdDFepEYcBMtZ9Qqdef+rnzWdRZuMbOywu5tO2w2N7rqjoANZ5k9vywhL6Br1VRjUIgTQx4E8w==}
- engines: {node: '>=6'}
-
- lexical@0.16.1:
- resolution: {integrity: sha512-+R05d3+N945OY8pTUjTqQrWoApjC+ctzvjnmNETtx9WmVAaiW0tQVG+AYLt5pDGY8dQXtd4RPorvnxBTECt9SA==}
-
- local-access@1.1.0:
- resolution: {integrity: sha512-XfegD5pyTAfb+GY6chk283Ox5z8WexG56OvM06RWLpAc/UHozO8X6xAxEkIitZOtsSMM1Yr3DkHgW5W+onLhCw==}
- engines: {node: '>=6'}
-
- locate-character@3.0.0:
- resolution: {integrity: sha512-SW13ws7BjaeJ6p7Q6CO2nchbYEc3X3J6WrmTTDto7yMPqVSZTUyY5Tjbid+Ab8gLnATtygYtiDIJGQRRn2ZOiA==}
-
- lodash-es@4.17.21:
- resolution: {integrity: sha512-mKnC+QJ9pWVzv+C4/U3rRsHapFfHvQFoFB92e52xeyGMcX6/OlIl78je1u8vePzYZSkkogMPJ2yjxxsb89cxyw==}
-
- lodash-unified@1.0.3:
- resolution: {integrity: sha512-WK9qSozxXOD7ZJQlpSqOT+om2ZfcT4yO+03FuzAHD0wF6S0l0090LRPDx3vhTTLZ8cFKpBn+IOcVXK6qOcIlfQ==}
- peerDependencies:
- '@types/lodash-es': '*'
- lodash: '*'
- lodash-es: '*'
-
- lodash._reinterpolate@3.0.0:
- resolution: {integrity: sha512-xYHt68QRoYGjeeM/XOE1uJtvXQAgvszfBhjV4yvsQH0u2i9I6cI6c6/eG4Hh3UAOVn0y/xAXwmTzEay49Q//HA==}
-
- lodash.template@4.5.0:
- resolution: {integrity: sha512-84vYFxIkmidUiFxidA/KjjH9pAycqW+h980j7Fuz5qxRtO9pgB7MDFTdys1N7A5mcucRiDyEq4fusljItR1T/A==}
-
- lodash.templatesettings@4.2.0:
- resolution: {integrity: sha512-stgLz+i3Aa9mZgnjr/O+v9ruKZsPsndy7qPZOchbqk2cnTU1ZaldKK+v7m54WoKIyxiuMZTKT2H81F8BeAc3ZQ==}
-
- lodash@4.17.21:
- resolution: {integrity: sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg==}
-
- lru-cache@6.0.0:
- resolution: {integrity: sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA==}
- engines: {node: '>=10'}
-
- luxon@3.5.0:
- resolution: {integrity: sha512-rh+Zjr6DNfUYR3bPwJEnuwDdqMbxZW7LOQfUN4B54+Cl+0o5zaU9RJ6bcidfDtC1cWCZXQ+nvX8bf6bAji37QQ==}
- engines: {node: '>=12'}
-
- macos-release@3.1.0:
- resolution: {integrity: sha512-/M/R0gCDgM+Cv1IuBG1XGdfTFnMEG6PZeT+KGWHO/OG+imqmaD9CH5vHBTycEM3+Kc4uG2Il+tFAuUWLqQOeUA==}
- engines: {node: ^12.20.0 || ^14.13.1 || >=16.0.0}
-
- magic-string@0.25.9:
- resolution: {integrity: sha512-RmF0AsMzgt25qzqqLc1+MbHmhdx0ojF2Fvs4XnOqz2ZOBXzzkEwc/dJQZCYHAn7v1jbVOjAZfK8msRn4BxO4VQ==}
-
- magic-string@0.30.1:
- resolution: {integrity: sha512-mbVKXPmS0z0G4XqFDCTllmDQ6coZzn94aMlb0o/A4HEHJCKcanlDZwYJgwnkmgD3jyWhUgj9VsPrfd972yPffA==}
- engines: {node: '>=12'}
-
- magic-string@0.30.12:
- resolution: {integrity: sha512-Ea8I3sQMVXr8JhN4z+H/d8zwo+tYDgHE9+5G4Wnrwhs0gaK9fXTKx0Tw5Xwsd/bCPTTZNRAdpyzvoeORe9LYpw==}
-
- magic-string@https://registry.npmmirror.com/magic-string/-/magic-string-0.25.9.tgz:
- resolution: {integrity: sha512-RmF0AsMzgt25qzqqLc1+MbHmhdx0ojF2Fvs4XnOqz2ZOBXzzkEwc/dJQZCYHAn7v1jbVOjAZfK8msRn4BxO4VQ==, tarball: https://registry.npmmirror.com/magic-string/-/magic-string-0.25.9.tgz}
- version: 0.25.9
-
- make-dir@3.1.0:
- resolution: {integrity: sha512-g3FeP20LNwhALb/6Cz6Dd4F2ngze0jz7tbzrD2wAV+o9FeNHe4rL+yK2md0J/fiSf1sa1ADhXqi5+oVwOM/eGw==}
- engines: {node: '>=8'}
-
- matchit@1.1.0:
- resolution: {integrity: sha512-+nGYoOlfHmxe5BW5tE0EMJppXEwdSf8uBA1GTZC7Q77kbT35+VKLYJMzVNWCHSsga1ps1tPYFtFyvxvKzWVmMA==}
- engines: {node: '>=6'}
-
- math-log2@1.0.1:
- resolution: {integrity: sha512-9W0yGtkaMAkf74XGYVy4Dqw3YUMnTNB2eeiw9aQbUl4A3KmuCEHTt2DgAB07ENzOYAjsYSAYufkAq0Zd+jU7zA==}
- engines: {node: '>=0.10.0'}
-
- md5@2.3.0:
- resolution: {integrity: sha512-T1GITYmFaKuO91vxyoQMFETst+O71VUPEU3ze5GNzDm0OWdP8v1ziTaAEPUr/3kLsY3Sftgz242A1SetQiDL7g==}
-
- mdn-data@2.0.28:
- resolution: {integrity: sha512-aylIc7Z9y4yzHYAJNuESG3hfhC+0Ibp/MAMiaOZgNv4pmEdFyfZhhhny4MNiAfWdBQ1RQ2mfDWmM1x8SvGyp8g==}
-
- mdn-data@2.0.30:
- resolution: {integrity: sha512-GaqWWShW4kv/G9IEucWScBx9G1/vsFZZJUO+tD26M8J8z3Kw5RDQjaoZe03YAClgeS/SWPOcb4nkFBTEi5DUEA==}
-
- memoize-one@6.0.0:
- resolution: {integrity: sha512-rkpe71W0N0c0Xz6QD0eJETuWAJGnJ9afsl1srmwPrI+yBCkge5EycXXbYRyvL29zZVUWQCY7InPRCv3GDXuZNw==}
-
- merge-stream@2.0.0:
- resolution: {integrity: sha512-abv/qOcuPfk3URPfDzmZU1LKmuw8kT+0nIHvKrKgFrwifol/doWcdA4ZqsWQ8ENrFKkd67Mfpo/LovbIUsbt3w==}
-
- merge2@https://registry.npmmirror.com/merge2/-/merge2-1.4.1.tgz:
- resolution: {integrity: sha512-8q7VEgMJW4J8tcfVPy8g09NcQwZdbwFEqhe/WZkoIzjn/3TGDwtOCYtXGxA3O8tPzpczCCDgv+P2P5y00ZJOOg==, tarball: https://registry.npmmirror.com/merge2/-/merge2-1.4.1.tgz}
- version: 1.4.1
- engines: {node: '>= 8'}
-
- meshoptimizer@0.18.1:
- resolution: {integrity: sha512-ZhoIoL7TNV4s5B6+rx5mC//fw8/POGyNxS/DZyCJeiZ12ScLfVwRE/GfsxwiTkMYYD5DmK2/JXnEVXqL4rF+Sw==}
-
- micromatch@4.0.5:
- resolution: {integrity: sha512-DMy+ERcEW2q8Z2Po+WNXuw3c5YaUSFjAO5GsJqfEl7UjvtIuFKO6ZrKvcItdy98dwFI2N1tg3zNIdKaQT+aNdA==}
- engines: {node: '>=8.6'}
-
- micromatch@https://registry.npmmirror.com/micromatch/-/micromatch-4.0.5.tgz:
- resolution: {integrity: sha512-DMy+ERcEW2q8Z2Po+WNXuw3c5YaUSFjAO5GsJqfEl7UjvtIuFKO6ZrKvcItdy98dwFI2N1tg3zNIdKaQT+aNdA==, tarball: https://registry.npmmirror.com/micromatch/-/micromatch-4.0.5.tgz}
- version: 4.0.5
- engines: {node: '>=8.6'}
-
- mime-db@1.52.0:
- resolution: {integrity: sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==}
- engines: {node: '>= 0.6'}
-
- mime-types@2.1.35:
- resolution: {integrity: sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==}
- engines: {node: '>= 0.6'}
-
- mimic-fn@2.1.0:
- resolution: {integrity: sha512-OqbOk5oEQeAZ8WXWydlu9HJjz9WVdEIvamMCcXmuqUYjTknH/sqsWvhQ3vgwKFRR1HpjvNBKQ37nbJgYzGqGcg==}
- engines: {node: '>=6'}
-
- minimatch@3.1.2:
- resolution: {integrity: sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==}
-
- minimatch@https://registry.npmmirror.com/minimatch/-/minimatch-5.1.1.tgz:
- resolution: {integrity: sha512-362NP+zlprccbEt/SkxKfRMHnNY85V74mVnpUpNyr3F35covl09Kec7/sEFLt3RA4oXmewtoaanoIf67SE5Y5g==, tarball: https://registry.npmmirror.com/minimatch/-/minimatch-5.1.1.tgz}
- version: 5.1.1
- engines: {node: '>=10'}
-
- minimist@1.2.7:
- resolution: {integrity: sha512-bzfL1YUZsP41gmu/qjrEk0Q6i2ix/cVeAhbCbqH9u3zYutS1cLg00qhrD0M2MVdCcx4Sc0UpP2eBWo9rotpq6g==}
-
- minipass@3.3.6:
- resolution: {integrity: sha512-DxiNidxSEK+tHG6zOIklvNOwm3hvCrbUrdtzY74U6HKTJxvIDfOUL5W5P2Ghd3DTkhhKPYGqeNUIh5qcM4YBfw==}
- engines: {node: '>=8'}
-
- minizlib@2.1.2:
- resolution: {integrity: sha512-bAxsR8BVfj60DWXHE3u30oHzfl4G7khkSuPW+qvpd7jFRHm7dLxOjUk1EHACJ/hxLY8phGJ0YhYHZo7jil7Qdg==}
- engines: {node: '>= 8'}
-
- mitt@3.0.1:
- resolution: {integrity: sha512-vKivATfr97l2/QBCYAkXYDbrIWPM2IIKEl7YPhjCvKlG3kE2gm+uBo6nEXK3M5/Ffh/FLpKExzOQ3JJoJGFKBw==}
-
- mkdirp@1.0.4:
- resolution: {integrity: sha512-vVqVZQyf3WLx2Shd0qJ9xuvqgAyKPLAiqITEtqW0oIUjzo3PePDd6fW9iFz30ef7Ysp/oiWqbhszeGWW2T6Gzw==}
- engines: {node: '>=10'}
- hasBin: true
-
- monaco-editor@0.50.0:
- resolution: {integrity: sha512-8CclLCmrRRh+sul7C08BmPBP3P8wVWfBHomsTcndxg5NRCEPfu/mc2AGU8k37ajjDVXcXFc12ORAMUkmk+lkFA==}
-
- mri@1.2.0:
- resolution: {integrity: sha512-tzzskb3bG8LvYGFF/mDTpq3jpI6Q9wc3LEmBaghu+DdCssd1FakN7Bc0hVNmEyGq1bq3RgfkCb3cmQLpNPOroA==}
- engines: {node: '>=4'}
-
- mrmime@1.0.1:
- resolution: {integrity: sha512-hzzEagAgDyoU1Q6yg5uI+AorQgdvMCur3FcKf7NhMKWsaYg+RnbTyHRa/9IlLF9rf455MOCtcqqrQQ83pPP7Uw==}
- engines: {node: '>=10'}
-
- ms@2.0.0:
- resolution: {integrity: sha512-Tpp60P6IUJDTuOq/5Z8cdskzJujfwqfOTkrwIwj7IRISpnkJnT6SyJ4PCPnGMoFjC9ddhal5KVIYtAt97ix05A==}
-
- ms@2.1.2:
- resolution: {integrity: sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==}
-
- muggle-string@https://registry.npmmirror.com/muggle-string/-/muggle-string-0.1.0.tgz:
- resolution: {integrity: sha512-Tr1knR3d2mKvvWthlk7202rywKbiOm4rVFLsfAaSIhJ6dt9o47W4S+JMtWhd/PW9Wrdew2/S2fSvhz3E2gkfEg==, tarball: https://registry.npmmirror.com/muggle-string/-/muggle-string-0.1.0.tgz}
- version: 0.1.0
-
- nanoid@3.3.7:
- resolution: {integrity: sha512-eSRppjcPIatRIMC1U6UngP8XFcz8MQWGQdt1MTBQ7NaAmvXDfvNxbvWV3x2y6CdEUciCSsDHDQZbhYaB8QEo2g==}
- engines: {node: ^10 || ^12 || ^13.7 || ^14 || >=15.0.1}
- hasBin: true
-
- nanoid@https://registry.npmmirror.com/nanoid/-/nanoid-3.3.4.tgz:
- resolution: {integrity: sha512-MqBkQh/OHTS2egovRtLk45wEyNXwF+cokD+1YPf9u5VfJiRdAiRwB2froX5Co9Rh20xs4siNPm8naNotSD6RBw==, tarball: https://registry.npmmirror.com/nanoid/-/nanoid-3.3.4.tgz}
- version: 3.3.4
- engines: {node: ^10 || ^12 || ^13.7 || ^14 || >=15.0.1}
- hasBin: true
-
- natural-compare-lite@1.4.0:
- resolution: {integrity: sha512-Tj+HTDSJJKaZnfiuw+iaF9skdPpTo2GtEly5JHnWV/hfv2Qj/9RKsGISQtLh2ox3l5EAGw487hnBee0sIJ6v2g==}
-
- negotiator@0.6.3:
- resolution: {integrity: sha512-+EUsqGPLsM+j/zdChZjsnX51g4XrHFOIXwfnCVPGlQk/k5giakcKsuxCObBRu6DSm9opw/O6slWbJdghQM4bBg==}
- engines: {node: '>= 0.6'}
-
- node-addon-api@3.2.1:
- resolution: {integrity: sha512-mmcei9JghVNDYydghQmeDX8KoAm0FAiYyIcUt/N4nhyAipB17pllZQDOJD2fotxABnt4Mdz+dKTO7eftLg4d0A==}
-
- node-addon-api@7.1.1:
- resolution: {integrity: sha512-5m3bsyrjFWE1xf7nz7YXdN4udnVtXK6/Yfgn5qnahL6bCkf2yKt4k3nuTKAtT4r3IG8JNR2ncsIMdZuAzJjHQQ==}
-
- node-fetch@2.6.7:
- resolution: {integrity: sha512-ZjMPFEfVx5j+y2yF35Kzx5sF7kDzxuDj6ziH4FFbOp87zKDZNx8yExJIb05OGF4Nlt9IHFIMBkRl41VdvcNdbQ==}
- engines: {node: 4.x || >=6.0.0}
- peerDependencies:
- encoding: ^0.1.0
- peerDependenciesMeta:
- encoding:
- optional: true
-
- nodejieba@2.5.2:
- resolution: {integrity: sha512-ByskJvaBrQ2eV+5M0OeD80S5NKoGaHc9zi3Z/PTKl/95eac2YF8RmWduq9AknLpkQLrLAIcqurrtC6BzjpKwwg==}
- engines: {node: '>= 10.20.0'}
-
- nopt@5.0.0:
- resolution: {integrity: sha512-Tbj67rffqceeLpcRXrT7vKAN8CwfPeIBgM7E6iBkmKLV7bEMwpGgYLGv0jACUsECaa/vuxP0IjEont6umdMgtQ==}
- engines: {node: '>=6'}
- hasBin: true
-
- normalize-path@https://registry.npmmirror.com/normalize-path/-/normalize-path-3.0.0.tgz:
- resolution: {integrity: sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==, tarball: https://registry.npmmirror.com/normalize-path/-/normalize-path-3.0.0.tgz}
- version: 3.0.0
- engines: {node: '>=0.10.0'}
-
- normalize-wheel-es@1.2.0:
- resolution: {integrity: sha512-Wj7+EJQ8mSuXr2iWfnujrimU35R2W4FAErEyTmJoJ7ucwTn2hOUSsRehMb5RSYkxXGTM7Y9QpvPmp++w5ftoJw==}
-
- normalize.css@8.0.1:
- resolution: {integrity: sha512-qizSNPO93t1YUuUhP22btGOo3chcvDFqFaj2TRybP0DMxkHOCTYwp3n34fel4a31ORXy4m1Xq0Gyqpb5m33qIg==}
-
- npm-run-path@4.0.1:
- resolution: {integrity: sha512-S48WzZW777zhNIrn7gxOlISNAqi9ZC/uQFnRdbeIHhZhCA6UqpkOT8T1G7BvfdgP4Er8gF4sUbaS0i7QvIfCWw==}
- engines: {node: '>=8'}
-
- npmlog@5.0.1:
- resolution: {integrity: sha512-AqZtDUWOMKs1G/8lwylVjrdYgqA4d9nu8hc+0gzRxlDb1I10+FHBGMXs6aiQHFdCUUlqH99MUMuLfzWDNDtfxw==}
- deprecated: This package is no longer supported.
-
- nth-check@2.1.1:
- resolution: {integrity: sha512-lqjrjmaOoAnWfMmBPL+XNnynZh2+swxiX3WUE0s4yEHI6m+AwrK2UZOimIRl3X/4QctVqS8AiZjFqyOGrMXb/w==}
-
- object-assign@4.1.1:
- resolution: {integrity: sha512-rJgTQnkUnH1sFw8yT6VSU3zD3sWmu6sZhIseY8VX+GRu3P6F7Fu+JNDoXfklElbLJSnc3FUQHVe4cU5hj+BcUg==}
- engines: {node: '>=0.10.0'}
-
- on-headers@1.0.2:
- resolution: {integrity: sha512-pZAE+FJLoyITytdqK0U5s+FIpjN0JP3OzFi/u8Rx+EV5/W+JTWGXG8xFzevE7AjBfDqHv/8vL8qQsIhHnqRkrA==}
- engines: {node: '>= 0.8'}
-
- once@1.4.0:
- resolution: {integrity: sha512-lNaJgI+2Q5URQBkccEKHTQOPaXdUxnZZElQTZY0MFUAuaEqe1E+Nyvgdz/aIyNi6Z9MzO5dv1H8n58/GELp3+w==}
-
- onetime@5.1.2:
- resolution: {integrity: sha512-kbpaSSGJTWdAY5KPVeMOKXSrPtr8C8C7wodJbcsd51jRnmD+GZu8Y0VoU6Dm5Z4vWr0Ig/1NKuWRKf7j5aaYSg==}
- engines: {node: '>=6'}
-
- open@8.4.2:
- resolution: {integrity: sha512-7x81NCL719oNbsq/3mh+hVrAWmFuEYUqrq/Iw3kUzH8ReypT9QQ0BLoJS7/G9k6N81XjW4qHWtjWwe/9eLy1EQ==}
- engines: {node: '>=12'}
-
- os-homedir@2.0.0:
- resolution: {integrity: sha512-saRNz0DSC5C/I++gFIaJTXoFJMRwiP5zHar5vV3xQ2TkgEw6hDCcU5F272JjUylpiVgBrZNQHnfjkLabTfb92Q==}
- engines: {node: '>=0.10.0'}
- deprecated: This is not needed anymore. Use `require('os').homedir()` instead.
-
- os-name@5.0.1:
- resolution: {integrity: sha512-0EQpaHUHq7olp2/YFUr+0vZi9tMpDTblHGz+Ch5RntKxiRXOAY0JOz1UlxhSjMSksHvkm13eD6elJj3M8Ht/kw==}
- engines: {node: ^12.20.0 || ^14.13.1 || >=16.0.0}
-
- parse-rect@1.2.0:
- resolution: {integrity: sha512-4QZ6KYbnE6RTwg9E0HpLchUM9EZt6DnDxajFZZDSV4p/12ZJEvPO702DZpGvRYEPo00yKDys7jASi+/w7aO8LA==}
-
- path-is-absolute@1.0.1:
- resolution: {integrity: sha512-AVbw3UJ2e9bq64vSaS9Am0fje1Pa8pbGqTTsmXfaIiMpnr5DlDhfJOuLj9Sf95ZPVDAUerDfEk88MPmPe7UCQg==}
- engines: {node: '>=0.10.0'}
-
- path-key@3.1.1:
- resolution: {integrity: sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q==}
- engines: {node: '>=8'}
-
- path-parse@https://registry.npmmirror.com/path-parse/-/path-parse-1.0.7.tgz:
- resolution: {integrity: sha512-LDJzPVEEEPR+y48z93A0Ed0yXb8pAByGWo/k5YYdYgpY2/2EsOsksJrq7lOHxryrVOn1ejG6oAp8ahvOIQD8sw==, tarball: https://registry.npmmirror.com/path-parse/-/path-parse-1.0.7.tgz}
- version: 1.0.7
-
- periscopic@3.1.0:
- resolution: {integrity: sha512-vKiQ8RRtkl9P+r/+oefh25C3fhybptkHKCZSPlcXiJux2tJF55GnEj3BVn4A5gKfq9NWWXXrxkHBwVPUfH0opw==}
-
- pick-by-alias@1.2.0:
- resolution: {integrity: sha512-ESj2+eBxhGrcA1azgHs7lARG5+5iLakc/6nlfbpjcLl00HuuUOIuORhYXN4D1HfvMSKuVtFQjAlnwi1JHEeDIw==}
-
- picocolors@1.1.1:
- resolution: {integrity: sha512-xceH2snhtb5M9liqDsmEw56le376mTZkEX/jEb/RxNFyegNul7eNslCXP9FDj/Lcu0X8KEyMceP2ntpaHrDEVA==}
-
- picocolors@https://registry.npmmirror.com/picocolors/-/picocolors-1.0.0.tgz:
- resolution: {integrity: sha512-1fygroTLlHu66zi26VoTDv8yRgm0Fccecssto+MhsZ0D/DGW2sm8E8AjW7NU5VVTRt5GxbeZ5qBuJr+HyLYkjQ==, tarball: https://registry.npmmirror.com/picocolors/-/picocolors-1.0.0.tgz}
- version: 1.0.0
-
- picomatch@2.3.1:
- resolution: {integrity: sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==}
- engines: {node: '>=8.6'}
-
- picomatch@https://registry.npmmirror.com/picomatch/-/picomatch-2.3.1.tgz:
- resolution: {integrity: sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==, tarball: https://registry.npmmirror.com/picomatch/-/picomatch-2.3.1.tgz}
- version: 2.3.1
- engines: {node: '>=8.6'}
-
- pinyin@2.11.2:
- resolution: {integrity: sha512-tAWDBcowj09j/vLUjty98nVqrbTVNhutf1VcyID4p0sxTFPzRyXw7n7Ic0HQwBdWFIWrrDP8bYiT64gaT6h3gA==}
- hasBin: true
-
- point-cluster@3.1.8:
- resolution: {integrity: sha512-7klIr45dpMeZuqjIK9+qBg3m2IhyZJNJkdqjJFw0Olq75FM8ojrTMjClVUrMjNYRVqtwztxCHH71Fyjhg+YwyQ==}
-
- polka@0.5.2:
- resolution: {integrity: sha512-FVg3vDmCqP80tOrs+OeNlgXYmFppTXdjD5E7I4ET1NjvtNmQrb1/mJibybKkb/d4NA7YWAr1ojxuhpL3FHqdlw==}
-
- postcss@8.4.47:
- resolution: {integrity: sha512-56rxCq7G/XfB4EkXq9Egn5GCqugWvDFjafDOThIdMBsI15iqPqR5r15TfSr1YPYeEI19YeaXMCbY6u88Y76GLQ==}
- engines: {node: ^10 || ^12 || >=14}
-
- postcss@https://registry.npmmirror.com/postcss/-/postcss-8.4.19.tgz:
- resolution: {integrity: sha512-h+pbPsyhlYj6N2ozBmHhHrs9DzGmbaarbLvWipMRO7RLS+v4onj26MPFXA5OBYFxyqYhUJK456SwDcY9H2/zsA==, tarball: https://registry.npmmirror.com/postcss/-/postcss-8.4.19.tgz}
- version: 8.4.19
- engines: {node: ^10 || ^12 || >=14}
-
- prismjs@1.29.0:
- resolution: {integrity: sha512-Kx/1w86q/epKcmte75LNrEoT+lX8pBpavuAbvJWRXar7Hz8jrtF+e3vY751p0R8H9HdArwaCTNDDzHg/ScJK1Q==}
- engines: {node: '>=6'}
-
- proxy-from-env@1.1.0:
- resolution: {integrity: sha512-D+zkORCbA9f1tdWRK0RaCR3GPv50cMxcrz4X8k5LTSUD1Dkw47mKJEZQNunItRTkWwgtaUSo1RVFRIG9ZXiFYg==}
-
- queue-microtask@https://registry.npmmirror.com/queue-microtask/-/queue-microtask-1.2.3.tgz:
- resolution: {integrity: sha512-NuaNSa6flKT5JaSYQzJok04JzTL1CA6aGhv5rfLW3PgqA+M2ChpZQnAC8h8i4ZFkBS8X5RqkDBHA7r4hej3K9A==, tarball: https://registry.npmmirror.com/queue-microtask/-/queue-microtask-1.2.3.tgz}
- version: 1.2.3
-
- readable-stream@3.6.0:
- resolution: {integrity: sha512-BViHy7LKeTz4oNnkcLJ+lVSL6vpiFeX6/d3oSH8zCW7UxP2onchk+vTGB143xuFjHS3deTgkKoXXymXqymiIdA==}
- engines: {node: '>= 6'}
-
- readdirp@4.0.2:
- resolution: {integrity: sha512-yDMz9g+VaZkqBYS/ozoBJwaBhTbZo3UNYQHNRw1D3UFQB8oHB4uS/tAODO+ZLjGWmUbKnIlOWO+aaIiAxrUWHA==}
- engines: {node: '>= 14.16.0'}
-
- readdirp@https://registry.npmmirror.com/readdirp/-/readdirp-3.6.0.tgz:
- resolution: {integrity: sha512-hOS089on8RduqdbhvQ5Z37A0ESjsqz6qnRcffsMU3495FuTdqSm+7bhJ29JvIOsBDEEnan5DPu9t3To9VRlMzA==, tarball: https://registry.npmmirror.com/readdirp/-/readdirp-3.6.0.tgz}
- version: 3.6.0
- engines: {node: '>=8.10.0'}
-
- regenerate-unicode-properties@9.0.0:
- resolution: {integrity: sha512-3E12UeNSPfjrgwjkR81m5J7Aw/T55Tu7nUyZVQYCKEOs+2dkxEY+DpPtZzO4YruuiPb7NkYLVcyJC4+zCbk5pA==}
- engines: {node: '>=4'}
-
- regenerate@1.4.2:
- resolution: {integrity: sha512-zrceR/XhGYU/d/opr2EKO7aRHUeiBI8qjtfHqADTwZd6Szfy16la6kqD0MIUs5z5hx6AaKa+PixpPrR289+I0A==}
-
- regexpu-core@4.8.0:
- resolution: {integrity: sha512-1F6bYsoYiz6is+oz70NWur2Vlh9KWtswuRuzJOfeYUrfPX2o8n74AnUVaOGDbUqVGO9fNHu48/pjJO4sNVwsOg==}
- engines: {node: '>=4'}
-
- regjsgen@0.5.2:
- resolution: {integrity: sha512-OFFT3MfrH90xIW8OOSyUrk6QHD5E9JOTeGodiJeBS3J6IwlgzJMNE/1bZklWz5oTg+9dCMyEetclvCVXOPoN3A==}
-
- regjsparser@0.7.0:
- resolution: {integrity: sha512-A4pcaORqmNMDVwUjWoTzuhwMGpP+NykpfqAsEgI1FSH/EzC7lrN5TMd+kN8YCovX+jMpu8eaqXgXPCa0g8FQNQ==}
- hasBin: true
-
- relative-time-format@1.1.6:
- resolution: {integrity: sha512-aCv3juQw4hT1/P/OrVltKWLlp15eW1GRcwP1XdxHrPdZE9MtgqFpegjnTjLhi2m2WI9MT/hQQtE+tjEWG1hgkQ==}
-
- repeat-string@https://registry.npmmirror.com/repeat-string/-/repeat-string-1.6.1.tgz:
- resolution: {integrity: sha512-PV0dzCYDNfRi1jCDbJzpW7jNNDRuCOG/jI5ctQcGKt/clZD+YcPS3yIlWuTJMmESC8aevCFmWJy5wjAFgNqN6w==, tarball: https://registry.npmmirror.com/repeat-string/-/repeat-string-1.6.1.tgz}
- version: 1.6.1
- engines: {node: '>=0.10'}
-
- require-directory@2.1.1:
- resolution: {integrity: sha512-fGxEI7+wsG9xrvdjsrlmL22OMTTiHRwAMroiEeMgq8gzoLC/PQr7RsRDSTLUg/bZAZtF+TVIkHc6/4RIKrui+Q==}
- engines: {node: '>=0.10.0'}
-
- require-from-string@2.0.2:
- resolution: {integrity: sha512-Xf0nWe6RseziFMu+Ap9biiUbmplq6S9/p+7w7YXP/JBHhrUDDUhwa+vANyubuqfZWTveU//DYVGsDG7RKL/vEw==}
- engines: {node: '>=0.10.0'}
-
- resolve@https://registry.npmmirror.com/resolve/-/resolve-1.22.1.tgz:
- resolution: {integrity: sha512-nBpuuYuY5jFsli/JIs1oldw6fOQCBioohqWZg/2hiaOybXOft4lonv85uDOKXdf8rhyK159cxU5cDcK/NKk8zw==, tarball: https://registry.npmmirror.com/resolve/-/resolve-1.22.1.tgz}
- version: 1.22.1
- hasBin: true
-
- reusify@https://registry.npmmirror.com/reusify/-/reusify-1.0.4.tgz:
- resolution: {integrity: sha512-U9nH88a3fc/ekCF1l0/UP1IosiuIjyTh7hBvXVMHYgVcfGvt897Xguj2UOLDeI5BG2m7/uwyaLVT6fbtCwTyzw==, tarball: https://registry.npmmirror.com/reusify/-/reusify-1.0.4.tgz}
- version: 1.0.4
- engines: {iojs: '>=1.0.0', node: '>=0.10.0'}
-
- rimraf@3.0.2:
- resolution: {integrity: sha512-JZkJMZkAGFFPP2YqXZXPbMlMBgsxzE8ILs4lMIX/2o0L9UBw9O/Y3o6wFw/i9YLapcUJWwqbi3kdxIPdC62TIA==}
- deprecated: Rimraf versions prior to v4 are no longer supported
- hasBin: true
-
- rollup-plugin-external-globals@0.8.0:
- resolution: {integrity: sha512-c65c7hPMCE//cLzC4dLVE25XkuHsBqSkZp+/5pvtZ1MFwqgQLRRkIfuCvI3PnI7Yj8HoXqYtdsRN9gYF5a4tVQ==}
- peerDependencies:
- rollup: ^2.25.0 || ^3.3.0
-
- rollup-plugin-visualizer@5.9.2:
- resolution: {integrity: sha512-waHktD5mlWrYFrhOLbti4YgQCn1uR24nYsNuXxg7LkPH8KdTXVWR9DNY1WU0QqokyMixVXJS4J04HNrVTMP01A==}
- engines: {node: '>=14'}
- hasBin: true
- peerDependencies:
- rollup: 2.x || 3.x
- peerDependenciesMeta:
- rollup:
- optional: true
-
- rollup@3.26.2:
- resolution: {integrity: sha512-6umBIGVz93er97pMgQO08LuH3m6PUb3jlDUUGFsNJB6VgTCUaDFpupf5JfU30529m/UKOgmiX+uY6Sx8cOYpLA==}
- engines: {node: '>=14.18.0', npm: '>=8.0.0'}
- hasBin: true
-
- rollup@https://registry.npmmirror.com/rollup/-/rollup-2.79.1.tgz:
- resolution: {integrity: sha512-uKxbd0IhMZOhjAiD5oAFp7BqvkA4Dv47qpOCtaNvng4HBwdbWtdOh8f5nZNuk2rp51PMGk3bzfWu5oayNEuYnw==, tarball: https://registry.npmmirror.com/rollup/-/rollup-2.79.1.tgz}
- version: 2.79.1
- engines: {node: '>=10.0.0'}
- hasBin: true
-
- run-parallel@https://registry.npmmirror.com/run-parallel/-/run-parallel-1.2.0.tgz:
- resolution: {integrity: sha512-5l4VyZR86LZ/lDxZTR6jqL8AFE2S0IFLMP26AbjsLVADxHdhB/c0GUsH+y39UfCi3dzz8OlQuPmnaJOMoDHQBA==, tarball: https://registry.npmmirror.com/run-parallel/-/run-parallel-1.2.0.tgz}
- version: 1.2.0
-
- sade@1.8.1:
- resolution: {integrity: sha512-xal3CZX1Xlo/k4ApwCFrHVACi9fBqJ7V+mwhBsuf/1IOKbBy098Fex+Wa/5QMubw09pSZ/u8EY8PWgevJsXp1A==}
- engines: {node: '>=6'}
-
- safe-buffer@5.1.2:
- resolution: {integrity: sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==}
-
- safe-buffer@5.2.1:
- resolution: {integrity: sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==}
-
- sass@1.80.1:
- resolution: {integrity: sha512-9lBwDZ7j3y/1DKj5Ec249EVGo5CVpwnzIyIj+cqlCjKkApLnzsJ/l9SnV4YnORvW9dQwQN+gQvh/mFZ8CnDs7Q==}
- engines: {node: '>=14.0.0'}
- hasBin: true
-
- select-dom@6.0.2:
- resolution: {integrity: sha512-fnRexLCjOE1XtMiV0xGACx4GJFMs/eatOQM0LWL9lxWB8DJ/FeXO39ztqUh6dZCOdoFo/g6jGJIIqNWoMsNLPA==}
-
- select@1.1.2:
- resolution: {integrity: sha512-OwpTSOfy6xSs1+pwcNrv0RBMOzI39Lp3qQKUTPVVPRjCdNa5JH/oPRiqsesIskK8TVgmRiHwO4KXlV2Li9dANA==}
-
- semiver@1.1.0:
- resolution: {integrity: sha512-QNI2ChmuioGC1/xjyYwyZYADILWyW6AmS1UH6gDj/SFUUUS4MBAWs/7mxnkRPc/F4iHezDP+O8t0dO8WHiEOdg==}
- engines: {node: '>=6'}
-
- semver@6.3.0:
- resolution: {integrity: sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==}
- hasBin: true
-
- semver@7.3.8:
- resolution: {integrity: sha512-NB1ctGL5rlHrPJtFDVIVzTyQylMLu9N9VICA6HSFJo8MCGVTMW6gfpicwKmmK/dAjTOrqu5l63JJOpDSrAis3A==}
- engines: {node: '>=10'}
- hasBin: true
-
- set-blocking@2.0.0:
- resolution: {integrity: sha512-KiKBS8AnWGEyLzofFfmvKwpdPzqiy16LvQfK3yv/fVH7Bj13/wl3JSR1J+rfgRE9q7xUJK4qvgS8raSOeLUehw==}
-
- shebang-command@2.0.0:
- resolution: {integrity: sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA==}
- engines: {node: '>=8'}
-
- shebang-regex@3.0.0:
- resolution: {integrity: sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==}
- engines: {node: '>=8'}
-
- showdown@2.1.0:
- resolution: {integrity: sha512-/6NVYu4U819R2pUIk79n67SYgJHWCce0a5xTP979WbNp0FL9MN1I1QK662IDU1b6JzKTvmhgI7T7JYIxBi3kMQ==}
- hasBin: true
-
- signal-exit@3.0.7:
- resolution: {integrity: sha512-wnD2ZE+l+SPC/uoS0vXeE9L1+0wuaMqKlfz9AMUo38JsyLSBWSFcHR1Rri62LZc12vLr1gb3jl7iwQhgwpAbGQ==}
-
- sirv-cli@1.0.14:
- resolution: {integrity: sha512-yyUTNr984ANKDloqepkYbBSqvx3buwYg2sQKPWjSU+IBia5loaoka2If8N9CMwt8AfP179cdEl7kYJ//iWJHjQ==}
- engines: {node: '>= 10'}
- hasBin: true
-
- sirv@1.0.19:
- resolution: {integrity: sha512-JuLThK3TnZG1TAKDwNIqNq6QA2afLOCcm+iE8D1Kj3GA40pSPsxQjjJl0J8X3tsR7T+CP1GavpzLwYkgVLWrZQ==}
- engines: {node: '>= 10'}
-
- source-map-js@1.0.2:
- resolution: {integrity: sha512-R0XvVJ9WusLiqTCEiGCmICCMplcCkIwwR11mOSD9CR5u+IXYdiseeEuXCVAjS54zqwkLcPNnmU4OeJ6tUrWhDw==}
- engines: {node: '>=0.10.0'}
-
- source-map-js@1.2.1:
- resolution: {integrity: sha512-UXWMKhLOwVKb728IUtQPXxfYU+usdybtUrK/8uGE8CQMvrhOpwvzDBwj0QhSL7MQc7vIsISBG8VQ8+IDQxpfQA==}
- engines: {node: '>=0.10.0'}
-
- source-map-js@https://registry.npmmirror.com/source-map-js/-/source-map-js-1.0.2.tgz:
- resolution: {integrity: sha512-R0XvVJ9WusLiqTCEiGCmICCMplcCkIwwR11mOSD9CR5u+IXYdiseeEuXCVAjS54zqwkLcPNnmU4OeJ6tUrWhDw==, tarball: https://registry.npmmirror.com/source-map-js/-/source-map-js-1.0.2.tgz}
- version: 1.0.2
- engines: {node: '>=0.10.0'}
-
- source-map@0.7.4:
- resolution: {integrity: sha512-l3BikUxvPOcn5E74dZiq5BGsTb5yEwhaTSzccU6t4sDOH8NWJCstKO5QT2CvtFoK6F0saL7p9xHAqHOlCPJygA==}
- engines: {node: '>= 8'}
-
- source-map@https://registry.npmmirror.com/source-map/-/source-map-0.6.1.tgz:
- resolution: {integrity: sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==, tarball: https://registry.npmmirror.com/source-map/-/source-map-0.6.1.tgz}
- version: 0.6.1
- engines: {node: '>=0.10.0'}
-
- sourcemap-codec@1.4.8:
- resolution: {integrity: sha512-9NykojV5Uih4lgo5So5dtw+f0JgJX30KCNI8gwhz2J9A15wD0Ml6tjHKwf6fTSa6fAdVBdZeNOs9eJ71qCk8vA==}
- deprecated: Please use @jridgewell/sourcemap-codec instead
-
- sourcemap-codec@https://registry.npmmirror.com/sourcemap-codec/-/sourcemap-codec-1.4.8.tgz:
- resolution: {integrity: sha512-9NykojV5Uih4lgo5So5dtw+f0JgJX30KCNI8gwhz2J9A15wD0Ml6tjHKwf6fTSa6fAdVBdZeNOs9eJ71qCk8vA==, tarball: https://registry.npmmirror.com/sourcemap-codec/-/sourcemap-codec-1.4.8.tgz}
- version: 1.4.8
-
- sprintf-js@1.0.3:
- resolution: {integrity: sha512-D9cPgkvLlV3t3IzL0D0YLvGA9Ahk4PcvVwUbN0dSGr1aP0Nrt4AEnTUbuGvquEC0mA64Gqt1fzirlRs5ibXx8g==}
-
- string-width@4.2.3:
- resolution: {integrity: sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==}
- engines: {node: '>=8'}
-
- string_decoder@1.3.0:
- resolution: {integrity: sha512-hkRX8U1WjJFd8LsDJ2yQ/wWWxaopEsABU1XfkM8A+j0+85JAGppt16cr1Whg6KIbb4okU6Mql6BOj+uup/wKeA==}
-
- strip-ansi@6.0.1:
- resolution: {integrity: sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==}
- engines: {node: '>=8'}
-
- strip-final-newline@2.0.0:
- resolution: {integrity: sha512-BrpvfNAE3dcvq7ll3xVumzjKjZQ5tI1sEUIKr3Uoks0XUl45St3FlatVqef9prk4jRDzhW6WZg+3bk93y6pLjA==}
- engines: {node: '>=6'}
-
- style-mod@4.1.2:
- resolution: {integrity: sha512-wnD1HyVqpJUI2+eKZ+eo1UwghftP6yuFheBqqe+bWCotBjC2K1YnteJILRMs3SM4V/0dLEW1SC27MWP5y+mwmw==}
-
- supports-color@5.5.0:
- resolution: {integrity: sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==}
- engines: {node: '>=4'}
-
- supports-preserve-symlinks-flag@https://registry.npmmirror.com/supports-preserve-symlinks-flag/-/supports-preserve-symlinks-flag-1.0.0.tgz:
- resolution: {integrity: sha512-ot0WnXS9fgdkgIcePe6RHNk1WA8+muPa6cSjeR3V8K27q9BB1rTE3R1p7Hv0z1ZyAc8s6Vvv8DIyWf681MAt0w==, tarball: https://registry.npmmirror.com/supports-preserve-symlinks-flag/-/supports-preserve-symlinks-flag-1.0.0.tgz}
- version: 1.0.0
- engines: {node: '>= 0.4'}
-
- svelte@4.2.19:
- resolution: {integrity: sha512-IY1rnGr6izd10B0A8LqsBfmlT5OILVuZ7XsI0vdGPEvuonFV7NYEUK4dAkm9Zg2q0Um92kYjTpS1CAP3Nh/KWw==}
- engines: {node: '>=16'}
-
- svgo@3.3.2:
- resolution: {integrity: sha512-OoohrmuUlBs8B8o6MB2Aevn+pRIH9zDALSR+6hhqVfa6fRwG/Qw9VUMSMW9VNg2CFc/MTIfabtdOVl9ODIJjpw==}
- engines: {node: '>=14.0.0'}
- hasBin: true
-
- tar@6.1.12:
- resolution: {integrity: sha512-jU4TdemS31uABHd+Lt5WEYJuzn+TJTCBLljvIAHZOz6M9Os5pJ4dD+vRFLxPa/n3T0iEFzpi+0x1UfuDZYbRMw==}
- engines: {node: '>=10'}
-
- three@0.166.1:
- resolution: {integrity: sha512-LtuafkKHHzm61AQA1be2MAYIw1IjmhOUxhBa0prrLpEMWbV7ijvxCRHjSgHPGp2493wLBzwKV46tA9nivLEgKg==}
-
- tiny-emitter@2.1.0:
- resolution: {integrity: sha512-NB6Dk1A9xgQPMoGqC5CVXn123gWyte215ONT5Pp5a0yt4nlEoO1ZWeCwpncaekPHXO60i47ihFnZPiRPjRMq4Q==}
-
- tiny-invariant@1.3.3:
- resolution: {integrity: sha512-+FbBPE1o9QAYvviau/qC5SE3caw21q3xkvWKBtja5vgqOWIHHJ3ioaq1VPfn/Szqctz2bU/oYeKd9/z5BL+PVg==}
-
- tinydate@1.3.0:
- resolution: {integrity: sha512-7cR8rLy2QhYHpsBDBVYnnWXm8uRTr38RoZakFSW7Bs7PzfMPNZthuMLkwqZv7MTu8lhQ91cOFYS5a7iFj2oR3w==}
- engines: {node: '>=4'}
-
- tlds@1.255.0:
- resolution: {integrity: sha512-tcwMRIioTcF/FcxLev8MJWxCp+GUALRhFEqbDoZrnowmKSGqPrl5pqS+Sut2m8BgJ6S4FExCSSpGffZ0Tks6Aw==}
- hasBin: true
-
- to-fast-properties@2.0.0:
- resolution: {integrity: sha512-/OaKK0xYrs3DmxRYqL/yDc+FxFUVYhDlXMhRmv3z915w2HF1tnN1omB354j8VUGO/hbRzyD6Y3sA7v7GS/ceog==}
- engines: {node: '>=4'}
-
- to-fast-properties@https://registry.npmmirror.com/to-fast-properties/-/to-fast-properties-2.0.0.tgz:
- resolution: {integrity: sha512-/OaKK0xYrs3DmxRYqL/yDc+FxFUVYhDlXMhRmv3z915w2HF1tnN1omB354j8VUGO/hbRzyD6Y3sA7v7GS/ceog==, tarball: https://registry.npmmirror.com/to-fast-properties/-/to-fast-properties-2.0.0.tgz}
- version: 2.0.0
- engines: {node: '>=4'}
-
- to-regex-range@5.0.1:
- resolution: {integrity: sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==}
- engines: {node: '>=8.0'}
-
- to-regex-range@https://registry.npmmirror.com/to-regex-range/-/to-regex-range-5.0.1.tgz:
- resolution: {integrity: sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==, tarball: https://registry.npmmirror.com/to-regex-range/-/to-regex-range-5.0.1.tgz}
- version: 5.0.1
- engines: {node: '>=8.0'}
-
- totalist@1.1.0:
- resolution: {integrity: sha512-gduQwd1rOdDMGxFG1gEvhV88Oirdo2p+KjoYFU7k2g+i7n6AFFbDQ5kMPUsW0pNbfQsB/cwXvT1i4Bue0s9g5g==}
- engines: {node: '>=6'}
-
- tr46@0.0.3:
- resolution: {integrity: sha512-N3WMsuqV66lT30CrXNbEjx4GEwlow3v6rr4mCcv6prnfwhS01rkgyFdjPNBYd9br7LpXV1+Emh01fHnq2Gdgrw==}
-
- trouter@2.0.1:
- resolution: {integrity: sha512-kr8SKKw94OI+xTGOkfsvwZQ8mWoikZDd2n8XZHjJVZUARZT+4/VV6cacRS6CLsH9bNm+HFIPU1Zx4CnNnb4qlQ==}
- engines: {node: '>=6'}
-
- tslib@2.4.1:
- resolution: {integrity: sha512-tGyy4dAjRIEwI7BzsB0lynWgOpfqjUdq91XXAlIWD2OwKBH7oCl/GZG/HT4BOHrTlPMOASlMQ7veyTqpmRcrNA==}
-
- typescript@https://registry.npmmirror.com/typescript/-/typescript-4.9.3.tgz:
- resolution: {integrity: sha512-CIfGzTelbKNEnLpLdGFgdyKhG23CKdKgQPOBc+OUNrkJ2vr+KSzsSV5kq5iWhEQbok+quxgGzrAtGWCyU7tHnA==, tarball: https://registry.npmmirror.com/typescript/-/typescript-4.9.3.tgz}
- version: 4.9.3
- engines: {node: '>=4.2.0'}
- hasBin: true
-
- unicode-canonical-property-names-ecmascript@2.0.0:
- resolution: {integrity: sha512-yY5PpDlfVIU5+y/BSCxAJRBIS1Zc2dDG3Ujq+sR0U+JjUevW2JhocOF+soROYDSaAezOzOKuyyixhD6mBknSmQ==}
- engines: {node: '>=4'}
-
- unicode-match-property-ecmascript@2.0.0:
- resolution: {integrity: sha512-5kaZCrbp5mmbz5ulBkDkbY0SsPOjKqVS35VpL9ulMPfSl0J0Xsm+9Evphv9CoIZFwre7aJoa94AY6seMKGVN5Q==}
- engines: {node: '>=4'}
-
- unicode-match-property-value-ecmascript@2.1.0:
- resolution: {integrity: sha512-qxkjQt6qjg/mYscYMC0XKRn3Rh0wFPlfxB0xkt9CfyTvpX1Ra0+rAmdX2QyAobptSEvuy4RtpPRui6XkV+8wjA==}
- engines: {node: '>=4'}
-
- unicode-property-aliases-ecmascript@2.1.0:
- resolution: {integrity: sha512-6t3foTQI9qne+OZoVQB/8x8rk2k1eVy1gRXhV3oFQ5T6R1dqQ1xtin3XqSlx3+ATBkliTaR/hHyJBm+LVPNM8w==}
- engines: {node: '>=4'}
-
- url-join@4.0.1:
- resolution: {integrity: sha512-jk1+QP6ZJqyOiuEI9AEWQfju/nB2Pw466kbA0LEZljHwKeMgd9WrAEgEGxjPDD2+TNbbb37rTyhEfrCXfuKXnA==}
-
- url-regex@5.0.0:
- resolution: {integrity: sha512-O08GjTiAFNsSlrUWfqF1jH0H1W3m35ZyadHrGv5krdnmPPoxP27oDTqux/579PtaroiSGm5yma6KT1mHFH6Y/g==}
- engines: {node: '>=8'}
-
- util-deprecate@1.0.2:
- resolution: {integrity: sha512-EPD5q1uXyFxJpCrLnCc1nHnq3gOa6DZBocAIiI2TaSCA7VCJ1UJDMagCzIkXNsUYfD1daK//LTEQ8xiIbrHtcw==}
-
- uuid@10.0.0:
- resolution: {integrity: sha512-8XkAphELsDnEGrDxUOHB3RGvXz6TeuYSGEZBOjtTtPm2lwhGBjLgOzLHB63IUWfBpNucQjND6d3AOudO+H3RWQ==}
- hasBin: true
-
- vanilla-jsoneditor@1.0.8:
- resolution: {integrity: sha512-5tKODR6J3IbGoMeBv4DEEU/Dlam4bPunOarIJKfrd5xs4bUife59Lp8Psjf86Y8OAM9mvxkrwR275XsBnvtO9A==}
-
- vanilla-picker@2.12.3:
- resolution: {integrity: sha512-qVkT1E7yMbUsB2mmJNFmaXMWE2hF8ffqzMMwe9zdAikd8u2VfnsVY2HQcOUi2F38bgbxzlJBEdS1UUhOXdF9GQ==}
-
- vary@1.1.2:
- resolution: {integrity: sha512-BNGbWLfd0eUPabhkXUVm0j8uuvREyTh5ovRa/dyow/BqAbZJyC+5fU+IzQOzmAKzYqYRAISoRhdQr3eIZ/PXqg==}
- engines: {node: '>= 0.8'}
-
- vite-aliases@https://registry.npmmirror.com/vite-aliases/-/vite-aliases-0.9.7.tgz:
- resolution: {integrity: sha512-MiqZx5NJQDn/I7gYzovop8GzlEDYCxR3i3JCCNap5OWEfBmo4b+jIfpmal22IoBREgx1TtkmwRkHJCBUhnL6jA==, tarball: https://registry.npmmirror.com/vite-aliases/-/vite-aliases-0.9.7.tgz}
- version: 0.9.7
- engines: {node: '>=16.6.0', npm: '>=7.0.0'}
-
- vite-plugin-dynamic-import@https://registry.npmmirror.com/vite-plugin-dynamic-import/-/vite-plugin-dynamic-import-1.2.4.tgz:
- resolution: {integrity: sha512-R6spqDhDm8VxaUNyEkybh9PM262ReO8WXJ6rjtyJ/H8Y4b+pq2uM9Xz5DMu7N7OxUGlqo9HSzT3VjhpbySXrng==, tarball: https://registry.npmmirror.com/vite-plugin-dynamic-import/-/vite-plugin-dynamic-import-1.2.4.tgz}
- version: 1.2.4
-
- vite-plugin-externalize-deps@0.7.0:
- resolution: {integrity: sha512-do2gPrR79Tm8UKcqsw3RTAtN4YO8GkVRBckWdJWINZ3Qdp3KN9S1oyUZxKszTB/iyg4zdOUweLOeBI8t86QVow==}
- version: 0.7.0
- peerDependencies:
- vite: ^2.0.0 || ^3.0.0 || ^4.0.0
-
- vite-svg-loader@5.1.0:
- resolution: {integrity: sha512-M/wqwtOEjgb956/+m5ZrYT/Iq6Hax0OakWbokj8+9PXOnB7b/4AxESHieEtnNEy7ZpjsjYW1/5nK8fATQMmRxw==}
- version: 5.1.0
- peerDependencies:
- vue: '>=3.2.13'
-
- vite@https://registry.npmmirror.com/vite/-/vite-3.2.4.tgz:
- resolution: {integrity: sha512-Z2X6SRAffOUYTa+sLy3NQ7nlHFU100xwanq1WDwqaiFiCe+25zdxP1TfCS5ojPV2oDDcXudHIoPnI1Z/66B7Yw==, tarball: https://registry.npmmirror.com/vite/-/vite-3.2.4.tgz}
- version: 3.2.4
- engines: {node: ^14.18.0 || >=16.0.0}
- hasBin: true
- peerDependencies:
- '@types/node': '>= 14'
- less: '*'
- sass: '*'
- stylus: '*'
- sugarss: '*'
- terser: ^5.4.0
- peerDependenciesMeta:
- '@types/node':
- optional: true
- less:
- optional: true
- sass:
- optional: true
- stylus:
- optional: true
- sugarss:
- optional: true
- terser:
- optional: true
-
- vue-chartjs@5.3.1:
- resolution: {integrity: sha512-rZjqcHBxKiHrBl0CIvcOlVEBwRhpWAVf6rDU3vUfa7HuSRmGtCslc0Oc8m16oAVuk0erzc1FCtH1VCriHsrz+A==}
- version: 5.3.1
- peerDependencies:
- chart.js: ^4.1.1
- vue: ^3.0.0-0 || ^2.7.0
-
- vue-clipboard3@2.0.0:
- resolution: {integrity: sha512-Q9S7dzWGax7LN5iiSPcu/K1GGm2gcBBlYwmMsUc5/16N6w90cbKow3FnPmPs95sungns4yvd9/+JhbAznECS2A==}
-
- vue-demi@0.14.10:
- resolution: {integrity: sha512-nMZBOwuzabUO0nLgIcc6rycZEebF6eeUfaiQx9+WSk8e29IbLvPU9feI6tqW4kTo3hvoYAJkMh8n8D0fuISphg==}
- version: 0.14.10
- engines: {node: '>=12'}
- hasBin: true
- peerDependencies:
- '@vue/composition-api': ^1.0.0-rc.1
- vue: ^3.0.0-0 || ^2.6.0
- peerDependenciesMeta:
- '@vue/composition-api':
- optional: true
-
- vue-i18n@9.13.1:
- resolution: {integrity: sha512-mh0GIxx0wPtPlcB1q4k277y0iKgo25xmDPWioVVYanjPufDBpvu5ySTjP5wOrSvlYQ2m1xI+CFhGdauv/61uQg==}
- version: 9.13.1
- engines: {node: '>= 16'}
- peerDependencies:
- vue: ^3.0.0
-
- vue-router@4.4.0:
- resolution: {integrity: sha512-HB+t2p611aIZraV2aPSRNXf0Z/oLZFrlygJm+sZbdJaW6lcFqEDQwnzUBXn+DApw+/QzDU/I9TeWx9izEjTmsA==}
- version: 4.4.0
- peerDependencies:
- vue: ^3.2.0
-
- vue-template-compiler@https://registry.npmmirror.com/vue-template-compiler/-/vue-template-compiler-2.7.14.tgz:
- resolution: {integrity: sha512-zyA5Y3ArvVG0NacJDkkzJuPQDF8RFeRlzV2vLeSnhSpieO6LK2OVbdLPi5MPPs09Ii+gMO8nY4S3iKQxBxDmWQ==, tarball: https://registry.npmmirror.com/vue-template-compiler/-/vue-template-compiler-2.7.14.tgz}
- version: 2.7.14
-
- vue-tsc@https://registry.npmmirror.com/vue-tsc/-/vue-tsc-1.0.11.tgz:
- resolution: {integrity: sha512-lj+6dEroPsE4wmQOPtjCzAf8x363Km5/tuEvMEoQaoRnzs9myBM46FNvCGIIPStYUGuaqF1W1bORmP2KDQEORA==, tarball: https://registry.npmmirror.com/vue-tsc/-/vue-tsc-1.0.11.tgz}
- version: 1.0.11
- hasBin: true
- peerDependencies:
- typescript: '*'
-
- vue@3.5.12:
- resolution: {integrity: sha512-CLVZtXtn2ItBIi/zHZ0Sg1Xkb7+PU32bJJ8Bmy7ts3jxXTcbfsEfBivFYYWz1Hur+lalqGAh65Coin0r+HRUfg==}
- version: 3.5.12
- peerDependencies:
- typescript: '*'
- peerDependenciesMeta:
- typescript:
- optional: true
-
- vuex@4.1.0:
- resolution: {integrity: sha512-hmV6UerDrPcgbSy9ORAtNXDr9M4wlNP4pEFKye4ujJF8oqgFFuxDCdOLS3eNoRTtq5O3hoBDh9Doj1bQMYHRbQ==}
- version: 4.1.0
- peerDependencies:
- vue: ^3.2.0
-
- w3c-keyname@2.2.8:
- resolution: {integrity: sha512-dpojBhNsCNN7T82Tm7k26A6G9ML3NkhDsnw9n/eoxSRlVBB4CEtIQ/KTCLI2Fwf3ataSXRhYFkQi3SlnFwPvPQ==}
-
- webidl-conversions@3.0.1:
- resolution: {integrity: sha512-2JAn3z8AR6rjK8Sm8orRC0h/bcl/DqL7tRPdGZ4I1CjdF+EaMLmYxBHyXuKL849eucPFhvBoxMsflfOb8kxaeQ==}
-
- whatwg-url@5.0.0:
- resolution: {integrity: sha512-saE57nupxk6v3HY35+jzBwYa0rKSy0XR8JSxZPwgLr7ys0IBzhGviA1/TUGJLmSVqs8pb9AnvICXEuOHLprYTw==}
-
- which@2.0.2:
- resolution: {integrity: sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==}
- engines: {node: '>= 8'}
- hasBin: true
-
- wide-align@1.1.5:
- resolution: {integrity: sha512-eDMORYaPNZ4sQIuuYPDHdQvf4gyCF9rEEV/yPxGfwPkRodwEgiMUUXTx/dex+Me0wxx53S+NgUHaP7y3MGlDmg==}
-
- windows-release@5.0.1:
- resolution: {integrity: sha512-y1xFdFvdMiDXI3xiOhMbJwt1Y7dUxidha0CWPs1NgjZIjZANTcX7+7bMqNjuezhzb8s5JGEiBAbQjQQYYy7ulw==}
- engines: {node: ^12.20.0 || ^14.13.1 || >=16.0.0}
-
- wrap-ansi@7.0.0:
- resolution: {integrity: sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q==}
- engines: {node: '>=10'}
-
- wrappy@1.0.2:
- resolution: {integrity: sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ==}
-
- y18n@5.0.8:
- resolution: {integrity: sha512-0pfFzegeDWJHJIAmTLRP2DwHjdF5s7jo9tuztdQxAhINCdvS+3nGINqPd00AphqJR/0LhANUS6/+7SCb98YOfA==}
- engines: {node: '>=10'}
-
- yallist@4.0.0:
- resolution: {integrity: sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==}
-
- yargs-parser@21.1.1:
- resolution: {integrity: sha512-tVpsJW7DdjecAiFpbIB1e3qxIQsE6NoPc5/eTdrbbIC4h0LVsWhnoa3g+m2HclBIujHzsxZ4VJVA+GUuc2/LBw==}
- engines: {node: '>=12'}
-
- yargs@17.7.2:
- resolution: {integrity: sha512-7dSzzRQ++CKnNI/krKnYRV7JKKPUXMEh61soaHKg9mrWEhzFWhFnxPxGl+69cD1Ou63C13NUPCnmIcrvqCuM6w==}
- engines: {node: '>=12'}
-
-snapshots:
-
- '@ampproject/remapping@2.3.0':
- dependencies:
- '@jridgewell/gen-mapping': 0.3.5
- '@jridgewell/trace-mapping': 0.3.25
-
- '@arr/every@1.0.1': {}
-
- '@babel/helper-string-parser@7.25.7': {}
-
- '@babel/helper-string-parser@https://registry.npmmirror.com/@babel/helper-string-parser/-/helper-string-parser-7.19.4.tgz': {}
-
- '@babel/helper-validator-identifier@7.25.7': {}
-
- '@babel/helper-validator-identifier@https://registry.npmmirror.com/@babel/helper-validator-identifier/-/helper-validator-identifier-7.19.1.tgz': {}
-
- '@babel/parser@7.25.8':
- dependencies:
- '@babel/types': 7.25.8
-
- '@babel/parser@https://registry.npmmirror.com/@babel/parser/-/parser-7.20.5.tgz':
- dependencies:
- '@babel/types': https://registry.npmmirror.com/@babel/types/-/types-7.20.5.tgz
-
- '@babel/types@7.25.8':
- dependencies:
- '@babel/helper-string-parser': 7.25.7
- '@babel/helper-validator-identifier': 7.25.7
- to-fast-properties: 2.0.0
-
- '@babel/types@https://registry.npmmirror.com/@babel/types/-/types-7.20.5.tgz':
- dependencies:
- '@babel/helper-string-parser': https://registry.npmmirror.com/@babel/helper-string-parser/-/helper-string-parser-7.19.4.tgz
- '@babel/helper-validator-identifier': https://registry.npmmirror.com/@babel/helper-validator-identifier/-/helper-validator-identifier-7.19.1.tgz
- to-fast-properties: https://registry.npmmirror.com/to-fast-properties/-/to-fast-properties-2.0.0.tgz
-
- '@codemirror/autocomplete@6.18.1(@codemirror/language@6.10.3)(@codemirror/state@6.4.1)(@codemirror/view@6.34.1)(@lezer/common@1.2.3)':
- dependencies:
- '@codemirror/language': 6.10.3
- '@codemirror/state': 6.4.1
- '@codemirror/view': 6.34.1
- '@lezer/common': 1.2.3
-
- '@codemirror/commands@6.7.0':
- dependencies:
- '@codemirror/language': 6.10.3
- '@codemirror/state': 6.4.1
- '@codemirror/view': 6.34.1
- '@lezer/common': 1.2.3
-
- '@codemirror/lang-json@6.0.1':
- dependencies:
- '@codemirror/language': 6.10.3
- '@lezer/json': 1.0.2
-
- '@codemirror/language@6.10.3':
- dependencies:
- '@codemirror/state': 6.4.1
- '@codemirror/view': 6.34.1
- '@lezer/common': 1.2.3
- '@lezer/highlight': 1.2.1
- '@lezer/lr': 1.4.2
- style-mod: 4.1.2
-
- '@codemirror/lint@6.8.2':
- dependencies:
- '@codemirror/state': 6.4.1
- '@codemirror/view': 6.34.1
- crelt: 1.0.6
-
- '@codemirror/search@6.5.6':
- dependencies:
- '@codemirror/state': 6.4.1
- '@codemirror/view': 6.34.1
- crelt: 1.0.6
-
- '@codemirror/state@6.4.1': {}
-
- '@codemirror/view@6.34.1':
- dependencies:
- '@codemirror/state': 6.4.1
- style-mod: 4.1.2
- w3c-keyname: 2.2.8
-
- '@ctrl/tinycolor@3.4.1': {}
-
- '@element-plus/icons-vue@2.3.1(vue@3.5.12(typescript@https://registry.npmmirror.com/typescript/-/typescript-4.9.3.tgz))':
- dependencies:
- vue: 3.5.12(typescript@https://registry.npmmirror.com/typescript/-/typescript-4.9.3.tgz)
-
- '@esbuild/android-arm@0.15.16':
- optional: true
-
- '@esbuild/linux-loong64@0.15.16':
- optional: true
-
- '@floating-ui/core@1.6.8':
- dependencies:
- '@floating-ui/utils': 0.2.8
-
- '@floating-ui/dom@1.6.11':
- dependencies:
- '@floating-ui/core': 1.6.8
- '@floating-ui/utils': 0.2.8
-
- '@floating-ui/utils@0.2.8': {}
-
- '@fortawesome/fontawesome-common-types@6.6.0': {}
-
- '@fortawesome/fontawesome-svg-core@6.6.0':
- dependencies:
- '@fortawesome/fontawesome-common-types': 6.6.0
-
- '@fortawesome/free-brands-svg-icons@6.6.0':
- dependencies:
- '@fortawesome/fontawesome-common-types': 6.6.0
-
- '@fortawesome/free-regular-svg-icons@6.6.0':
- dependencies:
- '@fortawesome/fontawesome-common-types': 6.6.0
-
- '@fortawesome/free-solid-svg-icons@6.6.0':
- dependencies:
- '@fortawesome/fontawesome-common-types': 6.6.0
-
- '@fortawesome/vue-fontawesome@3.0.8(@fortawesome/fontawesome-svg-core@6.6.0)(vue@3.5.12(typescript@https://registry.npmmirror.com/typescript/-/typescript-4.9.3.tgz))':
- dependencies:
- '@fortawesome/fontawesome-svg-core': 6.6.0
- vue: 3.5.12(typescript@https://registry.npmmirror.com/typescript/-/typescript-4.9.3.tgz)
-
- '@intlify/core-base@9.13.1':
- dependencies:
- '@intlify/message-compiler': 9.13.1
- '@intlify/shared': 9.13.1
-
- '@intlify/message-compiler@9.13.1':
- dependencies:
- '@intlify/shared': 9.13.1
- source-map-js: 1.0.2
-
- '@intlify/shared@9.13.1': {}
-
- '@jridgewell/gen-mapping@0.3.5':
- dependencies:
- '@jridgewell/set-array': 1.2.1
- '@jridgewell/sourcemap-codec': 1.5.0
- '@jridgewell/trace-mapping': 0.3.25
-
- '@jridgewell/resolve-uri@3.1.2': {}
-
- '@jridgewell/set-array@1.2.1': {}
-
- '@jridgewell/sourcemap-codec@1.4.15': {}
-
- '@jridgewell/sourcemap-codec@1.5.0': {}
-
- '@jridgewell/trace-mapping@0.3.25':
- dependencies:
- '@jridgewell/resolve-uri': 3.1.2
- '@jridgewell/sourcemap-codec': 1.5.0
-
- '@jsep-plugin/assignment@1.2.1(jsep@1.3.9)':
- dependencies:
- jsep: 1.3.9
-
- '@jsep-plugin/regex@1.0.3(jsep@1.3.9)':
- dependencies:
- jsep: 1.3.9
-
- '@jsonquerylang/jsonquery@3.1.1': {}
-
- '@kurkle/color@0.3.2': {}
-
- '@lexical/clipboard@0.16.1':
- dependencies:
- '@lexical/html': 0.16.1
- '@lexical/list': 0.16.1
- '@lexical/selection': 0.16.1
- '@lexical/utils': 0.16.1
- lexical: 0.16.1
-
- '@lexical/code@0.16.1':
- dependencies:
- '@lexical/utils': 0.16.1
- lexical: 0.16.1
- prismjs: 1.29.0
-
- '@lexical/history@0.16.1':
- dependencies:
- '@lexical/utils': 0.16.1
- lexical: 0.16.1
-
- '@lexical/html@0.16.1':
- dependencies:
- '@lexical/selection': 0.16.1
- '@lexical/utils': 0.16.1
- lexical: 0.16.1
-
- '@lexical/link@0.16.1':
- dependencies:
- '@lexical/utils': 0.16.1
- lexical: 0.16.1
-
- '@lexical/list@0.16.1':
- dependencies:
- '@lexical/utils': 0.16.1
- lexical: 0.16.1
-
- '@lexical/markdown@0.16.1':
- dependencies:
- '@lexical/code': 0.16.1
- '@lexical/link': 0.16.1
- '@lexical/list': 0.16.1
- '@lexical/rich-text': 0.16.1
- '@lexical/text': 0.16.1
- '@lexical/utils': 0.16.1
- lexical: 0.16.1
-
- '@lexical/rich-text@0.16.1':
- dependencies:
- '@lexical/clipboard': 0.16.1
- '@lexical/selection': 0.16.1
- '@lexical/utils': 0.16.1
- lexical: 0.16.1
-
- '@lexical/selection@0.16.1':
- dependencies:
- lexical: 0.16.1
-
- '@lexical/table@0.16.1':
- dependencies:
- '@lexical/utils': 0.16.1
- lexical: 0.16.1
-
- '@lexical/text@0.16.1':
- dependencies:
- lexical: 0.16.1
-
- '@lexical/utils@0.16.1':
- dependencies:
- '@lexical/list': 0.16.1
- '@lexical/selection': 0.16.1
- '@lexical/table': 0.16.1
- lexical: 0.16.1
-
- '@lezer/common@1.2.3': {}
-
- '@lezer/highlight@1.2.1':
- dependencies:
- '@lezer/common': 1.2.3
-
- '@lezer/json@1.0.2':
- dependencies:
- '@lezer/common': 1.2.3
- '@lezer/highlight': 1.2.1
- '@lezer/lr': 1.4.2
-
- '@lezer/lr@1.4.2':
- dependencies:
- '@lezer/common': 1.2.3
-
- '@mapbox/node-pre-gyp@1.0.10':
- dependencies:
- detect-libc: 2.0.1
- https-proxy-agent: 5.0.1
- make-dir: 3.1.0
- node-fetch: 2.6.7
- nopt: 5.0.0
- npmlog: 5.0.1
- rimraf: 3.0.2
- semver: 7.3.8
- tar: 6.1.12
- transitivePeerDependencies:
- - encoding
- - supports-color
- optional: true
-
- '@nodelib/fs.scandir@https://registry.npmmirror.com/@nodelib/fs.scandir/-/fs.scandir-2.1.5.tgz':
- dependencies:
- '@nodelib/fs.stat': https://registry.npmmirror.com/@nodelib/fs.stat/-/fs.stat-2.0.5.tgz
- run-parallel: https://registry.npmmirror.com/run-parallel/-/run-parallel-1.2.0.tgz
-
- '@nodelib/fs.stat@https://registry.npmmirror.com/@nodelib/fs.stat/-/fs.stat-2.0.5.tgz': {}
-
- '@nodelib/fs.walk@https://registry.npmmirror.com/@nodelib/fs.walk/-/fs.walk-1.2.8.tgz':
- dependencies:
- '@nodelib/fs.scandir': https://registry.npmmirror.com/@nodelib/fs.scandir/-/fs.scandir-2.1.5.tgz
- fastq: https://registry.npmmirror.com/fastq/-/fastq-1.14.0.tgz
-
- '@parcel/watcher-android-arm64@2.4.1':
- optional: true
-
- '@parcel/watcher-darwin-arm64@2.4.1':
- optional: true
-
- '@parcel/watcher-darwin-x64@2.4.1':
- optional: true
-
- '@parcel/watcher-freebsd-x64@2.4.1':
- optional: true
-
- '@parcel/watcher-linux-arm-glibc@2.4.1':
- optional: true
-
- '@parcel/watcher-linux-arm64-glibc@2.4.1':
- optional: true
-
- '@parcel/watcher-linux-arm64-musl@2.4.1':
- optional: true
-
- '@parcel/watcher-linux-x64-glibc@2.4.1':
- optional: true
-
- '@parcel/watcher-linux-x64-musl@2.4.1':
- optional: true
-
- '@parcel/watcher-win32-arm64@2.4.1':
- optional: true
-
- '@parcel/watcher-win32-ia32@2.4.1':
- optional: true
-
- '@parcel/watcher-win32-x64@2.4.1':
- optional: true
-
- '@parcel/watcher@2.4.1':
- dependencies:
- detect-libc: 1.0.3
- is-glob: 4.0.3
- micromatch: 4.0.5
- node-addon-api: 7.1.1
- optionalDependencies:
- '@parcel/watcher-android-arm64': 2.4.1
- '@parcel/watcher-darwin-arm64': 2.4.1
- '@parcel/watcher-darwin-x64': 2.4.1
- '@parcel/watcher-freebsd-x64': 2.4.1
- '@parcel/watcher-linux-arm-glibc': 2.4.1
- '@parcel/watcher-linux-arm64-glibc': 2.4.1
- '@parcel/watcher-linux-arm64-musl': 2.4.1
- '@parcel/watcher-linux-x64-glibc': 2.4.1
- '@parcel/watcher-linux-x64-musl': 2.4.1
- '@parcel/watcher-win32-arm64': 2.4.1
- '@parcel/watcher-win32-ia32': 2.4.1
- '@parcel/watcher-win32-x64': 2.4.1
-
- '@polka/url@0.5.0': {}
-
- '@polka/url@1.0.0-next.21': {}
-
- '@popperjs/core@2.11.8': {}
-
- '@replit/codemirror-indentation-markers@6.5.3(@codemirror/language@6.10.3)(@codemirror/state@6.4.1)(@codemirror/view@6.34.1)':
- dependencies:
- '@codemirror/language': 6.10.3
- '@codemirror/state': 6.4.1
- '@codemirror/view': 6.34.1
-
- '@rollup/pluginutils@5.0.2(rollup@3.26.2)':
- dependencies:
- '@types/estree': 1.0.0
- estree-walker: 2.0.2
- picomatch: 2.3.1
- optionalDependencies:
- rollup: 3.26.2
-
- '@sphinxxxx/color-conversion@2.2.2': {}
-
- '@sxzz/popperjs-es@2.11.7': {}
-
- '@trysound/sax@0.2.0': {}
-
- '@tweenjs/tween.js@23.1.3': {}
-
- '@types/estree@1.0.0': {}
-
- '@types/estree@1.0.6': {}
-
- '@types/getos@3.0.4': {}
-
- '@types/humanize-duration@3.27.4': {}
-
- '@types/javascript-time-ago@2.0.8': {}
-
- '@types/lodash-es@4.17.12':
- dependencies:
- '@types/lodash': 4.17.10
-
- '@types/lodash@4.17.10': {}
-
- '@types/md5@2.3.5': {}
-
- '@types/node@https://registry.npmmirror.com/@types/node/-/node-18.11.10.tgz': {}
-
- '@types/pinyin@2.10.2': {}
-
- '@types/showdown@2.0.6': {}
-
- '@types/stats.js@0.17.3': {}
-
- '@types/three@0.166.0':
- dependencies:
- '@tweenjs/tween.js': 23.1.3
- '@types/stats.js': 0.17.3
- '@types/webxr': 0.5.20
- fflate: 0.8.2
- meshoptimizer: 0.18.1
-
- '@types/url-join@4.0.3': {}
-
- '@types/web-bluetooth@0.0.16': {}
-
- '@types/webxr@0.5.20': {}
-
- '@vitejs/plugin-vue@https://registry.npmmirror.com/@vitejs/plugin-vue/-/plugin-vue-3.2.0.tgz(vite@https://registry.npmmirror.com/vite/-/vite-3.2.4.tgz(@types/node@https://registry.npmmirror.com/@types/node/-/node-18.11.10.tgz)(sass@1.80.1))(vue@3.5.12(typescript@https://registry.npmmirror.com/typescript/-/typescript-4.9.3.tgz))':
- dependencies:
- vite: https://registry.npmmirror.com/vite/-/vite-3.2.4.tgz(@types/node@https://registry.npmmirror.com/@types/node/-/node-18.11.10.tgz)(sass@1.80.1)
- vue: 3.5.12(typescript@https://registry.npmmirror.com/typescript/-/typescript-4.9.3.tgz)
-
- '@volar/language-core@https://registry.npmmirror.com/@volar/language-core/-/language-core-1.0.11.tgz':
- dependencies:
- '@volar/source-map': https://registry.npmmirror.com/@volar/source-map/-/source-map-1.0.11.tgz
- '@vue/reactivity': https://registry.npmmirror.com/@vue/reactivity/-/reactivity-3.2.45.tgz
- muggle-string: https://registry.npmmirror.com/muggle-string/-/muggle-string-0.1.0.tgz
-
- '@volar/source-map@https://registry.npmmirror.com/@volar/source-map/-/source-map-1.0.11.tgz':
- dependencies:
- muggle-string: https://registry.npmmirror.com/muggle-string/-/muggle-string-0.1.0.tgz
-
- '@volar/typescript@https://registry.npmmirror.com/@volar/typescript/-/typescript-1.0.11.tgz':
- dependencies:
- '@volar/language-core': https://registry.npmmirror.com/@volar/language-core/-/language-core-1.0.11.tgz
-
- '@volar/vue-language-core@https://registry.npmmirror.com/@volar/vue-language-core/-/vue-language-core-1.0.11.tgz':
- dependencies:
- '@volar/language-core': https://registry.npmmirror.com/@volar/language-core/-/language-core-1.0.11.tgz
- '@volar/source-map': https://registry.npmmirror.com/@volar/source-map/-/source-map-1.0.11.tgz
- '@vue/compiler-dom': https://registry.npmmirror.com/@vue/compiler-dom/-/compiler-dom-3.2.45.tgz
- '@vue/compiler-sfc': https://registry.npmmirror.com/@vue/compiler-sfc/-/compiler-sfc-3.2.45.tgz
- '@vue/reactivity': https://registry.npmmirror.com/@vue/reactivity/-/reactivity-3.2.45.tgz
- '@vue/shared': https://registry.npmmirror.com/@vue/shared/-/shared-3.2.45.tgz
- minimatch: https://registry.npmmirror.com/minimatch/-/minimatch-5.1.1.tgz
- vue-template-compiler: https://registry.npmmirror.com/vue-template-compiler/-/vue-template-compiler-2.7.14.tgz
-
- '@volar/vue-typescript@https://registry.npmmirror.com/@volar/vue-typescript/-/vue-typescript-1.0.11.tgz':
- dependencies:
- '@volar/typescript': https://registry.npmmirror.com/@volar/typescript/-/typescript-1.0.11.tgz
- '@volar/vue-language-core': https://registry.npmmirror.com/@volar/vue-language-core/-/vue-language-core-1.0.11.tgz
-
- '@vue/compiler-core@3.5.12':
- dependencies:
- '@babel/parser': 7.25.8
- '@vue/shared': 3.5.12
- entities: 4.5.0
- estree-walker: 2.0.2
- source-map-js: 1.2.1
-
- '@vue/compiler-core@https://registry.npmmirror.com/@vue/compiler-core/-/compiler-core-3.2.45.tgz':
- dependencies:
- '@babel/parser': https://registry.npmmirror.com/@babel/parser/-/parser-7.20.5.tgz
- '@vue/shared': https://registry.npmmirror.com/@vue/shared/-/shared-3.2.45.tgz
- estree-walker: https://registry.npmmirror.com/estree-walker/-/estree-walker-2.0.2.tgz
- source-map: https://registry.npmmirror.com/source-map/-/source-map-0.6.1.tgz
-
- '@vue/compiler-dom@3.5.12':
- dependencies:
- '@vue/compiler-core': 3.5.12
- '@vue/shared': 3.5.12
-
- '@vue/compiler-dom@https://registry.npmmirror.com/@vue/compiler-dom/-/compiler-dom-3.2.45.tgz':
- dependencies:
- '@vue/compiler-core': https://registry.npmmirror.com/@vue/compiler-core/-/compiler-core-3.2.45.tgz
- '@vue/shared': https://registry.npmmirror.com/@vue/shared/-/shared-3.2.45.tgz
-
- '@vue/compiler-sfc@3.5.12':
- dependencies:
- '@babel/parser': 7.25.8
- '@vue/compiler-core': 3.5.12
- '@vue/compiler-dom': 3.5.12
- '@vue/compiler-ssr': 3.5.12
- '@vue/shared': 3.5.12
- estree-walker: 2.0.2
- magic-string: 0.30.12
- postcss: 8.4.47
- source-map-js: 1.2.1
-
- '@vue/compiler-sfc@https://registry.npmmirror.com/@vue/compiler-sfc/-/compiler-sfc-3.2.45.tgz':
- dependencies:
- '@babel/parser': https://registry.npmmirror.com/@babel/parser/-/parser-7.20.5.tgz
- '@vue/compiler-core': https://registry.npmmirror.com/@vue/compiler-core/-/compiler-core-3.2.45.tgz
- '@vue/compiler-dom': https://registry.npmmirror.com/@vue/compiler-dom/-/compiler-dom-3.2.45.tgz
- '@vue/compiler-ssr': https://registry.npmmirror.com/@vue/compiler-ssr/-/compiler-ssr-3.2.45.tgz
- '@vue/reactivity-transform': https://registry.npmmirror.com/@vue/reactivity-transform/-/reactivity-transform-3.2.45.tgz
- '@vue/shared': https://registry.npmmirror.com/@vue/shared/-/shared-3.2.45.tgz
- estree-walker: https://registry.npmmirror.com/estree-walker/-/estree-walker-2.0.2.tgz
- magic-string: https://registry.npmmirror.com/magic-string/-/magic-string-0.25.9.tgz
- postcss: https://registry.npmmirror.com/postcss/-/postcss-8.4.19.tgz
- source-map: https://registry.npmmirror.com/source-map/-/source-map-0.6.1.tgz
-
- '@vue/compiler-ssr@3.5.12':
- dependencies:
- '@vue/compiler-dom': 3.5.12
- '@vue/shared': 3.5.12
-
- '@vue/compiler-ssr@https://registry.npmmirror.com/@vue/compiler-ssr/-/compiler-ssr-3.2.45.tgz':
- dependencies:
- '@vue/compiler-dom': https://registry.npmmirror.com/@vue/compiler-dom/-/compiler-dom-3.2.45.tgz
- '@vue/shared': https://registry.npmmirror.com/@vue/shared/-/shared-3.2.45.tgz
-
- '@vue/devtools-api@6.4.5': {}
-
- '@vue/devtools-api@6.6.4': {}
-
- '@vue/reactivity-transform@https://registry.npmmirror.com/@vue/reactivity-transform/-/reactivity-transform-3.2.45.tgz':
- dependencies:
- '@babel/parser': https://registry.npmmirror.com/@babel/parser/-/parser-7.20.5.tgz
- '@vue/compiler-core': https://registry.npmmirror.com/@vue/compiler-core/-/compiler-core-3.2.45.tgz
- '@vue/shared': https://registry.npmmirror.com/@vue/shared/-/shared-3.2.45.tgz
- estree-walker: https://registry.npmmirror.com/estree-walker/-/estree-walker-2.0.2.tgz
- magic-string: https://registry.npmmirror.com/magic-string/-/magic-string-0.25.9.tgz
-
- '@vue/reactivity@3.5.12':
- dependencies:
- '@vue/shared': 3.5.12
-
- '@vue/reactivity@https://registry.npmmirror.com/@vue/reactivity/-/reactivity-3.2.45.tgz':
- dependencies:
- '@vue/shared': https://registry.npmmirror.com/@vue/shared/-/shared-3.2.45.tgz
-
- '@vue/runtime-core@3.5.12':
- dependencies:
- '@vue/reactivity': 3.5.12
- '@vue/shared': 3.5.12
-
- '@vue/runtime-dom@3.5.12':
- dependencies:
- '@vue/reactivity': 3.5.12
- '@vue/runtime-core': 3.5.12
- '@vue/shared': 3.5.12
- csstype: 3.1.3
-
- '@vue/server-renderer@3.5.12(vue@3.5.12(typescript@https://registry.npmmirror.com/typescript/-/typescript-4.9.3.tgz))':
- dependencies:
- '@vue/compiler-ssr': 3.5.12
- '@vue/shared': 3.5.12
- vue: 3.5.12(typescript@https://registry.npmmirror.com/typescript/-/typescript-4.9.3.tgz)
-
- '@vue/shared@3.5.12': {}
-
- '@vue/shared@https://registry.npmmirror.com/@vue/shared/-/shared-3.2.45.tgz': {}
-
- '@vueuse/core@9.13.0(vue@3.5.12(typescript@https://registry.npmmirror.com/typescript/-/typescript-4.9.3.tgz))':
- dependencies:
- '@types/web-bluetooth': 0.0.16
- '@vueuse/metadata': 9.13.0
- '@vueuse/shared': 9.13.0(vue@3.5.12(typescript@https://registry.npmmirror.com/typescript/-/typescript-4.9.3.tgz))
- vue-demi: 0.14.10(vue@3.5.12(typescript@https://registry.npmmirror.com/typescript/-/typescript-4.9.3.tgz))
- transitivePeerDependencies:
- - '@vue/composition-api'
- - vue
-
- '@vueuse/metadata@9.13.0': {}
-
- '@vueuse/shared@9.13.0(vue@3.5.12(typescript@https://registry.npmmirror.com/typescript/-/typescript-4.9.3.tgz))':
- dependencies:
- vue-demi: 0.14.10(vue@3.5.12(typescript@https://registry.npmmirror.com/typescript/-/typescript-4.9.3.tgz))
- transitivePeerDependencies:
- - '@vue/composition-api'
- - vue
-
- abbrev@1.1.1:
- optional: true
-
- accepts@1.3.8:
- dependencies:
- mime-types: 2.1.35
- negotiator: 0.6.3
-
- acorn-dynamic-import@4.0.0(acorn@6.4.2):
- dependencies:
- acorn: 6.4.2
-
- acorn-jsx@5.3.2(acorn@6.4.2):
- dependencies:
- acorn: 6.4.2
-
- acorn@6.4.2: {}
-
- acorn@8.13.0: {}
-
- agent-base@6.0.2:
- dependencies:
- debug: 4.3.4
- transitivePeerDependencies:
- - supports-color
- optional: true
-
- ajv@8.17.1:
- dependencies:
- fast-deep-equal: 3.1.3
- fast-uri: 3.0.3
- json-schema-traverse: 1.0.0
- require-from-string: 2.0.2
-
- ansi-regex@5.0.1: {}
-
- ansi-styles@3.2.1:
- dependencies:
- color-convert: 1.9.3
-
- ansi-styles@4.3.0:
- dependencies:
- color-convert: 2.0.1
-
- anymatch@https://registry.npmmirror.com/anymatch/-/anymatch-3.1.3.tgz:
- dependencies:
- normalize-path: https://registry.npmmirror.com/normalize-path/-/normalize-path-3.0.0.tgz
- picomatch: https://registry.npmmirror.com/picomatch/-/picomatch-2.3.1.tgz
-
- aproba@2.0.0:
- optional: true
-
- are-we-there-yet@2.0.0:
- dependencies:
- delegates: 1.0.0
- readable-stream: 3.6.0
- optional: true
-
- argparse@1.0.10:
- dependencies:
- sprintf-js: 1.0.3
-
- aria-query@5.3.2: {}
-
- array-bounds@1.0.1: {}
-
- array-normalize@1.1.4:
- dependencies:
- array-bounds: 1.0.1
-
- array-timsort@https://registry.npmmirror.com/array-timsort/-/array-timsort-1.0.3.tgz: {}
-
- async-validator@4.2.5: {}
-
- asynckit@0.4.0: {}
-
- atom-material-icons@3.0.0:
- dependencies:
- compression: 1.7.4
- js-yaml: 3.14.1
- polka: 0.5.2
- select-dom: 6.0.2
- sirv: 1.0.19
- sirv-cli: 1.0.14
- optionalDependencies:
- lodash.template: 4.5.0
- minimist: 1.2.7
- transitivePeerDependencies:
- - supports-color
-
- attr-accept@2.2.2: {}
-
- axios@1.7.7:
- dependencies:
- follow-redirects: 1.15.9
- form-data: 4.0.1
- proxy-from-env: 1.1.0
- transitivePeerDependencies:
- - debug
-
- axobject-query@4.1.0: {}
-
- balanced-match@1.0.2:
- optional: true
-
- balanced-match@https://registry.npmmirror.com/balanced-match/-/balanced-match-1.0.2.tgz: {}
-
- binary-extensions@https://registry.npmmirror.com/binary-extensions/-/binary-extensions-2.2.0.tgz: {}
-
- binary-search-bounds@2.0.5: {}
-
- boolbase@1.0.0: {}
-
- brace-expansion@1.1.11:
- dependencies:
- balanced-match: 1.0.2
- concat-map: 0.0.1
- optional: true
-
- brace-expansion@https://registry.npmmirror.com/brace-expansion/-/brace-expansion-2.0.1.tgz:
- dependencies:
- balanced-match: https://registry.npmmirror.com/balanced-match/-/balanced-match-1.0.2.tgz
-
- braces@3.0.2:
- dependencies:
- fill-range: 7.0.1
-
- braces@https://registry.npmmirror.com/braces/-/braces-3.0.2.tgz:
- dependencies:
- fill-range: https://registry.npmmirror.com/fill-range/-/fill-range-7.0.1.tgz
-
- buble@0.19.8:
- dependencies:
- acorn: 6.4.2
- acorn-dynamic-import: 4.0.0(acorn@6.4.2)
- acorn-jsx: 5.3.2(acorn@6.4.2)
- chalk: 2.4.2
- magic-string: 0.25.9
- minimist: 1.2.7
- os-homedir: 2.0.0
- regexpu-core: 4.8.0
-
- bubleify@1.2.1:
- dependencies:
- buble: 0.19.8
-
- bytes@3.0.0: {}
-
- chalk@2.4.2:
- dependencies:
- ansi-styles: 3.2.1
- escape-string-regexp: 1.0.5
- supports-color: 5.5.0
-
- charenc@0.0.2: {}
-
- chart.js@4.4.5:
- dependencies:
- '@kurkle/color': 0.3.2
-
- chartjs-adapter-date-fns@3.0.0(chart.js@4.4.5)(date-fns@4.1.0):
- dependencies:
- chart.js: 4.4.5
- date-fns: 4.1.0
-
- chokidar@4.0.1:
- dependencies:
- readdirp: 4.0.2
-
- chokidar@https://registry.npmmirror.com/chokidar/-/chokidar-3.5.3.tgz:
- dependencies:
- anymatch: https://registry.npmmirror.com/anymatch/-/anymatch-3.1.3.tgz
- braces: https://registry.npmmirror.com/braces/-/braces-3.0.2.tgz
- glob-parent: https://registry.npmmirror.com/glob-parent/-/glob-parent-5.1.2.tgz
- is-binary-path: https://registry.npmmirror.com/is-binary-path/-/is-binary-path-2.1.0.tgz
- is-glob: https://registry.npmmirror.com/is-glob/-/is-glob-4.0.3.tgz
- normalize-path: https://registry.npmmirror.com/normalize-path/-/normalize-path-3.0.0.tgz
- readdirp: https://registry.npmmirror.com/readdirp/-/readdirp-3.6.0.tgz
- optionalDependencies:
- fsevents: 2.3.2
-
- chownr@2.0.0:
- optional: true
-
- clamp@1.0.1: {}
-
- clipboard@2.0.11:
- dependencies:
- good-listener: 1.2.2
- select: 1.1.2
- tiny-emitter: 2.1.0
-
- cliui@8.0.1:
- dependencies:
- string-width: 4.2.3
- strip-ansi: 6.0.1
- wrap-ansi: 7.0.0
-
- code-red@1.0.4:
- dependencies:
- '@jridgewell/sourcemap-codec': 1.5.0
- '@types/estree': 1.0.6
- acorn: 8.13.0
- estree-walker: 3.0.3
- periscopic: 3.1.0
-
- codemirror-wrapped-line-indent@1.0.8(@codemirror/language@6.10.3)(@codemirror/state@6.4.1)(@codemirror/view@6.34.1):
- dependencies:
- '@codemirror/language': 6.10.3
- '@codemirror/state': 6.4.1
- '@codemirror/view': 6.34.1
-
- color-convert@1.9.3:
- dependencies:
- color-name: 1.1.3
-
- color-convert@2.0.1:
- dependencies:
- color-name: 1.1.4
-
- color-name@1.1.3: {}
-
- color-name@1.1.4: {}
-
- color-support@1.1.3:
- optional: true
-
- combined-stream@1.0.8:
- dependencies:
- delayed-stream: 1.0.0
-
- commander@1.1.1:
- dependencies:
- keypress: 0.1.0
-
- commander@7.2.0: {}
-
- commander@9.5.0: {}
-
- comment-json@https://registry.npmmirror.com/comment-json/-/comment-json-4.2.3.tgz:
- dependencies:
- array-timsort: https://registry.npmmirror.com/array-timsort/-/array-timsort-1.0.3.tgz
- core-util-is: https://registry.npmmirror.com/core-util-is/-/core-util-is-1.0.3.tgz
- esprima: https://registry.npmmirror.com/esprima/-/esprima-4.0.1.tgz
- has-own-prop: https://registry.npmmirror.com/has-own-prop/-/has-own-prop-2.0.0.tgz
- repeat-string: https://registry.npmmirror.com/repeat-string/-/repeat-string-1.6.1.tgz
-
- compressible@2.0.18:
- dependencies:
- mime-db: 1.52.0
-
- compression@1.7.4:
- dependencies:
- accepts: 1.3.8
- bytes: 3.0.0
- compressible: 2.0.18
- debug: 2.6.9
- on-headers: 1.0.2
- safe-buffer: 5.1.2
- vary: 1.1.2
- transitivePeerDependencies:
- - supports-color
-
- concat-map@0.0.1:
- optional: true
-
- consola@https://registry.npmmirror.com/consola/-/consola-2.15.3.tgz: {}
-
- console-clear@1.1.1: {}
-
- console-control-strings@1.1.0:
- optional: true
-
- core-js@3.38.1: {}
-
- core-util-is@https://registry.npmmirror.com/core-util-is/-/core-util-is-1.0.3.tgz: {}
-
- crawlab-ui@0.7.0-rc3(dlwfpvqjxhvrevoazxvsg7egqu):
- dependencies:
- '@element-plus/icons-vue': 2.3.1(vue@3.5.12(typescript@https://registry.npmmirror.com/typescript/-/typescript-4.9.3.tgz))
- '@fortawesome/fontawesome-common-types': 6.6.0
- '@fortawesome/fontawesome-svg-core': 6.6.0
- '@fortawesome/free-brands-svg-icons': 6.6.0
- '@fortawesome/free-regular-svg-icons': 6.6.0
- '@fortawesome/free-solid-svg-icons': 6.6.0
- '@fortawesome/vue-fontawesome': 3.0.8(@fortawesome/fontawesome-svg-core@6.6.0)(vue@3.5.12(typescript@https://registry.npmmirror.com/typescript/-/typescript-4.9.3.tgz))
- '@lexical/code': 0.16.1
- '@lexical/history': 0.16.1
- '@lexical/html': 0.16.1
- '@lexical/link': 0.16.1
- '@lexical/list': 0.16.1
- '@lexical/markdown': 0.16.1
- '@lexical/rich-text': 0.16.1
- '@lexical/selection': 0.16.1
- '@lexical/table': 0.16.1
- '@lexical/text': 0.16.1
- '@lexical/utils': 0.16.1
- '@popperjs/core': 2.11.8
- '@types/getos': 3.0.4
- '@types/humanize-duration': 3.27.4
- '@types/javascript-time-ago': 2.0.8
- '@types/lodash': 4.17.10
- '@types/md5': 2.3.5
- '@types/pinyin': 2.10.2
- '@types/showdown': 2.0.6
- '@types/three': 0.166.0
- '@types/url-join': 4.0.3
- async-validator: 4.2.5
- atom-material-icons: 3.0.0
- axios: 1.7.7
- chart.js: 4.4.5
- chartjs-adapter-date-fns: 3.0.0(chart.js@4.4.5)(date-fns@4.1.0)
- clipboard: 2.0.11
- core-js: 3.38.1
- crawlab-vue3-dropzone: 3.0.3(vue@3.5.12(typescript@https://registry.npmmirror.com/typescript/-/typescript-4.9.3.tgz))
- cron-parser: 4.9.0
- cronstrue: 2.50.0
- dayjs: 1.11.13
- element-plus: 2.8.5(vue@3.5.12(typescript@https://registry.npmmirror.com/typescript/-/typescript-4.9.3.tgz))
- font-awesome: 4.7.0
- humanize-duration: 3.32.1
- javascript-time-ago: 2.5.11
- json-editor-vue: 0.17.2(@lezer/common@1.2.3)(vue@3.5.12(typescript@https://registry.npmmirror.com/typescript/-/typescript-4.9.3.tgz))
- lexical: 0.16.1
- lodash: 4.17.21
- md5: 2.3.0
- mitt: 3.0.1
- monaco-editor: 0.50.0
- normalize.css: 8.0.1
- os-name: 5.0.1
- pinyin: 2.11.2
- point-cluster: 3.1.8
- prismjs: 1.29.0
- showdown: 2.1.0
- three: 0.166.1
- tiny-invariant: 1.3.3
- url-join: 4.0.1
- url-regex: 5.0.0
- uuid: 10.0.0
- vite-svg-loader: 5.1.0(vue@3.5.12(typescript@https://registry.npmmirror.com/typescript/-/typescript-4.9.3.tgz))
- vue: 3.5.12(typescript@https://registry.npmmirror.com/typescript/-/typescript-4.9.3.tgz)
- vue-chartjs: 5.3.1(chart.js@4.4.5)(vue@3.5.12(typescript@https://registry.npmmirror.com/typescript/-/typescript-4.9.3.tgz))
- vue-clipboard3: 2.0.0
- vue-i18n: 9.13.1(vue@3.5.12(typescript@https://registry.npmmirror.com/typescript/-/typescript-4.9.3.tgz))
- vue-router: 4.4.0(vue@3.5.12(typescript@https://registry.npmmirror.com/typescript/-/typescript-4.9.3.tgz))
- vuex: 4.1.0(vue@3.5.12(typescript@https://registry.npmmirror.com/typescript/-/typescript-4.9.3.tgz))
- transitivePeerDependencies:
- - '@lezer/common'
- - '@vue/composition-api'
-
- crawlab-vue3-dropzone@3.0.3(vue@3.5.12(typescript@https://registry.npmmirror.com/typescript/-/typescript-4.9.3.tgz)):
- dependencies:
- attr-accept: 2.2.2
- file-selector: 0.2.4
- vue: 3.5.12(typescript@https://registry.npmmirror.com/typescript/-/typescript-4.9.3.tgz)
-
- crelt@1.0.6: {}
-
- cron-parser@4.9.0:
- dependencies:
- luxon: 3.5.0
-
- cronstrue@2.50.0: {}
-
- cross-spawn@7.0.3:
- dependencies:
- path-key: 3.1.1
- shebang-command: 2.0.0
- which: 2.0.2
-
- crypt@0.0.2: {}
-
- css-select@5.1.0:
- dependencies:
- boolbase: 1.0.0
- css-what: 6.1.0
- domhandler: 5.0.3
- domutils: 3.1.0
- nth-check: 2.1.1
-
- css-tree@2.2.1:
- dependencies:
- mdn-data: 2.0.28
- source-map-js: 1.2.1
-
- css-tree@2.3.1:
- dependencies:
- mdn-data: 2.0.30
- source-map-js: 1.2.1
-
- css-what@6.1.0: {}
-
- csso@5.0.5:
- dependencies:
- css-tree: 2.2.1
-
- csstype@3.1.3: {}
-
- date-fns@4.1.0: {}
-
- dayjs@1.11.13: {}
-
- de-indent@https://registry.npmmirror.com/de-indent/-/de-indent-1.0.2.tgz: {}
-
- debug@2.6.9:
- dependencies:
- ms: 2.0.0
-
- debug@4.3.4:
- dependencies:
- ms: 2.1.2
- optional: true
-
- define-lazy-prop@2.0.0: {}
-
- defined@1.0.1: {}
-
- delayed-stream@1.0.0: {}
-
- delegate@3.2.0: {}
-
- delegates@1.0.0:
- optional: true
-
- detect-libc@1.0.3: {}
-
- detect-libc@2.0.1:
- optional: true
-
- diff-sequences@29.6.3: {}
-
- dom-serializer@2.0.0:
- dependencies:
- domelementtype: 2.3.0
- domhandler: 5.0.3
- entities: 4.5.0
-
- domelementtype@2.3.0: {}
-
- domhandler@5.0.3:
- dependencies:
- domelementtype: 2.3.0
-
- domutils@3.1.0:
- dependencies:
- dom-serializer: 2.0.0
- domelementtype: 2.3.0
- domhandler: 5.0.3
-
- dtype@2.0.0: {}
-
- element-plus@2.8.5(vue@3.5.12(typescript@https://registry.npmmirror.com/typescript/-/typescript-4.9.3.tgz)):
- dependencies:
- '@ctrl/tinycolor': 3.4.1
- '@element-plus/icons-vue': 2.3.1(vue@3.5.12(typescript@https://registry.npmmirror.com/typescript/-/typescript-4.9.3.tgz))
- '@floating-ui/dom': 1.6.11
- '@popperjs/core': '@sxzz/popperjs-es@2.11.7'
- '@types/lodash': 4.17.10
- '@types/lodash-es': 4.17.12
- '@vueuse/core': 9.13.0(vue@3.5.12(typescript@https://registry.npmmirror.com/typescript/-/typescript-4.9.3.tgz))
- async-validator: 4.2.5
- dayjs: 1.11.13
- escape-html: 1.0.3
- lodash: 4.17.21
- lodash-es: 4.17.21
- lodash-unified: 1.0.3(@types/lodash-es@4.17.12)(lodash-es@4.17.21)(lodash@4.17.21)
- memoize-one: 6.0.0
- normalize-wheel-es: 1.2.0
- vue: 3.5.12(typescript@https://registry.npmmirror.com/typescript/-/typescript-4.9.3.tgz)
- transitivePeerDependencies:
- - '@vue/composition-api'
-
- emoji-regex@8.0.0: {}
-
- entities@4.5.0: {}
-
- esbuild-android-64@0.15.16:
- optional: true
-
- esbuild-android-arm64@0.15.16:
- optional: true
-
- esbuild-darwin-64@0.15.16:
- optional: true
-
- esbuild-darwin-arm64@0.15.16:
- optional: true
-
- esbuild-freebsd-64@0.15.16:
- optional: true
-
- esbuild-freebsd-arm64@0.15.16:
- optional: true
-
- esbuild-linux-32@0.15.16:
- optional: true
-
- esbuild-linux-64@0.15.16:
- optional: true
-
- esbuild-linux-arm64@0.15.16:
- optional: true
-
- esbuild-linux-arm@0.15.16:
- optional: true
-
- esbuild-linux-mips64le@0.15.16:
- optional: true
-
- esbuild-linux-ppc64le@0.15.16:
- optional: true
-
- esbuild-linux-riscv64@0.15.16:
- optional: true
-
- esbuild-linux-s390x@0.15.16:
- optional: true
-
- esbuild-netbsd-64@0.15.16:
- optional: true
-
- esbuild-openbsd-64@0.15.16:
- optional: true
-
- esbuild-sunos-64@0.15.16:
- optional: true
-
- esbuild-windows-32@0.15.16:
- optional: true
-
- esbuild-windows-64@0.15.16:
- optional: true
-
- esbuild-windows-arm64@0.15.16:
- optional: true
-
- esbuild@https://registry.npmmirror.com/esbuild/-/esbuild-0.15.16.tgz:
- optionalDependencies:
- '@esbuild/android-arm': 0.15.16
- '@esbuild/linux-loong64': 0.15.16
- esbuild-android-64: 0.15.16
- esbuild-android-arm64: 0.15.16
- esbuild-darwin-64: 0.15.16
- esbuild-darwin-arm64: 0.15.16
- esbuild-freebsd-64: 0.15.16
- esbuild-freebsd-arm64: 0.15.16
- esbuild-linux-32: 0.15.16
- esbuild-linux-64: 0.15.16
- esbuild-linux-arm: 0.15.16
- esbuild-linux-arm64: 0.15.16
- esbuild-linux-mips64le: 0.15.16
- esbuild-linux-ppc64le: 0.15.16
- esbuild-linux-riscv64: 0.15.16
- esbuild-linux-s390x: 0.15.16
- esbuild-netbsd-64: 0.15.16
- esbuild-openbsd-64: 0.15.16
- esbuild-sunos-64: 0.15.16
- esbuild-windows-32: 0.15.16
- esbuild-windows-64: 0.15.16
- esbuild-windows-arm64: 0.15.16
-
- escalade@3.1.1: {}
-
- escape-html@1.0.3: {}
-
- escape-string-regexp@1.0.5: {}
-
- esprima@4.0.1: {}
-
- esprima@https://registry.npmmirror.com/esprima/-/esprima-4.0.1.tgz: {}
-
- estree-walker@2.0.2: {}
-
- estree-walker@3.0.3:
- dependencies:
- '@types/estree': 1.0.0
-
- estree-walker@https://registry.npmmirror.com/estree-walker/-/estree-walker-2.0.2.tgz: {}
-
- execa@5.1.1:
- dependencies:
- cross-spawn: 7.0.3
- get-stream: 6.0.1
- human-signals: 2.1.0
- is-stream: 2.0.1
- merge-stream: 2.0.0
- npm-run-path: 4.0.1
- onetime: 5.1.2
- signal-exit: 3.0.7
- strip-final-newline: 2.0.0
-
- fast-deep-equal@3.1.3: {}
-
- fast-glob@https://registry.npmmirror.com/fast-glob/-/fast-glob-3.2.12.tgz:
- dependencies:
- '@nodelib/fs.stat': https://registry.npmmirror.com/@nodelib/fs.stat/-/fs.stat-2.0.5.tgz
- '@nodelib/fs.walk': https://registry.npmmirror.com/@nodelib/fs.walk/-/fs.walk-1.2.8.tgz
- glob-parent: https://registry.npmmirror.com/glob-parent/-/glob-parent-5.1.2.tgz
- merge2: https://registry.npmmirror.com/merge2/-/merge2-1.4.1.tgz
- micromatch: https://registry.npmmirror.com/micromatch/-/micromatch-4.0.5.tgz
-
- fast-uri@3.0.3: {}
-
- fastq@https://registry.npmmirror.com/fastq/-/fastq-1.14.0.tgz:
- dependencies:
- reusify: https://registry.npmmirror.com/reusify/-/reusify-1.0.4.tgz
-
- fflate@0.8.2: {}
-
- file-selector@0.2.4:
- dependencies:
- tslib: 2.4.1
-
- fill-range@7.0.1:
- dependencies:
- to-regex-range: 5.0.1
-
- fill-range@https://registry.npmmirror.com/fill-range/-/fill-range-7.0.1.tgz:
- dependencies:
- to-regex-range: https://registry.npmmirror.com/to-regex-range/-/to-regex-range-5.0.1.tgz
-
- flatten-vertex-data@1.0.2:
- dependencies:
- dtype: 2.0.0
-
- follow-redirects@1.15.9: {}
-
- font-awesome@4.7.0: {}
-
- form-data@4.0.1:
- dependencies:
- asynckit: 0.4.0
- combined-stream: 1.0.8
- mime-types: 2.1.35
-
- fs-minipass@2.1.0:
- dependencies:
- minipass: 3.3.6
- optional: true
-
- fs.realpath@1.0.0:
- optional: true
-
- fsevents@2.3.2:
- optional: true
-
- function-bind@https://registry.npmmirror.com/function-bind/-/function-bind-1.1.1.tgz: {}
-
- gauge@3.0.2:
- dependencies:
- aproba: 2.0.0
- color-support: 1.1.3
- console-control-strings: 1.1.0
- has-unicode: 2.0.1
- object-assign: 4.1.1
- signal-exit: 3.0.7
- string-width: 4.2.3
- strip-ansi: 6.0.1
- wide-align: 1.1.5
- optional: true
-
- get-caller-file@2.0.5: {}
-
- get-port@3.2.0: {}
-
- get-stream@6.0.1: {}
-
- glob-parent@https://registry.npmmirror.com/glob-parent/-/glob-parent-5.1.2.tgz:
- dependencies:
- is-glob: https://registry.npmmirror.com/is-glob/-/is-glob-4.0.3.tgz
-
- glob@7.2.3:
- dependencies:
- fs.realpath: 1.0.0
- inflight: 1.0.6
- inherits: 2.0.4
- minimatch: 3.1.2
- once: 1.4.0
- path-is-absolute: 1.0.1
- optional: true
-
- good-listener@1.2.2:
- dependencies:
- delegate: 3.2.0
-
- has-flag@3.0.0: {}
-
- has-own-prop@https://registry.npmmirror.com/has-own-prop/-/has-own-prop-2.0.0.tgz: {}
-
- has-unicode@2.0.1:
- optional: true
-
- has@https://registry.npmmirror.com/has/-/has-1.0.3.tgz:
- dependencies:
- function-bind: https://registry.npmmirror.com/function-bind/-/function-bind-1.1.1.tgz
-
- he@https://registry.npmmirror.com/he/-/he-1.2.0.tgz: {}
-
- https-proxy-agent@5.0.1:
- dependencies:
- agent-base: 6.0.2
- debug: 4.3.4
- transitivePeerDependencies:
- - supports-color
- optional: true
-
- human-signals@2.1.0: {}
-
- humanize-duration@3.32.1: {}
-
- immutable-json-patch@6.0.1: {}
-
- immutable@4.3.7: {}
-
- inflight@1.0.6:
- dependencies:
- once: 1.4.0
- wrappy: 1.0.2
- optional: true
-
- inherits@2.0.4:
- optional: true
-
- ip-regex@4.3.0: {}
-
- is-binary-path@https://registry.npmmirror.com/is-binary-path/-/is-binary-path-2.1.0.tgz:
- dependencies:
- binary-extensions: https://registry.npmmirror.com/binary-extensions/-/binary-extensions-2.2.0.tgz
-
- is-buffer@1.1.6: {}
-
- is-core-module@https://registry.npmmirror.com/is-core-module/-/is-core-module-2.11.0.tgz:
- dependencies:
- has: https://registry.npmmirror.com/has/-/has-1.0.3.tgz
-
- is-docker@2.2.1: {}
-
- is-extglob@2.1.1: {}
-
- is-extglob@https://registry.npmmirror.com/is-extglob/-/is-extglob-2.1.1.tgz: {}
-
- is-fullwidth-code-point@3.0.0: {}
-
- is-glob@4.0.3:
- dependencies:
- is-extglob: 2.1.1
-
- is-glob@https://registry.npmmirror.com/is-glob/-/is-glob-4.0.3.tgz:
- dependencies:
- is-extglob: https://registry.npmmirror.com/is-extglob/-/is-extglob-2.1.1.tgz
-
- is-number@7.0.0: {}
-
- is-number@https://registry.npmmirror.com/is-number/-/is-number-7.0.0.tgz: {}
-
- is-obj@1.0.1: {}
-
- is-reference@3.0.1:
- dependencies:
- '@types/estree': 1.0.0
-
- is-stream@2.0.1: {}
-
- is-wsl@2.2.0:
- dependencies:
- is-docker: 2.2.1
-
- isexe@2.0.0: {}
-
- javascript-time-ago@2.5.11:
- dependencies:
- relative-time-format: 1.1.6
-
- jmespath@0.16.0: {}
-
- js-yaml@3.14.1:
- dependencies:
- argparse: 1.0.10
- esprima: 4.0.1
-
- jsep@1.3.9: {}
-
- jsesc@0.5.0: {}
-
- json-editor-vue@0.17.2(@lezer/common@1.2.3)(vue@3.5.12(typescript@https://registry.npmmirror.com/typescript/-/typescript-4.9.3.tgz)):
- dependencies:
- vanilla-jsoneditor: 1.0.8(@lezer/common@1.2.3)
- vue: 3.5.12(typescript@https://registry.npmmirror.com/typescript/-/typescript-4.9.3.tgz)
- vue-demi: 0.14.10(vue@3.5.12(typescript@https://registry.npmmirror.com/typescript/-/typescript-4.9.3.tgz))
- transitivePeerDependencies:
- - '@lezer/common'
-
- json-schema-traverse@1.0.0: {}
-
- json-source-map@0.6.1: {}
-
- jsonpath-plus@10.0.1:
- dependencies:
- '@jsep-plugin/assignment': 1.2.1(jsep@1.3.9)
- '@jsep-plugin/regex': 1.0.3(jsep@1.3.9)
- jsep: 1.3.9
-
- jsonrepair@3.8.1: {}
-
- keypress@0.1.0: {}
-
- kleur@3.0.3: {}
-
- lexical@0.16.1: {}
-
- local-access@1.1.0: {}
-
- locate-character@3.0.0: {}
-
- lodash-es@4.17.21: {}
-
- lodash-unified@1.0.3(@types/lodash-es@4.17.12)(lodash-es@4.17.21)(lodash@4.17.21):
- dependencies:
- '@types/lodash-es': 4.17.12
- lodash: 4.17.21
- lodash-es: 4.17.21
-
- lodash._reinterpolate@3.0.0:
- optional: true
-
- lodash.template@4.5.0:
- dependencies:
- lodash._reinterpolate: 3.0.0
- lodash.templatesettings: 4.2.0
- optional: true
-
- lodash.templatesettings@4.2.0:
- dependencies:
- lodash._reinterpolate: 3.0.0
- optional: true
-
- lodash@4.17.21: {}
-
- lru-cache@6.0.0:
- dependencies:
- yallist: 4.0.0
- optional: true
-
- luxon@3.5.0: {}
-
- macos-release@3.1.0: {}
-
- magic-string@0.25.9:
- dependencies:
- sourcemap-codec: 1.4.8
-
- magic-string@0.30.1:
- dependencies:
- '@jridgewell/sourcemap-codec': 1.4.15
-
- magic-string@0.30.12:
- dependencies:
- '@jridgewell/sourcemap-codec': 1.5.0
-
- magic-string@https://registry.npmmirror.com/magic-string/-/magic-string-0.25.9.tgz:
- dependencies:
- sourcemap-codec: https://registry.npmmirror.com/sourcemap-codec/-/sourcemap-codec-1.4.8.tgz
-
- make-dir@3.1.0:
- dependencies:
- semver: 6.3.0
- optional: true
-
- matchit@1.1.0:
- dependencies:
- '@arr/every': 1.0.1
-
- math-log2@1.0.1: {}
-
- md5@2.3.0:
- dependencies:
- charenc: 0.0.2
- crypt: 0.0.2
- is-buffer: 1.1.6
-
- mdn-data@2.0.28: {}
-
- mdn-data@2.0.30: {}
-
- memoize-one@6.0.0: {}
-
- merge-stream@2.0.0: {}
-
- merge2@https://registry.npmmirror.com/merge2/-/merge2-1.4.1.tgz: {}
-
- meshoptimizer@0.18.1: {}
-
- micromatch@4.0.5:
- dependencies:
- braces: 3.0.2
- picomatch: 2.3.1
-
- micromatch@https://registry.npmmirror.com/micromatch/-/micromatch-4.0.5.tgz:
- dependencies:
- braces: https://registry.npmmirror.com/braces/-/braces-3.0.2.tgz
- picomatch: https://registry.npmmirror.com/picomatch/-/picomatch-2.3.1.tgz
-
- mime-db@1.52.0: {}
-
- mime-types@2.1.35:
- dependencies:
- mime-db: 1.52.0
-
- mimic-fn@2.1.0: {}
-
- minimatch@3.1.2:
- dependencies:
- brace-expansion: 1.1.11
- optional: true
-
- minimatch@https://registry.npmmirror.com/minimatch/-/minimatch-5.1.1.tgz:
- dependencies:
- brace-expansion: https://registry.npmmirror.com/brace-expansion/-/brace-expansion-2.0.1.tgz
-
- minimist@1.2.7: {}
-
- minipass@3.3.6:
- dependencies:
- yallist: 4.0.0
- optional: true
-
- minizlib@2.1.2:
- dependencies:
- minipass: 3.3.6
- yallist: 4.0.0
- optional: true
-
- mitt@3.0.1: {}
-
- mkdirp@1.0.4:
- optional: true
-
- monaco-editor@0.50.0: {}
-
- mri@1.2.0: {}
-
- mrmime@1.0.1: {}
-
- ms@2.0.0: {}
-
- ms@2.1.2:
- optional: true
-
- muggle-string@https://registry.npmmirror.com/muggle-string/-/muggle-string-0.1.0.tgz: {}
-
- nanoid@3.3.7: {}
-
- nanoid@https://registry.npmmirror.com/nanoid/-/nanoid-3.3.4.tgz: {}
-
- natural-compare-lite@1.4.0: {}
-
- negotiator@0.6.3: {}
-
- node-addon-api@3.2.1:
- optional: true
-
- node-addon-api@7.1.1: {}
-
- node-fetch@2.6.7:
- dependencies:
- whatwg-url: 5.0.0
- optional: true
-
- nodejieba@2.5.2:
- dependencies:
- '@mapbox/node-pre-gyp': 1.0.10
- node-addon-api: 3.2.1
- transitivePeerDependencies:
- - encoding
- - supports-color
- optional: true
-
- nopt@5.0.0:
- dependencies:
- abbrev: 1.1.1
- optional: true
-
- normalize-path@https://registry.npmmirror.com/normalize-path/-/normalize-path-3.0.0.tgz: {}
-
- normalize-wheel-es@1.2.0: {}
-
- normalize.css@8.0.1: {}
-
- npm-run-path@4.0.1:
- dependencies:
- path-key: 3.1.1
-
- npmlog@5.0.1:
- dependencies:
- are-we-there-yet: 2.0.0
- console-control-strings: 1.1.0
- gauge: 3.0.2
- set-blocking: 2.0.0
- optional: true
-
- nth-check@2.1.1:
- dependencies:
- boolbase: 1.0.0
-
- object-assign@4.1.1: {}
-
- on-headers@1.0.2: {}
-
- once@1.4.0:
- dependencies:
- wrappy: 1.0.2
- optional: true
-
- onetime@5.1.2:
- dependencies:
- mimic-fn: 2.1.0
-
- open@8.4.2:
- dependencies:
- define-lazy-prop: 2.0.0
- is-docker: 2.2.1
- is-wsl: 2.2.0
-
- os-homedir@2.0.0: {}
-
- os-name@5.0.1:
- dependencies:
- macos-release: 3.1.0
- windows-release: 5.0.1
-
- parse-rect@1.2.0:
- dependencies:
- pick-by-alias: 1.2.0
-
- path-is-absolute@1.0.1:
- optional: true
-
- path-key@3.1.1: {}
-
- path-parse@https://registry.npmmirror.com/path-parse/-/path-parse-1.0.7.tgz: {}
-
- periscopic@3.1.0:
- dependencies:
- '@types/estree': 1.0.6
- estree-walker: 3.0.3
- is-reference: 3.0.1
-
- pick-by-alias@1.2.0: {}
-
- picocolors@1.1.1: {}
-
- picocolors@https://registry.npmmirror.com/picocolors/-/picocolors-1.0.0.tgz: {}
-
- picomatch@2.3.1: {}
-
- picomatch@https://registry.npmmirror.com/picomatch/-/picomatch-2.3.1.tgz: {}
-
- pinyin@2.11.2:
- dependencies:
- commander: 1.1.1
- object-assign: 4.1.1
- optionalDependencies:
- nodejieba: 2.5.2
- transitivePeerDependencies:
- - encoding
- - supports-color
-
- point-cluster@3.1.8:
- dependencies:
- array-bounds: 1.0.1
- array-normalize: 1.1.4
- binary-search-bounds: 2.0.5
- bubleify: 1.2.1
- clamp: 1.0.1
- defined: 1.0.1
- dtype: 2.0.0
- flatten-vertex-data: 1.0.2
- is-obj: 1.0.1
- math-log2: 1.0.1
- parse-rect: 1.2.0
- pick-by-alias: 1.2.0
-
- polka@0.5.2:
- dependencies:
- '@polka/url': 0.5.0
- trouter: 2.0.1
-
- postcss@8.4.47:
- dependencies:
- nanoid: 3.3.7
- picocolors: 1.1.1
- source-map-js: 1.2.1
-
- postcss@https://registry.npmmirror.com/postcss/-/postcss-8.4.19.tgz:
- dependencies:
- nanoid: https://registry.npmmirror.com/nanoid/-/nanoid-3.3.4.tgz
- picocolors: https://registry.npmmirror.com/picocolors/-/picocolors-1.0.0.tgz
- source-map-js: https://registry.npmmirror.com/source-map-js/-/source-map-js-1.0.2.tgz
-
- prismjs@1.29.0: {}
-
- proxy-from-env@1.1.0: {}
-
- queue-microtask@https://registry.npmmirror.com/queue-microtask/-/queue-microtask-1.2.3.tgz: {}
-
- readable-stream@3.6.0:
- dependencies:
- inherits: 2.0.4
- string_decoder: 1.3.0
- util-deprecate: 1.0.2
- optional: true
-
- readdirp@4.0.2: {}
-
- readdirp@https://registry.npmmirror.com/readdirp/-/readdirp-3.6.0.tgz:
- dependencies:
- picomatch: https://registry.npmmirror.com/picomatch/-/picomatch-2.3.1.tgz
-
- regenerate-unicode-properties@9.0.0:
- dependencies:
- regenerate: 1.4.2
-
- regenerate@1.4.2: {}
-
- regexpu-core@4.8.0:
- dependencies:
- regenerate: 1.4.2
- regenerate-unicode-properties: 9.0.0
- regjsgen: 0.5.2
- regjsparser: 0.7.0
- unicode-match-property-ecmascript: 2.0.0
- unicode-match-property-value-ecmascript: 2.1.0
-
- regjsgen@0.5.2: {}
-
- regjsparser@0.7.0:
- dependencies:
- jsesc: 0.5.0
-
- relative-time-format@1.1.6: {}
-
- repeat-string@https://registry.npmmirror.com/repeat-string/-/repeat-string-1.6.1.tgz: {}
-
- require-directory@2.1.1: {}
-
- require-from-string@2.0.2: {}
-
- resolve@https://registry.npmmirror.com/resolve/-/resolve-1.22.1.tgz:
- dependencies:
- is-core-module: https://registry.npmmirror.com/is-core-module/-/is-core-module-2.11.0.tgz
- path-parse: https://registry.npmmirror.com/path-parse/-/path-parse-1.0.7.tgz
- supports-preserve-symlinks-flag: https://registry.npmmirror.com/supports-preserve-symlinks-flag/-/supports-preserve-symlinks-flag-1.0.0.tgz
-
- reusify@https://registry.npmmirror.com/reusify/-/reusify-1.0.4.tgz: {}
-
- rimraf@3.0.2:
- dependencies:
- glob: 7.2.3
- optional: true
-
- rollup-plugin-external-globals@0.8.0(rollup@3.26.2):
- dependencies:
- '@rollup/pluginutils': 5.0.2(rollup@3.26.2)
- estree-walker: 3.0.3
- is-reference: 3.0.1
- magic-string: 0.30.1
- rollup: 3.26.2
-
- rollup-plugin-visualizer@5.9.2(rollup@3.26.2):
- dependencies:
- open: 8.4.2
- picomatch: 2.3.1
- source-map: 0.7.4
- yargs: 17.7.2
- optionalDependencies:
- rollup: 3.26.2
-
- rollup@3.26.2:
- optionalDependencies:
- fsevents: 2.3.2
-
- rollup@https://registry.npmmirror.com/rollup/-/rollup-2.79.1.tgz:
- optionalDependencies:
- fsevents: 2.3.2
-
- run-parallel@https://registry.npmmirror.com/run-parallel/-/run-parallel-1.2.0.tgz:
- dependencies:
- queue-microtask: https://registry.npmmirror.com/queue-microtask/-/queue-microtask-1.2.3.tgz
-
- sade@1.8.1:
- dependencies:
- mri: 1.2.0
-
- safe-buffer@5.1.2: {}
-
- safe-buffer@5.2.1:
- optional: true
-
- sass@1.80.1:
- dependencies:
- '@parcel/watcher': 2.4.1
- chokidar: 4.0.1
- immutable: 4.3.7
- source-map-js: 1.2.1
-
- select-dom@6.0.2: {}
-
- select@1.1.2: {}
-
- semiver@1.1.0: {}
-
- semver@6.3.0:
- optional: true
-
- semver@7.3.8:
- dependencies:
- lru-cache: 6.0.0
- optional: true
-
- set-blocking@2.0.0:
- optional: true
-
- shebang-command@2.0.0:
- dependencies:
- shebang-regex: 3.0.0
-
- shebang-regex@3.0.0: {}
-
- showdown@2.1.0:
- dependencies:
- commander: 9.5.0
-
- signal-exit@3.0.7: {}
-
- sirv-cli@1.0.14:
- dependencies:
- console-clear: 1.1.1
- get-port: 3.2.0
- kleur: 3.0.3
- local-access: 1.1.0
- sade: 1.8.1
- semiver: 1.1.0
- sirv: 1.0.19
- tinydate: 1.3.0
-
- sirv@1.0.19:
- dependencies:
- '@polka/url': 1.0.0-next.21
- mrmime: 1.0.1
- totalist: 1.1.0
-
- source-map-js@1.0.2: {}
-
- source-map-js@1.2.1: {}
-
- source-map-js@https://registry.npmmirror.com/source-map-js/-/source-map-js-1.0.2.tgz: {}
-
- source-map@0.7.4: {}
-
- source-map@https://registry.npmmirror.com/source-map/-/source-map-0.6.1.tgz: {}
-
- sourcemap-codec@1.4.8: {}
-
- sourcemap-codec@https://registry.npmmirror.com/sourcemap-codec/-/sourcemap-codec-1.4.8.tgz: {}
-
- sprintf-js@1.0.3: {}
-
- string-width@4.2.3:
- dependencies:
- emoji-regex: 8.0.0
- is-fullwidth-code-point: 3.0.0
- strip-ansi: 6.0.1
-
- string_decoder@1.3.0:
- dependencies:
- safe-buffer: 5.2.1
- optional: true
-
- strip-ansi@6.0.1:
- dependencies:
- ansi-regex: 5.0.1
-
- strip-final-newline@2.0.0: {}
-
- style-mod@4.1.2: {}
-
- supports-color@5.5.0:
- dependencies:
- has-flag: 3.0.0
-
- supports-preserve-symlinks-flag@https://registry.npmmirror.com/supports-preserve-symlinks-flag/-/supports-preserve-symlinks-flag-1.0.0.tgz: {}
-
- svelte@4.2.19:
- dependencies:
- '@ampproject/remapping': 2.3.0
- '@jridgewell/sourcemap-codec': 1.5.0
- '@jridgewell/trace-mapping': 0.3.25
- '@types/estree': 1.0.6
- acorn: 8.13.0
- aria-query: 5.3.2
- axobject-query: 4.1.0
- code-red: 1.0.4
- css-tree: 2.3.1
- estree-walker: 3.0.3
- is-reference: 3.0.1
- locate-character: 3.0.0
- magic-string: 0.30.12
- periscopic: 3.1.0
-
- svgo@3.3.2:
- dependencies:
- '@trysound/sax': 0.2.0
- commander: 7.2.0
- css-select: 5.1.0
- css-tree: 2.3.1
- css-what: 6.1.0
- csso: 5.0.5
- picocolors: 1.1.1
-
- tar@6.1.12:
- dependencies:
- chownr: 2.0.0
- fs-minipass: 2.1.0
- minipass: 3.3.6
- minizlib: 2.1.2
- mkdirp: 1.0.4
- yallist: 4.0.0
- optional: true
-
- three@0.166.1: {}
-
- tiny-emitter@2.1.0: {}
-
- tiny-invariant@1.3.3: {}
-
- tinydate@1.3.0: {}
-
- tlds@1.255.0: {}
-
- to-fast-properties@2.0.0: {}
-
- to-fast-properties@https://registry.npmmirror.com/to-fast-properties/-/to-fast-properties-2.0.0.tgz: {}
-
- to-regex-range@5.0.1:
- dependencies:
- is-number: 7.0.0
-
- to-regex-range@https://registry.npmmirror.com/to-regex-range/-/to-regex-range-5.0.1.tgz:
- dependencies:
- is-number: https://registry.npmmirror.com/is-number/-/is-number-7.0.0.tgz
-
- totalist@1.1.0: {}
-
- tr46@0.0.3:
- optional: true
-
- trouter@2.0.1:
- dependencies:
- matchit: 1.1.0
-
- tslib@2.4.1: {}
-
- typescript@https://registry.npmmirror.com/typescript/-/typescript-4.9.3.tgz: {}
-
- unicode-canonical-property-names-ecmascript@2.0.0: {}
-
- unicode-match-property-ecmascript@2.0.0:
- dependencies:
- unicode-canonical-property-names-ecmascript: 2.0.0
- unicode-property-aliases-ecmascript: 2.1.0
-
- unicode-match-property-value-ecmascript@2.1.0: {}
-
- unicode-property-aliases-ecmascript@2.1.0: {}
-
- url-join@4.0.1: {}
-
- url-regex@5.0.0:
- dependencies:
- ip-regex: 4.3.0
- tlds: 1.255.0
-
- util-deprecate@1.0.2:
- optional: true
-
- uuid@10.0.0: {}
-
- vanilla-jsoneditor@1.0.8(@lezer/common@1.2.3):
- dependencies:
- '@codemirror/autocomplete': 6.18.1(@codemirror/language@6.10.3)(@codemirror/state@6.4.1)(@codemirror/view@6.34.1)(@lezer/common@1.2.3)
- '@codemirror/commands': 6.7.0
- '@codemirror/lang-json': 6.0.1
- '@codemirror/language': 6.10.3
- '@codemirror/lint': 6.8.2
- '@codemirror/search': 6.5.6
- '@codemirror/state': 6.4.1
- '@codemirror/view': 6.34.1
- '@fortawesome/free-regular-svg-icons': 6.6.0
- '@fortawesome/free-solid-svg-icons': 6.6.0
- '@jsonquerylang/jsonquery': 3.1.1
- '@lezer/highlight': 1.2.1
- '@replit/codemirror-indentation-markers': 6.5.3(@codemirror/language@6.10.3)(@codemirror/state@6.4.1)(@codemirror/view@6.34.1)
- ajv: 8.17.1
- codemirror-wrapped-line-indent: 1.0.8(@codemirror/language@6.10.3)(@codemirror/state@6.4.1)(@codemirror/view@6.34.1)
- diff-sequences: 29.6.3
- immutable-json-patch: 6.0.1
- jmespath: 0.16.0
- json-source-map: 0.6.1
- jsonpath-plus: 10.0.1
- jsonrepair: 3.8.1
- lodash-es: 4.17.21
- memoize-one: 6.0.0
- natural-compare-lite: 1.4.0
- sass: 1.80.1
- svelte: 4.2.19
- vanilla-picker: 2.12.3
- transitivePeerDependencies:
- - '@lezer/common'
-
- vanilla-picker@2.12.3:
- dependencies:
- '@sphinxxxx/color-conversion': 2.2.2
-
- vary@1.1.2: {}
-
- vite-aliases@https://registry.npmmirror.com/vite-aliases/-/vite-aliases-0.9.7.tgz(@types/node@https://registry.npmmirror.com/@types/node/-/node-18.11.10.tgz)(sass@1.80.1):
- dependencies:
- chokidar: https://registry.npmmirror.com/chokidar/-/chokidar-3.5.3.tgz
- comment-json: https://registry.npmmirror.com/comment-json/-/comment-json-4.2.3.tgz
- consola: https://registry.npmmirror.com/consola/-/consola-2.15.3.tgz
- fast-glob: https://registry.npmmirror.com/fast-glob/-/fast-glob-3.2.12.tgz
- vite: https://registry.npmmirror.com/vite/-/vite-3.2.4.tgz(@types/node@https://registry.npmmirror.com/@types/node/-/node-18.11.10.tgz)(sass@1.80.1)
- transitivePeerDependencies:
- - '@types/node'
- - less
- - sass
- - stylus
- - sugarss
- - terser
-
- vite-plugin-dynamic-import@https://registry.npmmirror.com/vite-plugin-dynamic-import/-/vite-plugin-dynamic-import-1.2.4.tgz:
- dependencies:
- fast-glob: https://registry.npmmirror.com/fast-glob/-/fast-glob-3.2.12.tgz
-
- vite-plugin-externalize-deps@0.7.0(vite@https://registry.npmmirror.com/vite/-/vite-3.2.4.tgz(@types/node@https://registry.npmmirror.com/@types/node/-/node-18.11.10.tgz)(sass@1.80.1)):
- dependencies:
- vite: https://registry.npmmirror.com/vite/-/vite-3.2.4.tgz(@types/node@https://registry.npmmirror.com/@types/node/-/node-18.11.10.tgz)(sass@1.80.1)
-
- vite-svg-loader@5.1.0(vue@3.5.12(typescript@https://registry.npmmirror.com/typescript/-/typescript-4.9.3.tgz)):
- dependencies:
- svgo: 3.3.2
- vue: 3.5.12(typescript@https://registry.npmmirror.com/typescript/-/typescript-4.9.3.tgz)
-
- vite@https://registry.npmmirror.com/vite/-/vite-3.2.4.tgz(@types/node@https://registry.npmmirror.com/@types/node/-/node-18.11.10.tgz)(sass@1.80.1):
- dependencies:
- esbuild: https://registry.npmmirror.com/esbuild/-/esbuild-0.15.16.tgz
- postcss: https://registry.npmmirror.com/postcss/-/postcss-8.4.19.tgz
- resolve: https://registry.npmmirror.com/resolve/-/resolve-1.22.1.tgz
- rollup: https://registry.npmmirror.com/rollup/-/rollup-2.79.1.tgz
- optionalDependencies:
- '@types/node': https://registry.npmmirror.com/@types/node/-/node-18.11.10.tgz
- fsevents: 2.3.2
- sass: 1.80.1
-
- vue-chartjs@5.3.1(chart.js@4.4.5)(vue@3.5.12(typescript@https://registry.npmmirror.com/typescript/-/typescript-4.9.3.tgz)):
- dependencies:
- chart.js: 4.4.5
- vue: 3.5.12(typescript@https://registry.npmmirror.com/typescript/-/typescript-4.9.3.tgz)
-
- vue-clipboard3@2.0.0:
- dependencies:
- clipboard: 2.0.11
-
- vue-demi@0.14.10(vue@3.5.12(typescript@https://registry.npmmirror.com/typescript/-/typescript-4.9.3.tgz)):
- dependencies:
- vue: 3.5.12(typescript@https://registry.npmmirror.com/typescript/-/typescript-4.9.3.tgz)
-
- vue-i18n@9.13.1(vue@3.5.12(typescript@https://registry.npmmirror.com/typescript/-/typescript-4.9.3.tgz)):
- dependencies:
- '@intlify/core-base': 9.13.1
- '@intlify/shared': 9.13.1
- '@vue/devtools-api': 6.6.4
- vue: 3.5.12(typescript@https://registry.npmmirror.com/typescript/-/typescript-4.9.3.tgz)
-
- vue-router@4.4.0(vue@3.5.12(typescript@https://registry.npmmirror.com/typescript/-/typescript-4.9.3.tgz)):
- dependencies:
- '@vue/devtools-api': 6.6.4
- vue: 3.5.12(typescript@https://registry.npmmirror.com/typescript/-/typescript-4.9.3.tgz)
-
- vue-template-compiler@https://registry.npmmirror.com/vue-template-compiler/-/vue-template-compiler-2.7.14.tgz:
- dependencies:
- de-indent: https://registry.npmmirror.com/de-indent/-/de-indent-1.0.2.tgz
- he: https://registry.npmmirror.com/he/-/he-1.2.0.tgz
-
- vue-tsc@https://registry.npmmirror.com/vue-tsc/-/vue-tsc-1.0.11.tgz(typescript@https://registry.npmmirror.com/typescript/-/typescript-4.9.3.tgz):
- dependencies:
- '@volar/vue-language-core': https://registry.npmmirror.com/@volar/vue-language-core/-/vue-language-core-1.0.11.tgz
- '@volar/vue-typescript': https://registry.npmmirror.com/@volar/vue-typescript/-/vue-typescript-1.0.11.tgz
- typescript: https://registry.npmmirror.com/typescript/-/typescript-4.9.3.tgz
-
- vue@3.5.12(typescript@https://registry.npmmirror.com/typescript/-/typescript-4.9.3.tgz):
- dependencies:
- '@vue/compiler-dom': 3.5.12
- '@vue/compiler-sfc': 3.5.12
- '@vue/runtime-dom': 3.5.12
- '@vue/server-renderer': 3.5.12(vue@3.5.12(typescript@https://registry.npmmirror.com/typescript/-/typescript-4.9.3.tgz))
- '@vue/shared': 3.5.12
- optionalDependencies:
- typescript: https://registry.npmmirror.com/typescript/-/typescript-4.9.3.tgz
-
- vuex@4.1.0(vue@3.5.12(typescript@https://registry.npmmirror.com/typescript/-/typescript-4.9.3.tgz)):
- dependencies:
- '@vue/devtools-api': 6.4.5
- vue: 3.5.12(typescript@https://registry.npmmirror.com/typescript/-/typescript-4.9.3.tgz)
-
- w3c-keyname@2.2.8: {}
-
- webidl-conversions@3.0.1:
- optional: true
-
- whatwg-url@5.0.0:
- dependencies:
- tr46: 0.0.3
- webidl-conversions: 3.0.1
- optional: true
-
- which@2.0.2:
- dependencies:
- isexe: 2.0.0
-
- wide-align@1.1.5:
- dependencies:
- string-width: 4.2.3
- optional: true
-
- windows-release@5.0.1:
- dependencies:
- execa: 5.1.1
-
- wrap-ansi@7.0.0:
- dependencies:
- ansi-styles: 4.3.0
- string-width: 4.2.3
- strip-ansi: 6.0.1
-
- wrappy@1.0.2:
- optional: true
-
- y18n@5.0.8: {}
-
- yallist@4.0.0:
- optional: true
-
- yargs-parser@21.1.1: {}
-
- yargs@17.7.2:
- dependencies:
- cliui: 8.0.1
- escalade: 3.1.1
- get-caller-file: 2.0.5
- require-directory: 2.1.1
- string-width: 4.2.3
- y18n: 5.0.8
- yargs-parser: 21.1.1
diff --git a/frontend/apps/crawlab/src/main.ts b/frontend/apps/crawlab/src/main.ts
deleted file mode 100644
index c63091ef..00000000
--- a/frontend/apps/crawlab/src/main.ts
+++ /dev/null
@@ -1,7 +0,0 @@
-import 'crawlab-ui/dist/style.css';
-import 'vue';
-import {createApp} from 'crawlab-ui';
-
-(async function () {
- await createApp();
-})();
diff --git a/frontend/apps/crawlab/src/shims-vue.d.ts b/frontend/apps/crawlab/src/shims-vue.d.ts
deleted file mode 100644
index ebcd61d0..00000000
--- a/frontend/apps/crawlab/src/shims-vue.d.ts
+++ /dev/null
@@ -1,6 +0,0 @@
-declare module '*.scss';
-declare module '*.vue' {
- import {DefineComponent} from 'vue';
- const component: DefineComponent<{}, {}, any>;
- export default component;
-}
diff --git a/frontend/apps/crawlab/tsconfig.json b/frontend/apps/crawlab/tsconfig.json
deleted file mode 100644
index 3b5835eb..00000000
--- a/frontend/apps/crawlab/tsconfig.json
+++ /dev/null
@@ -1,41 +0,0 @@
-{
- "compilerOptions": {
- "declaration": true,
- "target": "esnext",
- "module": "esnext",
- "strict": true,
- "jsx": "preserve",
- "importHelpers": true,
- "moduleResolution": "node",
- "skipLibCheck": true,
- "esModuleInterop": true,
- "allowSyntheticDefaultImports": true,
- "sourceMap": true,
- "baseUrl": ".",
- "paths": {
- "@/*": [
- "src/*"
- ]
- },
- "lib": [
- "esnext",
- "dom",
- "dom.iterable",
- "scripthost"
- ],
- "typeRoots": [
- "src/interfaces"
- ]
- },
- "include": [
- "src/shims-vue.d.ts",
- "src/**/*.ts",
- "src/**/*.tsx",
- "src/**/*.vue",
- "__test__/**/*.spec.ts"
- ],
- "exclude": [
- "node_modules",
- "src/**/*.js"
- ]
-}
diff --git a/frontend/apps/crawlab/vite.config.ts b/frontend/apps/crawlab/vite.config.ts
deleted file mode 100644
index 180ffd21..00000000
--- a/frontend/apps/crawlab/vite.config.ts
+++ /dev/null
@@ -1,46 +0,0 @@
-import { resolve } from 'path';
-import { defineConfig, UserConfig } from 'vite';
-import vue from '@vitejs/plugin-vue';
-import dynamicImport from 'vite-plugin-dynamic-import';
-import { visualizer } from 'rollup-plugin-visualizer';
-
-export default defineConfig(({ mode }) => {
- const config: UserConfig = {
- build: {
- rollupOptions: {
- output: {
- manualChunks(id, meta) {
- if (id.includes('node_modules')) {
- const arr = id.toString().split('node_modules/');
- const modulePath = arr[arr.length - 1];
- return modulePath?.split('/')?.[0];
- }
- if (id.includes('three.min.js')) {
- return 'three';
- }
- },
- },
- },
- },
- optimizeDeps: {
- include: ['element-plus', 'axios'],
- },
- resolve: {
- dedupe: ['vue', 'vue-router', 'vuex', 'axios', 'element-plus'],
- alias: {
- '@': resolve(__dirname, 'src'),
- },
- extensions: ['.js', '.ts', '.jsx', '.tsx', '.json', '.vue', '.scss'],
- },
- plugins: [vue(), dynamicImport()],
- server: {
- cors: true,
- },
- };
-
- if (mode === 'analyze') {
- config.plugins.push(visualizer({ open: true, gzipSize: true }));
- }
-
- return config;
-});
diff --git a/frontend/crawlab-ui/.browserslistrc b/frontend/crawlab-ui/.browserslistrc
new file mode 100644
index 00000000..214388fe
--- /dev/null
+++ b/frontend/crawlab-ui/.browserslistrc
@@ -0,0 +1,3 @@
+> 1%
+last 2 versions
+not dead
diff --git a/frontend/crawlab-ui/.dockerignore b/frontend/crawlab-ui/.dockerignore
new file mode 100644
index 00000000..c2658d7d
--- /dev/null
+++ b/frontend/crawlab-ui/.dockerignore
@@ -0,0 +1 @@
+node_modules/
diff --git a/frontend/crawlab-ui/.editorconfig b/frontend/crawlab-ui/.editorconfig
new file mode 100644
index 00000000..ec9574ea
--- /dev/null
+++ b/frontend/crawlab-ui/.editorconfig
@@ -0,0 +1,11 @@
+root = true
+
+[*]
+tab_width = 2
+charset = utf-8
+end_of_line = lf
+indent_size = 2
+indent_style = space
+insert_final_newline = true
+max_line_length = 120
+trim_trailing_whitespace = true
diff --git a/frontend/crawlab-ui/.env b/frontend/crawlab-ui/.env
new file mode 100644
index 00000000..f5303165
--- /dev/null
+++ b/frontend/crawlab-ui/.env
@@ -0,0 +1,2 @@
+VITE_APP_ENV=development
+VITE_APP_API_BASE_URL=http://localhost:8000
diff --git a/frontend/.env.development b/frontend/crawlab-ui/.env.development
similarity index 67%
rename from frontend/.env.development
rename to frontend/crawlab-ui/.env.development
index dd2e38d0..7a6b5c0f 100644
--- a/frontend/.env.development
+++ b/frontend/crawlab-ui/.env.development
@@ -1 +1,2 @@
+NODE_ENV=development
VITE_APP_API_BASE_URL=http://localhost:8000
diff --git a/frontend/crawlab-ui/.env.production b/frontend/crawlab-ui/.env.production
new file mode 100644
index 00000000..46edac91
--- /dev/null
+++ b/frontend/crawlab-ui/.env.production
@@ -0,0 +1,2 @@
+VITE_APP_ENV=production
+VITE_APP_API_BASE_URL=http://localhost:8000
diff --git a/frontend/crawlab-ui/.env.test b/frontend/crawlab-ui/.env.test
new file mode 100644
index 00000000..02404873
--- /dev/null
+++ b/frontend/crawlab-ui/.env.test
@@ -0,0 +1,3 @@
+NODE_ENV=development
+VITE_APP_ENV=test
+VITE_APP_API_BASE_URL=http://localhost:8000
diff --git a/frontend/crawlab-ui/.eslintignore b/frontend/crawlab-ui/.eslintignore
new file mode 100644
index 00000000..b16742a5
--- /dev/null
+++ b/frontend/crawlab-ui/.eslintignore
@@ -0,0 +1,3 @@
+./src/i18n/**/*.ts
+*/**/*.js
+*.js
diff --git a/frontend/crawlab-ui/.gitattributes b/frontend/crawlab-ui/.gitattributes
new file mode 100644
index 00000000..142c2927
--- /dev/null
+++ b/frontend/crawlab-ui/.gitattributes
@@ -0,0 +1,5 @@
+*.ts linguist-detectable=false
+*.css linguist-detectable=false
+*.scss linguist-detectable=false
+*.js linguist-detectable=true
+*.vue linguist-detectable=true
diff --git a/frontend/crawlab-ui/.github/.git_commit_template.txt b/frontend/crawlab-ui/.github/.git_commit_template.txt
new file mode 100644
index 00000000..5040eb39
--- /dev/null
+++ b/frontend/crawlab-ui/.github/.git_commit_template.txt
@@ -0,0 +1,2 @@
+# [TYPE](SCOPE):DESCRIPTION#[ISSUE]
+# example feat(button):add type 'button' for form usage #1234
diff --git a/frontend/crawlab-ui/.github/CONTRIBUTING.en-US.md b/frontend/crawlab-ui/.github/CONTRIBUTING.en-US.md
new file mode 100644
index 00000000..0b97154c
--- /dev/null
+++ b/frontend/crawlab-ui/.github/CONTRIBUTING.en-US.md
@@ -0,0 +1,34 @@
+# Element Plus Contributing Guide
+
+Hi! Thank you for choosing Element Plus.
+
+Element Plus is a Vue 3.0 based component library for developers, designers and product managers.
+
+We are excited that you are interested in contributing to Element Plus. Before submitting your contribution though, please make sure to take a moment and read through the following guidelines.
+
+## Issue Guidelines
+
+- Issues are exclusively for bug reports, feature requests and design-related topics. Other questions may be closed directly. If any questions come up when you are using Element Plus, please hit [Discord](https://discord.link/ElementPlus) for help.
+
+- Before submitting an issue, please check if similar problems have already been issued.
+
+- Please specify which version of `Element Plus` and `Vue` you are using, and provide OS and browser information. [JSFiddle](https://jsfiddle.net/) is recommended to build a live demo so that your issue can be reproduced clearly.
+
+## Pull Request Guidelines
+
+- Fork this repository to your own account. Do not create branches here.
+
+- Commit info should be formatted as `[Component Name]: Info about commit.` (e.g. `Button: Fix xxx bug`)
+
+- **DO NOT** include files inside `lib` directory.
+
+- Make sure that running `npm run build` outputs the correct files.
+
+- Rebase before creating a PR to keep commit history clear.
+
+- Make sure PRs are created to `dev` branch instead of `master` branch.
+
+- If your PR fixes a bug, please provide a description about the related bug.
+
+- Merging a PR takes two maintainers: one approves the changes after reviewing, and then the other reviews and merges.
+
diff --git a/frontend/crawlab-ui/.github/CONTRIBUTING.zh-CN.md b/frontend/crawlab-ui/.github/CONTRIBUTING.zh-CN.md
new file mode 100644
index 00000000..be70646c
--- /dev/null
+++ b/frontend/crawlab-ui/.github/CONTRIBUTING.zh-CN.md
@@ -0,0 +1,31 @@
+# Element Plus 贡献指南
+
+Hi! 首先感谢你使用 Element Plus。
+
+Element Plus 是一套为开发者、设计师和产品经理准备的开源组件库,旨在快速搭建页面。它基于 Vue 3.0 开发,并提供了配套的设计资源,充分满足可定制化的需求。
+
+Element Plus 的成长离不开大家的支持,如果你愿意为 Element Plus 贡献代码或提供建议,请阅读以下内容。
+
+## Issue 规范
+- issue 仅用于提交 Bug 或 Feature 以及设计相关的内容,其它内容可能会被直接关闭。如果你在使用时产生了疑问,请到 Slack 或 [Discord](https://discord.link/ElementPlus) 里咨询。
+
+- 在提交 issue 之前,请搜索相关内容是否已被提出。
+
+- 请说明 Element Plus 和 Vue 的版本号,并提供操作系统和浏览器信息。推荐使用 [JSFiddle](https://jsfiddle.net/) 生成在线 demo,这能够更直观地重现问题。
+
+## Pull Request 规范
+- 请先 fork 一份到自己的项目下,不要直接在仓库下建分支。
+
+- commit 信息要以`[组件名]: 描述信息` 的形式填写,例如 `Button: fix xxx bug`。
+
+- **不要提交** `lib` 里面打包的文件。
+
+- 执行 `npm run build` 后可以正确打包文件。
+
+- 提交 PR 前请 rebase,确保 commit 记录的整洁。
+
+- 确保 PR 是提交到 `dev` 分支,而不是 `master` 分支。
+
+- 如果是修复 bug,请在 PR 中给出描述信息。
+
+- 合并代码需要两名维护人员参与:一人进行 review 后 approve,另一人再次 review,通过后即可合并。
diff --git a/frontend/crawlab-ui/.github/ISSUE_TEMPLATE.md b/frontend/crawlab-ui/.github/ISSUE_TEMPLATE.md
new file mode 100644
index 00000000..446ac677
--- /dev/null
+++ b/frontend/crawlab-ui/.github/ISSUE_TEMPLATE.md
@@ -0,0 +1,15 @@
+
diff --git a/frontend/crawlab-ui/.github/ISSUE_TEMPLATE/config.yml b/frontend/crawlab-ui/.github/ISSUE_TEMPLATE/config.yml
new file mode 100644
index 00000000..77a8fa2b
--- /dev/null
+++ b/frontend/crawlab-ui/.github/ISSUE_TEMPLATE/config.yml
@@ -0,0 +1,5 @@
+blank_issues_enabled: true
+contact_links:
+ - name: Create new issue
+ url: https://elementui.github.io/issue-generator/#/en-US?repo=element-plus
+ about: The issue which is not created via https://elementui.github.io/issue-generator will be closed immediately.
diff --git a/frontend/crawlab-ui/.github/pull_request_template.md b/frontend/crawlab-ui/.github/pull_request_template.md
new file mode 100644
index 00000000..55103bc4
--- /dev/null
+++ b/frontend/crawlab-ui/.github/pull_request_template.md
@@ -0,0 +1,5 @@
+Please make sure these boxes are checked before submitting your PR, thank you!
+
+* [ ] Make sure you follow Element's contributing guide [English](https://github.com/element-plus/element-plus/blob/master/.github/CONTRIBUTING.en-US.md) | ([中文](https://github.com/element-plus/element-plus/blob/master/.github/CONTRIBUTING.zh-CN.md) | [Español](https://github.com/element-plus/element-plus/blob/master/.github/CONTRIBUTING.es.md) | [Français](https://github.com/element-plus/element-plus/blob/master/.github/CONTRIBUTING.fr-FR.md)).
+* [ ] Make sure you are merging your commits to `dev` branch.
+* [ ] Add some descriptions and refer to relative issues for your PR.
diff --git a/frontend/crawlab-ui/.github/workflows/publish-npm.yml b/frontend/crawlab-ui/.github/workflows/publish-npm.yml
new file mode 100644
index 00000000..a1941ca3
--- /dev/null
+++ b/frontend/crawlab-ui/.github/workflows/publish-npm.yml
@@ -0,0 +1,37 @@
+name: Publish to NPM registry
+
+on:
+ pull_request:
+ branches: [ main ]
+ push:
+ branches: [ main ]
+ release:
+ types: [ created ]
+
+jobs:
+ publish:
+ runs-on: ubuntu-latest
+ steps:
+ - uses: actions/checkout@v2
+ - uses: actions/setup-node@v2
+ with:
+ node-version: '20.13.1'
+ registry-url: https://registry.npmjs.com/
+ - name: Get version
+ run: echo "TAG_VERSION=${GITHUB_REF#refs/*/}" >> $GITHUB_ENV
+ - name: Initial setup
+ run: |
+ npm i -g pnpm@9
+ - name: Install dependencies
+ run: pnpm install
+ - name: Build
+ run: pnpm run build
+ env:
+ TAG_VERSION: ${{env.TAG_VERSION}}
+ - if: ${{ github.event_name == 'release' }}
+ name: Publish npm
+ run: npm publish --registry ${REGISTRY}
+ env:
+ NODE_AUTH_TOKEN: ${{secrets.NPM_PUBLISH_TOKEN}}
+ TAG_VERSION: ${{env.TAG_VERSION}}
+ REGISTRY: https://registry.npmjs.com/
diff --git a/frontend/.gitignore b/frontend/crawlab-ui/.gitignore
similarity index 86%
rename from frontend/.gitignore
rename to frontend/crawlab-ui/.gitignore
index 87f50dc6..0e95d095 100644
--- a/frontend/.gitignore
+++ b/frontend/crawlab-ui/.gitignore
@@ -4,7 +4,7 @@ node_modules
# local env files
-.env.local
+.env.development.local
.env.*.local
# Log files
@@ -29,6 +29,8 @@ lib/
**/dist
**/node_modules
**/package-lock.json
+**/pkg
+dist_local/
stats.html
-.turbo
\ No newline at end of file
+.turbo
diff --git a/frontend/crawlab-ui/.prettierrc b/frontend/crawlab-ui/.prettierrc
new file mode 100644
index 00000000..61d6aaae
--- /dev/null
+++ b/frontend/crawlab-ui/.prettierrc
@@ -0,0 +1,12 @@
+{
+ "semi": true,
+ "trailingComma": "es5",
+ "singleQuote": true,
+ "printWidth": 80,
+ "tabWidth": 2,
+ "useTabs": false,
+ "bracketSpacing": true,
+ "jsxBracketSameLine": false,
+ "arrowParens": "avoid",
+ "endOfLine": "lf"
+}
diff --git a/frontend/crawlab-ui/LICENSE b/frontend/crawlab-ui/LICENSE
new file mode 100644
index 00000000..e41311d2
--- /dev/null
+++ b/frontend/crawlab-ui/LICENSE
@@ -0,0 +1,29 @@
+BSD 3-Clause License
+
+Copyright (c) 2020, Crawlab Team
+All rights reserved.
+
+Redistribution and use in source and binary forms, with or without
+modification, are permitted provided that the following conditions are met:
+
+1. Redistributions of source code must retain the above copyright notice, this
+ list of conditions and the following disclaimer.
+
+2. Redistributions in binary form must reproduce the above copyright notice,
+ this list of conditions and the following disclaimer in the documentation
+ and/or other materials provided with the distribution.
+
+3. Neither the name of the copyright holder nor the names of its
+ contributors may be used to endorse or promote products derived from
+ this software without specific prior written permission.
+
+THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
+AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
+DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
+FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
+DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
+SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
+CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
+OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
diff --git a/frontend/crawlab-ui/README.md b/frontend/crawlab-ui/README.md
new file mode 100644
index 00000000..760ac6bd
--- /dev/null
+++ b/frontend/crawlab-ui/README.md
@@ -0,0 +1,126 @@
+# Crawlab-UI
+
+This is the UI components and modules to support the frontend development
+for [Crawlab](https://github.com/crawlab-team/crawlab).
+
+## How to Install
+
+Use `npm` or `yarn` to install `crawlab-ui`.
+
+```
+# npm
+npm install crawlab-ui -S
+
+# or use yarn
+yarn add crawlab-ui -S
+```
+
+## How to Use
+
+It is similar to [Element-Plus](https://github.com/element-plus/element-plus), you can import components from
+Crawlab-UI. Crawlab-UI is built based on Element-Plus so that you can comfortably use it with Element-Plus.
+
+### Use Globally Installed Components
+
+Below is an example of entry file (main.ts) using Crawlab-UI globally.
+
+```ts
+import {createApp} from 'vue';
+import CrawlabUI from 'crawlab-ui';
+
+const app = createApp(App);
+app
+ .use(CrawlabUI) // install globally
+ .mount('#app');
+```
+
+Below is an example of using globally installed Crawlab-UI in a Vue 3 component.
+
+```vue
+
+
+
+
+
+
+
+
+
+
+
+
+```
+
+### Use Standalone Components
+
+Below is an example of using standalone components in a Vue 3 component.
+
+```vue
+
+
+
+
+
+
+
+
+
+
+
+
+
+```
+
+### Use Web Application
+
+Crawlab-UI has a built-in web application for Crawlab frontend. You can simply use it to start Crawlab frontend Vue 3
+SPA.
+
+```ts
+// index.ts or index.js or other entry file
+import {createApp} from 'crawlab-ui';
+
+createApp();
+```
+
+And that's it! After you build or start serving it, you can view the Crawlab frontend web application in the browser.
+
+## Development
+
+### Add a new model
+
+Related directories:
+
+- `src/views`
+- `src/components`
+- `src/store`
+- `src/router`
+- `src/i18n`
+- `src/services`
+- `src/interfaces/i18n`
diff --git a/frontend/apps/crawlab/index.html b/frontend/crawlab-ui/index.html
similarity index 61%
rename from frontend/apps/crawlab/index.html
rename to frontend/crawlab-ui/index.html
index 8e829861..1b705548 100644
--- a/frontend/apps/crawlab/index.html
+++ b/frontend/crawlab-ui/index.html
@@ -4,16 +4,16 @@
-
+
Crawlab
-
-
+
+
-
+
diff --git a/frontend/crawlab-ui/package.json b/frontend/crawlab-ui/package.json
new file mode 100644
index 00000000..b60cc301
--- /dev/null
+++ b/frontend/crawlab-ui/package.json
@@ -0,0 +1,175 @@
+{
+ "name": "crawlab-ui",
+ "version": "0.7.0-rc7",
+ "private": false,
+ "author": {
+ "name": "Marvin Zhang",
+ "email": "tikazyq@163.com"
+ },
+ "repository": {
+ "type": "git",
+ "url": "https://github.com/crawlab-team/crawlab-ui"
+ },
+ "license": "BSD-3-Clause",
+ "type": "module",
+ "files": [
+ "dist",
+ "public",
+ "src",
+ "typings",
+ "vite.config.js"
+ ],
+ "typings": "typings/index.d.ts",
+ "main": "./dist/crawlab-ui.umd.cjs",
+ "module": "./dist/crawlab-ui.umd.cjs",
+ "exports": {
+ ".": {
+ "import": "./dist/crawlab-ui.js",
+ "require": "./dist/crawlab-ui.umd.cjs"
+ },
+ "./dist/style.css": {
+ "import": "./dist/style.css",
+ "require": "./dist/style.css"
+ }
+ },
+ "scripts": {
+ "dev": "vite build --watch --mode development",
+ "serve": "vite",
+ "serve:host": "vite --host",
+ "serve:local": "vite serve --port=8081 --mode local",
+ "serve:dist": "serve dist",
+ "gen:index": "node ./scripts/gen-index.js",
+ "gen:svg": "node ./scripts/gen-svg.js",
+ "gen:dts": "node ./scripts/gen-dts.js",
+ "gen:dts:check": "vue-tsc --noEmit --rootDir ./src",
+ "gen:interfaces": "node ./scripts/gen-interfaces.js",
+ "prebuild": "pnpm run prebuild:gen",
+ "prebuild:gen": "pnpm run gen:index && pnpm run gen:svg",
+ "postbuild": "pnpm run postbuild:gen",
+ "postbuild:gen": "pnpm run gen:interfaces",
+ "build": "pnpm run prebuild && pnpm run build:lib && pnpm run postbuild",
+ "build:local": "vite build --mode local",
+ "build:lib": "vite build",
+ "lint": "vite lint",
+ "test": "jest"
+ },
+ "peerDependencies": {
+ "@element-plus/icons-vue": "^2.3.1",
+ "@fortawesome/fontawesome-common-types": "^6.5.2",
+ "@fortawesome/fontawesome-svg-core": "^6.5.2",
+ "@fortawesome/free-brands-svg-icons": "^6.5.2",
+ "@fortawesome/free-regular-svg-icons": "^6.5.2",
+ "@fortawesome/free-solid-svg-icons": "^6.5.2",
+ "@fortawesome/vue-fontawesome": "^3.0.8",
+ "@lexical/code": "^0.16.1",
+ "@lexical/history": "^0.16.1",
+ "@lexical/html": "^0.16.1",
+ "@lexical/link": "^0.16.1",
+ "@lexical/list": "^0.16.1",
+ "@lexical/markdown": "^0.16.1",
+ "@lexical/rich-text": "^0.16.1",
+ "@lexical/selection": "^0.16.1",
+ "@lexical/table": "^0.16.1",
+ "@lexical/text": "^0.16.1",
+ "@lexical/utils": "^0.16.1",
+ "@popperjs/core": "^2.11.8",
+ "@types/getos": "^3.0.4",
+ "@types/humanize-duration": "^3.27.4",
+ "@types/javascript-time-ago": "^2.0.8",
+ "@types/lodash": "^4.17.6",
+ "@types/markdown-it": "^14.1.2",
+ "@types/md5": "^2.3.5",
+ "@types/pinyin": "^2.10.2",
+ "@types/three": "^0.166.0",
+ "@types/url-join": "^4.0.3",
+ "async-validator": "^4.2.5",
+ "atom-material-icons": "^3.0.0",
+ "axios": "^1.7.2",
+ "chalk": "^5.3.0",
+ "chart.js": "^4.4.3",
+ "chartjs-adapter-date-fns": "^3.0.0",
+ "clipboard": "^2.0.11",
+ "core-js": "^3.37.1",
+ "crawlab-vue3-dropzone": "3.0.3",
+ "cron-parser": "^4.9.0",
+ "cronstrue": "^2.50.0",
+ "dayjs": "^1.11.11",
+ "dompurify": "^3.2.4",
+ "element-plus": "^2.7.6",
+ "font-awesome": "^4.7.0",
+ "humanize-duration": "^3.32.1",
+ "javascript-time-ago": "^2.5.10",
+ "lexical": "^0.16.1",
+ "lodash": "^4.17.21",
+ "markdown-it": "^14.1.0",
+ "md5": "^2.3.0",
+ "mitt": "^3.0.1",
+ "monaco-editor": "^0.50.0",
+ "normalize.css": "^8.0.1",
+ "os-name": "^5.0.1",
+ "pinyin": "^2.10.2",
+ "point-cluster": "^3.1.8",
+ "prismjs": "^1.29.0",
+ "three": "^0.166.1",
+ "tiny-invariant": "^1.3.3",
+ "url-join": "^4.0.1",
+ "url-regex": "^5.0.0",
+ "vite-svg-loader": "^5.1.0",
+ "vue": "^3.4",
+ "vue-chartjs": "^5.3.1",
+ "vue-clipboard3": "^2.0.0",
+ "vue-i18n": "9.13.1",
+ "vue-router": "^4.4.0",
+ "vuex": "^4.1.0"
+ },
+ "devDependencies": {
+ "@lexical/code": "^0.16.1",
+ "@lexical/html": "^0.16.1",
+ "@lexical/link": "^0.16.1",
+ "@lexical/list": "^0.16.1",
+ "@lexical/markdown": "^0.16.1",
+ "@lexical/selection": "^0.16.1",
+ "@lexical/table": "^0.16.1",
+ "@types/jest": "^29.5.12",
+ "@types/node": "^20.14.9",
+ "@types/node-os-utils": "^1.3.4",
+ "@types/three": "^0.166.0",
+ "@vitejs/plugin-vue": "^5.2.1",
+ "@vitejs/plugin-vue-jsx": "^4.1.1",
+ "@vue/compiler-sfc": "^3.4.31",
+ "autoprefixer": "^10.4.19",
+ "base64-img": "^1.0.4",
+ "chartjs-adapter-date-fns": "^3.0.0",
+ "cross-env": "^7.0.3",
+ "esno": "^4.7.0",
+ "fast-glob": "^3.3.2",
+ "fs-extra": "^11.2.0",
+ "globals": "^15.8.0",
+ "highlight.js": "^11.11.1",
+ "increase-memory-limit": "^1.0.7",
+ "jest": "^29.7.0",
+ "klaw-sync": "^6.0.0",
+ "mitt": "^3.0.1",
+ "postcss": "^8.4.39",
+ "prettier": "^3.3.2",
+ "prismjs": "^1.29.0",
+ "rollup-plugin-external-globals": "^0.10.0",
+ "rollup-plugin-visualizer": "^5.12.0",
+ "sass": "^1.80.4",
+ "serve": "^14.2.3",
+ "tailwindcss": "^3.4.4",
+ "typescript": "^5.5.3",
+ "vite": "^6.0.9",
+ "vite-plugin-dynamic-import": "^1.5.0",
+ "vite-plugin-static-copy": "^2.2.0",
+ "vite-svg-loader": "^5.1.0",
+ "vue-chartjs": "^5.3.1",
+ "vue-tsc": "^2.0.24"
+ },
+ "dependencies": {
+ "@types/semver": "^7.5.8",
+ "json-editor-vue": "^0.17.0",
+ "semver": "^7.6.3",
+ "uuid": "^10.0.0"
+ }
+}
diff --git a/frontend/crawlab-ui/postcss.config.js b/frontend/crawlab-ui/postcss.config.js
new file mode 100644
index 00000000..2aa7205d
--- /dev/null
+++ b/frontend/crawlab-ui/postcss.config.js
@@ -0,0 +1,6 @@
+export default {
+ plugins: {
+ tailwindcss: {},
+ autoprefixer: {},
+ },
+};
diff --git a/frontend/crawlab-ui/public/css/font-awesome.min.css b/frontend/crawlab-ui/public/css/font-awesome.min.css
new file mode 100644
index 00000000..540440ce
--- /dev/null
+++ b/frontend/crawlab-ui/public/css/font-awesome.min.css
@@ -0,0 +1,4 @@
+/*!
+ * Font Awesome 4.7.0 by @davegandy - http://fontawesome.io - @fontawesome
+ * License - http://fontawesome.io/license (Font: SIL OFL 1.1, CSS: MIT License)
+ */@font-face{font-family:'FontAwesome';src:url('../fonts/fontawesome-webfont.eot?v=4.7.0');src:url('../fonts/fontawesome-webfont.eot?#iefix&v=4.7.0') format('embedded-opentype'),url('../fonts/fontawesome-webfont.woff2?v=4.7.0') format('woff2'),url('../fonts/fontawesome-webfont.woff?v=4.7.0') format('woff'),url('../fonts/fontawesome-webfont.ttf?v=4.7.0') format('truetype'),url('../fonts/fontawesome-webfont.svg?v=4.7.0#fontawesomeregular') format('svg');font-weight:normal;font-style:normal}.fa{display:inline-block;font:normal normal normal 14px/1 FontAwesome;font-size:inherit;text-rendering:auto;-webkit-font-smoothing:antialiased;-moz-osx-font-smoothing:grayscale}.fa-lg{font-size:1.33333333em;line-height:.75em;vertical-align:-15%}.fa-2x{font-size:2em}.fa-3x{font-size:3em}.fa-4x{font-size:4em}.fa-5x{font-size:5em}.fa-fw{width:1.28571429em;text-align:center}.fa-ul{padding-left:0;margin-left:2.14285714em;list-style-type:none}.fa-ul>li{position:relative}.fa-li{position:absolute;left:-2.14285714em;width:2.14285714em;top:.14285714em;text-align:center}.fa-li.fa-lg{left:-1.85714286em}.fa-border{padding:.2em .25em .15em;border:solid .08em #eee;border-radius:.1em}.fa-pull-left{float:left}.fa-pull-right{float:right}.fa.fa-pull-left{margin-right:.3em}.fa.fa-pull-right{margin-left:.3em}.pull-right{float:right}.pull-left{float:left}.fa.pull-left{margin-right:.3em}.fa.pull-right{margin-left:.3em}.fa-spin{-webkit-animation:fa-spin 2s infinite linear;animation:fa-spin 2s infinite linear}.fa-pulse{-webkit-animation:fa-spin 1s infinite steps(8);animation:fa-spin 1s infinite steps(8)}@-webkit-keyframes fa-spin{0%{-webkit-transform:rotate(0deg);transform:rotate(0deg)}100%{-webkit-transform:rotate(359deg);transform:rotate(359deg)}}@keyframes fa-spin{0%{-webkit-transform:rotate(0deg);transform:rotate(0deg)}100%{-webkit-transform:rotate(359deg);transform:rotate(359deg)}}.fa-rotate-90{-ms-filter:"progid:DXImageTransform.Microsoft.BasicImage(rotation=1)";-webkit-transform:rotate(90deg);-ms-transform:rotate(90deg);transform:rotate(90deg)}.fa-rotate-180{-ms-filter:"progid:DXImageTransform.Microsoft.BasicImage(rotation=2)";-webkit-transform:rotate(180deg);-ms-transform:rotate(180deg);transform:rotate(180deg)}.fa-rotate-270{-ms-filter:"progid:DXImageTransform.Microsoft.BasicImage(rotation=3)";-webkit-transform:rotate(270deg);-ms-transform:rotate(270deg);transform:rotate(270deg)}.fa-flip-horizontal{-ms-filter:"progid:DXImageTransform.Microsoft.BasicImage(rotation=0, mirror=1)";-webkit-transform:scale(-1, 1);-ms-transform:scale(-1, 1);transform:scale(-1, 1)}.fa-flip-vertical{-ms-filter:"progid:DXImageTransform.Microsoft.BasicImage(rotation=2, mirror=1)";-webkit-transform:scale(1, -1);-ms-transform:scale(1, -1);transform:scale(1, -1)}:root .fa-rotate-90,:root .fa-rotate-180,:root .fa-rotate-270,:root .fa-flip-horizontal,:root .fa-flip-vertical{filter:none}.fa-stack{position:relative;display:inline-block;width:2em;height:2em;line-height:2em;vertical-align:middle}.fa-stack-1x,.fa-stack-2x{position:absolute;left:0;width:100%;text-align:center}.fa-stack-1x{line-height:inherit}.fa-stack-2x{font-size:2em}.fa-inverse{color:#fff}.fa-glass:before{content:"\f000"}.fa-music:before{content:"\f001"}.fa-search:before{content:"\f002"}.fa-envelope-o:before{content:"\f003"}.fa-heart:before{content:"\f004"}.fa-star:before{content:"\f005"}.fa-star-o:before{content:"\f006"}.fa-user:before{content:"\f007"}.fa-film:before{content:"\f008"}.fa-th-large:before{content:"\f009"}.fa-th:before{content:"\f00a"}.fa-th-list:before{content:"\f00b"}.fa-check:before{content:"\f00c"}.fa-remove:before,.fa-close:before,.fa-times:before{content:"\f00d"}.fa-search-plus:before{content:"\f00e"}.fa-search-minus:before{content:"\f010"}.fa-power-off:before{content:"\f011"}.fa-signal:before{content:"\f012"}.fa-gear:before,.fa-cog:before{content:"\f013"}.fa-trash-o:before{content:"\f014"}.fa-home:before{content:"\f015"}.fa-file-o:before{content:"\f016"}.fa-clock-o:before{content:"\f017"}.fa-road:before{content:"\f018"}.fa-download:before{content:"\f019"}.fa-arrow-circle-o-down:before{content:"\f01a"}.fa-arrow-circle-o-up:before{content:"\f01b"}.fa-inbox:before{content:"\f01c"}.fa-play-circle-o:before{content:"\f01d"}.fa-rotate-right:before,.fa-repeat:before{content:"\f01e"}.fa-refresh:before{content:"\f021"}.fa-list-alt:before{content:"\f022"}.fa-lock:before{content:"\f023"}.fa-flag:before{content:"\f024"}.fa-headphones:before{content:"\f025"}.fa-volume-off:before{content:"\f026"}.fa-volume-down:before{content:"\f027"}.fa-volume-up:before{content:"\f028"}.fa-qrcode:before{content:"\f029"}.fa-barcode:before{content:"\f02a"}.fa-tag:before{content:"\f02b"}.fa-tags:before{content:"\f02c"}.fa-book:before{content:"\f02d"}.fa-bookmark:before{content:"\f02e"}.fa-print:before{content:"\f02f"}.fa-camera:before{content:"\f030"}.fa-font:before{content:"\f031"}.fa-bold:before{content:"\f032"}.fa-italic:before{content:"\f033"}.fa-text-height:before{content:"\f034"}.fa-text-width:before{content:"\f035"}.fa-align-left:before{content:"\f036"}.fa-align-center:before{content:"\f037"}.fa-align-right:before{content:"\f038"}.fa-align-justify:before{content:"\f039"}.fa-list:before{content:"\f03a"}.fa-dedent:before,.fa-outdent:before{content:"\f03b"}.fa-indent:before{content:"\f03c"}.fa-video-camera:before{content:"\f03d"}.fa-photo:before,.fa-image:before,.fa-picture-o:before{content:"\f03e"}.fa-pencil:before{content:"\f040"}.fa-map-marker:before{content:"\f041"}.fa-adjust:before{content:"\f042"}.fa-tint:before{content:"\f043"}.fa-edit:before,.fa-pencil-square-o:before{content:"\f044"}.fa-share-square-o:before{content:"\f045"}.fa-check-square-o:before{content:"\f046"}.fa-arrows:before{content:"\f047"}.fa-step-backward:before{content:"\f048"}.fa-fast-backward:before{content:"\f049"}.fa-backward:before{content:"\f04a"}.fa-play:before{content:"\f04b"}.fa-pause:before{content:"\f04c"}.fa-stop:before{content:"\f04d"}.fa-forward:before{content:"\f04e"}.fa-fast-forward:before{content:"\f050"}.fa-step-forward:before{content:"\f051"}.fa-eject:before{content:"\f052"}.fa-chevron-left:before{content:"\f053"}.fa-chevron-right:before{content:"\f054"}.fa-plus-circle:before{content:"\f055"}.fa-minus-circle:before{content:"\f056"}.fa-times-circle:before{content:"\f057"}.fa-check-circle:before{content:"\f058"}.fa-question-circle:before{content:"\f059"}.fa-info-circle:before{content:"\f05a"}.fa-crosshairs:before{content:"\f05b"}.fa-times-circle-o:before{content:"\f05c"}.fa-check-circle-o:before{content:"\f05d"}.fa-ban:before{content:"\f05e"}.fa-arrow-left:before{content:"\f060"}.fa-arrow-right:before{content:"\f061"}.fa-arrow-up:before{content:"\f062"}.fa-arrow-down:before{content:"\f063"}.fa-mail-forward:before,.fa-share:before{content:"\f064"}.fa-expand:before{content:"\f065"}.fa-compress:before{content:"\f066"}.fa-plus:before{content:"\f067"}.fa-minus:before{content:"\f068"}.fa-asterisk:before{content:"\f069"}.fa-exclamation-circle:before{content:"\f06a"}.fa-gift:before{content:"\f06b"}.fa-leaf:before{content:"\f06c"}.fa-fire:before{content:"\f06d"}.fa-eye:before{content:"\f06e"}.fa-eye-slash:before{content:"\f070"}.fa-warning:before,.fa-exclamation-triangle:before{content:"\f071"}.fa-plane:before{content:"\f072"}.fa-calendar:before{content:"\f073"}.fa-random:before{content:"\f074"}.fa-comment:before{content:"\f075"}.fa-magnet:before{content:"\f076"}.fa-chevron-up:before{content:"\f077"}.fa-chevron-down:before{content:"\f078"}.fa-retweet:before{content:"\f079"}.fa-shopping-cart:before{content:"\f07a"}.fa-folder:before{content:"\f07b"}.fa-folder-open:before{content:"\f07c"}.fa-arrows-v:before{content:"\f07d"}.fa-arrows-h:before{content:"\f07e"}.fa-bar-chart-o:before,.fa-bar-chart:before{content:"\f080"}.fa-twitter-square:before{content:"\f081"}.fa-facebook-square:before{content:"\f082"}.fa-camera-retro:before{content:"\f083"}.fa-key:before{content:"\f084"}.fa-gears:before,.fa-cogs:before{content:"\f085"}.fa-comments:before{content:"\f086"}.fa-thumbs-o-up:before{content:"\f087"}.fa-thumbs-o-down:before{content:"\f088"}.fa-star-half:before{content:"\f089"}.fa-heart-o:before{content:"\f08a"}.fa-sign-out:before{content:"\f08b"}.fa-linkedin-square:before{content:"\f08c"}.fa-thumb-tack:before{content:"\f08d"}.fa-external-link:before{content:"\f08e"}.fa-sign-in:before{content:"\f090"}.fa-trophy:before{content:"\f091"}.fa-github-square:before{content:"\f092"}.fa-upload:before{content:"\f093"}.fa-lemon-o:before{content:"\f094"}.fa-phone:before{content:"\f095"}.fa-square-o:before{content:"\f096"}.fa-bookmark-o:before{content:"\f097"}.fa-phone-square:before{content:"\f098"}.fa-twitter:before{content:"\f099"}.fa-facebook-f:before,.fa-facebook:before{content:"\f09a"}.fa-github:before{content:"\f09b"}.fa-unlock:before{content:"\f09c"}.fa-credit-card:before{content:"\f09d"}.fa-feed:before,.fa-rss:before{content:"\f09e"}.fa-hdd-o:before{content:"\f0a0"}.fa-bullhorn:before{content:"\f0a1"}.fa-bell:before{content:"\f0f3"}.fa-certificate:before{content:"\f0a3"}.fa-hand-o-right:before{content:"\f0a4"}.fa-hand-o-left:before{content:"\f0a5"}.fa-hand-o-up:before{content:"\f0a6"}.fa-hand-o-down:before{content:"\f0a7"}.fa-arrow-circle-left:before{content:"\f0a8"}.fa-arrow-circle-right:before{content:"\f0a9"}.fa-arrow-circle-up:before{content:"\f0aa"}.fa-arrow-circle-down:before{content:"\f0ab"}.fa-globe:before{content:"\f0ac"}.fa-wrench:before{content:"\f0ad"}.fa-tasks:before{content:"\f0ae"}.fa-filter:before{content:"\f0b0"}.fa-briefcase:before{content:"\f0b1"}.fa-arrows-alt:before{content:"\f0b2"}.fa-group:before,.fa-users:before{content:"\f0c0"}.fa-chain:before,.fa-link:before{content:"\f0c1"}.fa-cloud:before{content:"\f0c2"}.fa-flask:before{content:"\f0c3"}.fa-cut:before,.fa-scissors:before{content:"\f0c4"}.fa-copy:before,.fa-files-o:before{content:"\f0c5"}.fa-paperclip:before{content:"\f0c6"}.fa-save:before,.fa-floppy-o:before{content:"\f0c7"}.fa-square:before{content:"\f0c8"}.fa-navicon:before,.fa-reorder:before,.fa-bars:before{content:"\f0c9"}.fa-list-ul:before{content:"\f0ca"}.fa-list-ol:before{content:"\f0cb"}.fa-strikethrough:before{content:"\f0cc"}.fa-underline:before{content:"\f0cd"}.fa-table:before{content:"\f0ce"}.fa-magic:before{content:"\f0d0"}.fa-truck:before{content:"\f0d1"}.fa-pinterest:before{content:"\f0d2"}.fa-pinterest-square:before{content:"\f0d3"}.fa-google-plus-square:before{content:"\f0d4"}.fa-google-plus:before{content:"\f0d5"}.fa-money:before{content:"\f0d6"}.fa-caret-down:before{content:"\f0d7"}.fa-caret-up:before{content:"\f0d8"}.fa-caret-left:before{content:"\f0d9"}.fa-caret-right:before{content:"\f0da"}.fa-columns:before{content:"\f0db"}.fa-unsorted:before,.fa-sort:before{content:"\f0dc"}.fa-sort-down:before,.fa-sort-desc:before{content:"\f0dd"}.fa-sort-up:before,.fa-sort-asc:before{content:"\f0de"}.fa-envelope:before{content:"\f0e0"}.fa-linkedin:before{content:"\f0e1"}.fa-rotate-left:before,.fa-undo:before{content:"\f0e2"}.fa-legal:before,.fa-gavel:before{content:"\f0e3"}.fa-dashboard:before,.fa-tachometer:before{content:"\f0e4"}.fa-comment-o:before{content:"\f0e5"}.fa-comments-o:before{content:"\f0e6"}.fa-flash:before,.fa-bolt:before{content:"\f0e7"}.fa-sitemap:before{content:"\f0e8"}.fa-umbrella:before{content:"\f0e9"}.fa-paste:before,.fa-clipboard:before{content:"\f0ea"}.fa-lightbulb-o:before{content:"\f0eb"}.fa-exchange:before{content:"\f0ec"}.fa-cloud-download:before{content:"\f0ed"}.fa-cloud-upload:before{content:"\f0ee"}.fa-user-md:before{content:"\f0f0"}.fa-stethoscope:before{content:"\f0f1"}.fa-suitcase:before{content:"\f0f2"}.fa-bell-o:before{content:"\f0a2"}.fa-coffee:before{content:"\f0f4"}.fa-cutlery:before{content:"\f0f5"}.fa-file-text-o:before{content:"\f0f6"}.fa-building-o:before{content:"\f0f7"}.fa-hospital-o:before{content:"\f0f8"}.fa-ambulance:before{content:"\f0f9"}.fa-medkit:before{content:"\f0fa"}.fa-fighter-jet:before{content:"\f0fb"}.fa-beer:before{content:"\f0fc"}.fa-h-square:before{content:"\f0fd"}.fa-plus-square:before{content:"\f0fe"}.fa-angle-double-left:before{content:"\f100"}.fa-angle-double-right:before{content:"\f101"}.fa-angle-double-up:before{content:"\f102"}.fa-angle-double-down:before{content:"\f103"}.fa-angle-left:before{content:"\f104"}.fa-angle-right:before{content:"\f105"}.fa-angle-up:before{content:"\f106"}.fa-angle-down:before{content:"\f107"}.fa-desktop:before{content:"\f108"}.fa-laptop:before{content:"\f109"}.fa-tablet:before{content:"\f10a"}.fa-mobile-phone:before,.fa-mobile:before{content:"\f10b"}.fa-circle-o:before{content:"\f10c"}.fa-quote-left:before{content:"\f10d"}.fa-quote-right:before{content:"\f10e"}.fa-spinner:before{content:"\f110"}.fa-circle:before{content:"\f111"}.fa-mail-reply:before,.fa-reply:before{content:"\f112"}.fa-github-alt:before{content:"\f113"}.fa-folder-o:before{content:"\f114"}.fa-folder-open-o:before{content:"\f115"}.fa-smile-o:before{content:"\f118"}.fa-frown-o:before{content:"\f119"}.fa-meh-o:before{content:"\f11a"}.fa-gamepad:before{content:"\f11b"}.fa-keyboard-o:before{content:"\f11c"}.fa-flag-o:before{content:"\f11d"}.fa-flag-checkered:before{content:"\f11e"}.fa-terminal:before{content:"\f120"}.fa-code:before{content:"\f121"}.fa-mail-reply-all:before,.fa-reply-all:before{content:"\f122"}.fa-star-half-empty:before,.fa-star-half-full:before,.fa-star-half-o:before{content:"\f123"}.fa-location-arrow:before{content:"\f124"}.fa-crop:before{content:"\f125"}.fa-code-fork:before{content:"\f126"}.fa-unlink:before,.fa-chain-broken:before{content:"\f127"}.fa-question:before{content:"\f128"}.fa-info:before{content:"\f129"}.fa-exclamation:before{content:"\f12a"}.fa-superscript:before{content:"\f12b"}.fa-subscript:before{content:"\f12c"}.fa-eraser:before{content:"\f12d"}.fa-puzzle-piece:before{content:"\f12e"}.fa-microphone:before{content:"\f130"}.fa-microphone-slash:before{content:"\f131"}.fa-shield:before{content:"\f132"}.fa-calendar-o:before{content:"\f133"}.fa-fire-extinguisher:before{content:"\f134"}.fa-rocket:before{content:"\f135"}.fa-maxcdn:before{content:"\f136"}.fa-chevron-circle-left:before{content:"\f137"}.fa-chevron-circle-right:before{content:"\f138"}.fa-chevron-circle-up:before{content:"\f139"}.fa-chevron-circle-down:before{content:"\f13a"}.fa-html5:before{content:"\f13b"}.fa-css3:before{content:"\f13c"}.fa-anchor:before{content:"\f13d"}.fa-unlock-alt:before{content:"\f13e"}.fa-bullseye:before{content:"\f140"}.fa-ellipsis-h:before{content:"\f141"}.fa-ellipsis-v:before{content:"\f142"}.fa-rss-square:before{content:"\f143"}.fa-play-circle:before{content:"\f144"}.fa-ticket:before{content:"\f145"}.fa-minus-square:before{content:"\f146"}.fa-minus-square-o:before{content:"\f147"}.fa-level-up:before{content:"\f148"}.fa-level-down:before{content:"\f149"}.fa-check-square:before{content:"\f14a"}.fa-pencil-square:before{content:"\f14b"}.fa-external-link-square:before{content:"\f14c"}.fa-share-square:before{content:"\f14d"}.fa-compass:before{content:"\f14e"}.fa-toggle-down:before,.fa-caret-square-o-down:before{content:"\f150"}.fa-toggle-up:before,.fa-caret-square-o-up:before{content:"\f151"}.fa-toggle-right:before,.fa-caret-square-o-right:before{content:"\f152"}.fa-euro:before,.fa-eur:before{content:"\f153"}.fa-gbp:before{content:"\f154"}.fa-dollar:before,.fa-usd:before{content:"\f155"}.fa-rupee:before,.fa-inr:before{content:"\f156"}.fa-cny:before,.fa-rmb:before,.fa-yen:before,.fa-jpy:before{content:"\f157"}.fa-ruble:before,.fa-rouble:before,.fa-rub:before{content:"\f158"}.fa-won:before,.fa-krw:before{content:"\f159"}.fa-bitcoin:before,.fa-btc:before{content:"\f15a"}.fa-file:before{content:"\f15b"}.fa-file-text:before{content:"\f15c"}.fa-sort-alpha-asc:before{content:"\f15d"}.fa-sort-alpha-desc:before{content:"\f15e"}.fa-sort-amount-asc:before{content:"\f160"}.fa-sort-amount-desc:before{content:"\f161"}.fa-sort-numeric-asc:before{content:"\f162"}.fa-sort-numeric-desc:before{content:"\f163"}.fa-thumbs-up:before{content:"\f164"}.fa-thumbs-down:before{content:"\f165"}.fa-youtube-square:before{content:"\f166"}.fa-youtube:before{content:"\f167"}.fa-xing:before{content:"\f168"}.fa-xing-square:before{content:"\f169"}.fa-youtube-play:before{content:"\f16a"}.fa-dropbox:before{content:"\f16b"}.fa-stack-overflow:before{content:"\f16c"}.fa-instagram:before{content:"\f16d"}.fa-flickr:before{content:"\f16e"}.fa-adn:before{content:"\f170"}.fa-bitbucket:before{content:"\f171"}.fa-bitbucket-square:before{content:"\f172"}.fa-tumblr:before{content:"\f173"}.fa-tumblr-square:before{content:"\f174"}.fa-long-arrow-down:before{content:"\f175"}.fa-long-arrow-up:before{content:"\f176"}.fa-long-arrow-left:before{content:"\f177"}.fa-long-arrow-right:before{content:"\f178"}.fa-apple:before{content:"\f179"}.fa-windows:before{content:"\f17a"}.fa-android:before{content:"\f17b"}.fa-linux:before{content:"\f17c"}.fa-dribbble:before{content:"\f17d"}.fa-skype:before{content:"\f17e"}.fa-foursquare:before{content:"\f180"}.fa-trello:before{content:"\f181"}.fa-female:before{content:"\f182"}.fa-male:before{content:"\f183"}.fa-gittip:before,.fa-gratipay:before{content:"\f184"}.fa-sun-o:before{content:"\f185"}.fa-moon-o:before{content:"\f186"}.fa-archive:before{content:"\f187"}.fa-bug:before{content:"\f188"}.fa-vk:before{content:"\f189"}.fa-weibo:before{content:"\f18a"}.fa-renren:before{content:"\f18b"}.fa-pagelines:before{content:"\f18c"}.fa-stack-exchange:before{content:"\f18d"}.fa-arrow-circle-o-right:before{content:"\f18e"}.fa-arrow-circle-o-left:before{content:"\f190"}.fa-toggle-left:before,.fa-caret-square-o-left:before{content:"\f191"}.fa-dot-circle-o:before{content:"\f192"}.fa-wheelchair:before{content:"\f193"}.fa-vimeo-square:before{content:"\f194"}.fa-turkish-lira:before,.fa-try:before{content:"\f195"}.fa-plus-square-o:before{content:"\f196"}.fa-space-shuttle:before{content:"\f197"}.fa-slack:before{content:"\f198"}.fa-envelope-square:before{content:"\f199"}.fa-wordpress:before{content:"\f19a"}.fa-openid:before{content:"\f19b"}.fa-institution:before,.fa-bank:before,.fa-university:before{content:"\f19c"}.fa-mortar-board:before,.fa-graduation-cap:before{content:"\f19d"}.fa-yahoo:before{content:"\f19e"}.fa-google:before{content:"\f1a0"}.fa-reddit:before{content:"\f1a1"}.fa-reddit-square:before{content:"\f1a2"}.fa-stumbleupon-circle:before{content:"\f1a3"}.fa-stumbleupon:before{content:"\f1a4"}.fa-delicious:before{content:"\f1a5"}.fa-digg:before{content:"\f1a6"}.fa-pied-piper-pp:before{content:"\f1a7"}.fa-pied-piper-alt:before{content:"\f1a8"}.fa-drupal:before{content:"\f1a9"}.fa-joomla:before{content:"\f1aa"}.fa-language:before{content:"\f1ab"}.fa-fax:before{content:"\f1ac"}.fa-building:before{content:"\f1ad"}.fa-child:before{content:"\f1ae"}.fa-paw:before{content:"\f1b0"}.fa-spoon:before{content:"\f1b1"}.fa-cube:before{content:"\f1b2"}.fa-cubes:before{content:"\f1b3"}.fa-behance:before{content:"\f1b4"}.fa-behance-square:before{content:"\f1b5"}.fa-steam:before{content:"\f1b6"}.fa-steam-square:before{content:"\f1b7"}.fa-recycle:before{content:"\f1b8"}.fa-automobile:before,.fa-car:before{content:"\f1b9"}.fa-cab:before,.fa-taxi:before{content:"\f1ba"}.fa-tree:before{content:"\f1bb"}.fa-spotify:before{content:"\f1bc"}.fa-deviantart:before{content:"\f1bd"}.fa-soundcloud:before{content:"\f1be"}.fa-database:before{content:"\f1c0"}.fa-file-pdf-o:before{content:"\f1c1"}.fa-file-word-o:before{content:"\f1c2"}.fa-file-excel-o:before{content:"\f1c3"}.fa-file-powerpoint-o:before{content:"\f1c4"}.fa-file-photo-o:before,.fa-file-picture-o:before,.fa-file-image-o:before{content:"\f1c5"}.fa-file-zip-o:before,.fa-file-archive-o:before{content:"\f1c6"}.fa-file-sound-o:before,.fa-file-audio-o:before{content:"\f1c7"}.fa-file-movie-o:before,.fa-file-video-o:before{content:"\f1c8"}.fa-file-code-o:before{content:"\f1c9"}.fa-vine:before{content:"\f1ca"}.fa-codepen:before{content:"\f1cb"}.fa-jsfiddle:before{content:"\f1cc"}.fa-life-bouy:before,.fa-life-buoy:before,.fa-life-saver:before,.fa-support:before,.fa-life-ring:before{content:"\f1cd"}.fa-circle-o-notch:before{content:"\f1ce"}.fa-ra:before,.fa-resistance:before,.fa-rebel:before{content:"\f1d0"}.fa-ge:before,.fa-empire:before{content:"\f1d1"}.fa-git-square:before{content:"\f1d2"}.fa-git:before{content:"\f1d3"}.fa-y-combinator-square:before,.fa-yc-square:before,.fa-hacker-news:before{content:"\f1d4"}.fa-tencent-weibo:before{content:"\f1d5"}.fa-qq:before{content:"\f1d6"}.fa-wechat:before,.fa-weixin:before{content:"\f1d7"}.fa-send:before,.fa-paper-plane:before{content:"\f1d8"}.fa-send-o:before,.fa-paper-plane-o:before{content:"\f1d9"}.fa-history:before{content:"\f1da"}.fa-circle-thin:before{content:"\f1db"}.fa-header:before{content:"\f1dc"}.fa-paragraph:before{content:"\f1dd"}.fa-sliders:before{content:"\f1de"}.fa-share-alt:before{content:"\f1e0"}.fa-share-alt-square:before{content:"\f1e1"}.fa-bomb:before{content:"\f1e2"}.fa-soccer-ball-o:before,.fa-futbol-o:before{content:"\f1e3"}.fa-tty:before{content:"\f1e4"}.fa-binoculars:before{content:"\f1e5"}.fa-plug:before{content:"\f1e6"}.fa-slideshare:before{content:"\f1e7"}.fa-twitch:before{content:"\f1e8"}.fa-yelp:before{content:"\f1e9"}.fa-newspaper-o:before{content:"\f1ea"}.fa-wifi:before{content:"\f1eb"}.fa-calculator:before{content:"\f1ec"}.fa-paypal:before{content:"\f1ed"}.fa-google-wallet:before{content:"\f1ee"}.fa-cc-visa:before{content:"\f1f0"}.fa-cc-mastercard:before{content:"\f1f1"}.fa-cc-discover:before{content:"\f1f2"}.fa-cc-amex:before{content:"\f1f3"}.fa-cc-paypal:before{content:"\f1f4"}.fa-cc-stripe:before{content:"\f1f5"}.fa-bell-slash:before{content:"\f1f6"}.fa-bell-slash-o:before{content:"\f1f7"}.fa-trash:before{content:"\f1f8"}.fa-copyright:before{content:"\f1f9"}.fa-at:before{content:"\f1fa"}.fa-eyedropper:before{content:"\f1fb"}.fa-paint-brush:before{content:"\f1fc"}.fa-birthday-cake:before{content:"\f1fd"}.fa-area-chart:before{content:"\f1fe"}.fa-pie-chart:before{content:"\f200"}.fa-line-chart:before{content:"\f201"}.fa-lastfm:before{content:"\f202"}.fa-lastfm-square:before{content:"\f203"}.fa-toggle-off:before{content:"\f204"}.fa-toggle-on:before{content:"\f205"}.fa-bicycle:before{content:"\f206"}.fa-bus:before{content:"\f207"}.fa-ioxhost:before{content:"\f208"}.fa-angellist:before{content:"\f209"}.fa-cc:before{content:"\f20a"}.fa-shekel:before,.fa-sheqel:before,.fa-ils:before{content:"\f20b"}.fa-meanpath:before{content:"\f20c"}.fa-buysellads:before{content:"\f20d"}.fa-connectdevelop:before{content:"\f20e"}.fa-dashcube:before{content:"\f210"}.fa-forumbee:before{content:"\f211"}.fa-leanpub:before{content:"\f212"}.fa-sellsy:before{content:"\f213"}.fa-shirtsinbulk:before{content:"\f214"}.fa-simplybuilt:before{content:"\f215"}.fa-skyatlas:before{content:"\f216"}.fa-cart-plus:before{content:"\f217"}.fa-cart-arrow-down:before{content:"\f218"}.fa-diamond:before{content:"\f219"}.fa-ship:before{content:"\f21a"}.fa-user-secret:before{content:"\f21b"}.fa-motorcycle:before{content:"\f21c"}.fa-street-view:before{content:"\f21d"}.fa-heartbeat:before{content:"\f21e"}.fa-venus:before{content:"\f221"}.fa-mars:before{content:"\f222"}.fa-mercury:before{content:"\f223"}.fa-intersex:before,.fa-transgender:before{content:"\f224"}.fa-transgender-alt:before{content:"\f225"}.fa-venus-double:before{content:"\f226"}.fa-mars-double:before{content:"\f227"}.fa-venus-mars:before{content:"\f228"}.fa-mars-stroke:before{content:"\f229"}.fa-mars-stroke-v:before{content:"\f22a"}.fa-mars-stroke-h:before{content:"\f22b"}.fa-neuter:before{content:"\f22c"}.fa-genderless:before{content:"\f22d"}.fa-facebook-official:before{content:"\f230"}.fa-pinterest-p:before{content:"\f231"}.fa-whatsapp:before{content:"\f232"}.fa-server:before{content:"\f233"}.fa-user-plus:before{content:"\f234"}.fa-user-times:before{content:"\f235"}.fa-hotel:before,.fa-bed:before{content:"\f236"}.fa-viacoin:before{content:"\f237"}.fa-train:before{content:"\f238"}.fa-subway:before{content:"\f239"}.fa-medium:before{content:"\f23a"}.fa-yc:before,.fa-y-combinator:before{content:"\f23b"}.fa-optin-monster:before{content:"\f23c"}.fa-opencart:before{content:"\f23d"}.fa-expeditedssl:before{content:"\f23e"}.fa-battery-4:before,.fa-battery:before,.fa-battery-full:before{content:"\f240"}.fa-battery-3:before,.fa-battery-three-quarters:before{content:"\f241"}.fa-battery-2:before,.fa-battery-half:before{content:"\f242"}.fa-battery-1:before,.fa-battery-quarter:before{content:"\f243"}.fa-battery-0:before,.fa-battery-empty:before{content:"\f244"}.fa-mouse-pointer:before{content:"\f245"}.fa-i-cursor:before{content:"\f246"}.fa-object-group:before{content:"\f247"}.fa-object-ungroup:before{content:"\f248"}.fa-sticky-note:before{content:"\f249"}.fa-sticky-note-o:before{content:"\f24a"}.fa-cc-jcb:before{content:"\f24b"}.fa-cc-diners-club:before{content:"\f24c"}.fa-clone:before{content:"\f24d"}.fa-balance-scale:before{content:"\f24e"}.fa-hourglass-o:before{content:"\f250"}.fa-hourglass-1:before,.fa-hourglass-start:before{content:"\f251"}.fa-hourglass-2:before,.fa-hourglass-half:before{content:"\f252"}.fa-hourglass-3:before,.fa-hourglass-end:before{content:"\f253"}.fa-hourglass:before{content:"\f254"}.fa-hand-grab-o:before,.fa-hand-rock-o:before{content:"\f255"}.fa-hand-stop-o:before,.fa-hand-paper-o:before{content:"\f256"}.fa-hand-scissors-o:before{content:"\f257"}.fa-hand-lizard-o:before{content:"\f258"}.fa-hand-spock-o:before{content:"\f259"}.fa-hand-pointer-o:before{content:"\f25a"}.fa-hand-peace-o:before{content:"\f25b"}.fa-trademark:before{content:"\f25c"}.fa-registered:before{content:"\f25d"}.fa-creative-commons:before{content:"\f25e"}.fa-gg:before{content:"\f260"}.fa-gg-circle:before{content:"\f261"}.fa-tripadvisor:before{content:"\f262"}.fa-odnoklassniki:before{content:"\f263"}.fa-odnoklassniki-square:before{content:"\f264"}.fa-get-pocket:before{content:"\f265"}.fa-wikipedia-w:before{content:"\f266"}.fa-safari:before{content:"\f267"}.fa-chrome:before{content:"\f268"}.fa-firefox:before{content:"\f269"}.fa-opera:before{content:"\f26a"}.fa-internet-explorer:before{content:"\f26b"}.fa-tv:before,.fa-television:before{content:"\f26c"}.fa-contao:before{content:"\f26d"}.fa-500px:before{content:"\f26e"}.fa-amazon:before{content:"\f270"}.fa-calendar-plus-o:before{content:"\f271"}.fa-calendar-minus-o:before{content:"\f272"}.fa-calendar-times-o:before{content:"\f273"}.fa-calendar-check-o:before{content:"\f274"}.fa-industry:before{content:"\f275"}.fa-map-pin:before{content:"\f276"}.fa-map-signs:before{content:"\f277"}.fa-map-o:before{content:"\f278"}.fa-map:before{content:"\f279"}.fa-commenting:before{content:"\f27a"}.fa-commenting-o:before{content:"\f27b"}.fa-houzz:before{content:"\f27c"}.fa-vimeo:before{content:"\f27d"}.fa-black-tie:before{content:"\f27e"}.fa-fonticons:before{content:"\f280"}.fa-reddit-alien:before{content:"\f281"}.fa-edge:before{content:"\f282"}.fa-credit-card-alt:before{content:"\f283"}.fa-codiepie:before{content:"\f284"}.fa-modx:before{content:"\f285"}.fa-fort-awesome:before{content:"\f286"}.fa-usb:before{content:"\f287"}.fa-product-hunt:before{content:"\f288"}.fa-mixcloud:before{content:"\f289"}.fa-scribd:before{content:"\f28a"}.fa-pause-circle:before{content:"\f28b"}.fa-pause-circle-o:before{content:"\f28c"}.fa-stop-circle:before{content:"\f28d"}.fa-stop-circle-o:before{content:"\f28e"}.fa-shopping-bag:before{content:"\f290"}.fa-shopping-basket:before{content:"\f291"}.fa-hashtag:before{content:"\f292"}.fa-bluetooth:before{content:"\f293"}.fa-bluetooth-b:before{content:"\f294"}.fa-percent:before{content:"\f295"}.fa-gitlab:before{content:"\f296"}.fa-wpbeginner:before{content:"\f297"}.fa-wpforms:before{content:"\f298"}.fa-envira:before{content:"\f299"}.fa-universal-access:before{content:"\f29a"}.fa-wheelchair-alt:before{content:"\f29b"}.fa-question-circle-o:before{content:"\f29c"}.fa-blind:before{content:"\f29d"}.fa-audio-description:before{content:"\f29e"}.fa-volume-control-phone:before{content:"\f2a0"}.fa-braille:before{content:"\f2a1"}.fa-assistive-listening-systems:before{content:"\f2a2"}.fa-asl-interpreting:before,.fa-american-sign-language-interpreting:before{content:"\f2a3"}.fa-deafness:before,.fa-hard-of-hearing:before,.fa-deaf:before{content:"\f2a4"}.fa-glide:before{content:"\f2a5"}.fa-glide-g:before{content:"\f2a6"}.fa-signing:before,.fa-sign-language:before{content:"\f2a7"}.fa-low-vision:before{content:"\f2a8"}.fa-viadeo:before{content:"\f2a9"}.fa-viadeo-square:before{content:"\f2aa"}.fa-snapchat:before{content:"\f2ab"}.fa-snapchat-ghost:before{content:"\f2ac"}.fa-snapchat-square:before{content:"\f2ad"}.fa-pied-piper:before{content:"\f2ae"}.fa-first-order:before{content:"\f2b0"}.fa-yoast:before{content:"\f2b1"}.fa-themeisle:before{content:"\f2b2"}.fa-google-plus-circle:before,.fa-google-plus-official:before{content:"\f2b3"}.fa-fa:before,.fa-font-awesome:before{content:"\f2b4"}.fa-handshake-o:before{content:"\f2b5"}.fa-envelope-open:before{content:"\f2b6"}.fa-envelope-open-o:before{content:"\f2b7"}.fa-linode:before{content:"\f2b8"}.fa-address-book:before{content:"\f2b9"}.fa-address-book-o:before{content:"\f2ba"}.fa-vcard:before,.fa-address-card:before{content:"\f2bb"}.fa-vcard-o:before,.fa-address-card-o:before{content:"\f2bc"}.fa-user-circle:before{content:"\f2bd"}.fa-user-circle-o:before{content:"\f2be"}.fa-user-o:before{content:"\f2c0"}.fa-id-badge:before{content:"\f2c1"}.fa-drivers-license:before,.fa-id-card:before{content:"\f2c2"}.fa-drivers-license-o:before,.fa-id-card-o:before{content:"\f2c3"}.fa-quora:before{content:"\f2c4"}.fa-free-code-camp:before{content:"\f2c5"}.fa-telegram:before{content:"\f2c6"}.fa-thermometer-4:before,.fa-thermometer:before,.fa-thermometer-full:before{content:"\f2c7"}.fa-thermometer-3:before,.fa-thermometer-three-quarters:before{content:"\f2c8"}.fa-thermometer-2:before,.fa-thermometer-half:before{content:"\f2c9"}.fa-thermometer-1:before,.fa-thermometer-quarter:before{content:"\f2ca"}.fa-thermometer-0:before,.fa-thermometer-empty:before{content:"\f2cb"}.fa-shower:before{content:"\f2cc"}.fa-bathtub:before,.fa-s15:before,.fa-bath:before{content:"\f2cd"}.fa-podcast:before{content:"\f2ce"}.fa-window-maximize:before{content:"\f2d0"}.fa-window-minimize:before{content:"\f2d1"}.fa-window-restore:before{content:"\f2d2"}.fa-times-rectangle:before,.fa-window-close:before{content:"\f2d3"}.fa-times-rectangle-o:before,.fa-window-close-o:before{content:"\f2d4"}.fa-bandcamp:before{content:"\f2d5"}.fa-grav:before{content:"\f2d6"}.fa-etsy:before{content:"\f2d7"}.fa-imdb:before{content:"\f2d8"}.fa-ravelry:before{content:"\f2d9"}.fa-eercast:before{content:"\f2da"}.fa-microchip:before{content:"\f2db"}.fa-snowflake-o:before{content:"\f2dc"}.fa-superpowers:before{content:"\f2dd"}.fa-wpexplorer:before{content:"\f2de"}.fa-meetup:before{content:"\f2e0"}.sr-only{position:absolute;width:1px;height:1px;padding:0;margin:-1px;overflow:hidden;clip:rect(0, 0, 0, 0);border:0}.sr-only-focusable:active,.sr-only-focusable:focus{position:static;width:auto;height:auto;margin:0;overflow:visible;clip:auto}
diff --git a/frontend/crawlab-ui/public/css/style.css b/frontend/crawlab-ui/public/css/style.css
new file mode 100644
index 00000000..cef78f8c
--- /dev/null
+++ b/frontend/crawlab-ui/public/css/style.css
@@ -0,0 +1,179 @@
+#loading-placeholder {
+ position: fixed;
+ background: white;
+ z-index: -1;
+ top: 0;
+ left: 0;
+ width: 100vw;
+ height: 100vh;
+ display: flex;
+ align-items: center;
+ justify-content: center;
+}
+
+#loading-placeholder .title-wrapper {
+ height: 54px;
+}
+
+#loading-placeholder .title {
+ font-family: "Verdana", serif;
+ font-weight: 600;
+ font-size: 48px;
+ color: #409EFF;
+ text-align: center;
+ cursor: default;
+ letter-spacing: -5px;
+ margin: 0;
+}
+
+#loading-placeholder .title > span {
+ display: inline-block;
+ animation: change-shape 1s infinite;
+}
+
+#loading-placeholder .title > span:nth-child(1) {
+ animation-delay: calc(1s / 7 * 0 / 2);
+}
+
+#loading-placeholder .title > span:nth-child(2) {
+ animation-delay: calc(1s / 7 * 1 / 2);
+}
+
+#loading-placeholder .title > span:nth-child(3) {
+ animation-delay: calc(1s / 7 * 2 / 2);
+}
+
+#loading-placeholder .title > span:nth-child(4) {
+ animation-delay: calc(1s / 7 * 3 / 2);
+}
+
+#loading-placeholder .title > span:nth-child(5) {
+ animation-delay: calc(1s / 7 * 4 / 2);
+}
+
+#loading-placeholder .title > span:nth-child(6) {
+ animation-delay: calc(1s / 7 * 5 / 2);
+}
+
+#loading-placeholder .title > span:nth-child(7) {
+ animation-delay: calc(1s / 7 * 6 / 2);
+}
+
+#loading-placeholder .sub-title-wrapper {
+ position: relative;
+ display: flex;
+ align-items: center;
+ justify-content: center;
+ margin-bottom: 10px;
+ height: 28px;
+}
+
+#loading-placeholder .sub-title-wrapper .sub-title {
+ position: absolute;
+ font-size: 18px;
+ font-weight: 300;
+ font-family: "Verdana", serif;
+ font-style: italic;
+ color: #67C23A;
+ transform: rotate3d(1, 0, 0, 90deg);
+ animation: flip 20s infinite;
+ /*color: #E6A23C;*/
+ /*color: #F56C6C;*/
+}
+
+#loading-placeholder .sub-title-wrapper > .sub-title:nth-child(1) {
+ animation-delay: calc(20s / 4 * 0);
+}
+
+#loading-placeholder .sub-title-wrapper > .sub-title:nth-child(2) {
+ animation-delay: calc(20s / 4 * 1);
+}
+
+#loading-placeholder .sub-title-wrapper > .sub-title:nth-child(3) {
+ animation-delay: calc(20s / 4 * 2);
+}
+
+#loading-placeholder .sub-title-wrapper > .sub-title:nth-child(4) {
+ animation-delay: calc(20s / 4 * 3);
+}
+
+#loading-placeholder .loading-text {
+ text-align: center;
+ font-weight: bolder;
+ font-family: "Verdana", serif;
+ font-style: italic;
+ color: #889aa4;
+ font-size: 14px;
+ animation: blink-loading 2s ease-in infinite;
+}
+
+@keyframes blink-loading {
+ 0% {
+ opacity: 100%;
+ }
+
+ 50% {
+ opacity: 50%;
+ }
+
+ 100% {
+ opacity: 100%;
+ }
+}
+
+@keyframes change-shape {
+ 0% {
+ transform: scale(1);
+ }
+
+ 25% {
+ transform: scale(1.2);
+ }
+
+ 50% {
+ transform: scale(1);
+ }
+
+ 100% {
+ transform: scale(1);
+ }
+}
+
+@keyframes flip {
+ 0% {
+ transform: rotate3d(1, 0, 0, 90deg);
+ }
+
+ 2% {
+ transform: rotate3d(1, 0, 0, 90deg);
+ }
+
+ 7% {
+ transform: rotate3d(1, 0, 0, 0);
+ }
+
+ 23% {
+ transform: rotate3d(1, 0, 0, 0);
+ }
+
+ 27% {
+ transform: rotate3d(1, 0, 0, 90deg);
+ }
+
+ 50% {
+ transform: rotate3d(1, 0, 0, 90deg);
+ }
+
+ 100% {
+ transform: rotate3d(1, 0, 0, 90deg);
+ }
+}
+
+#login-canvas {
+ position: fixed;
+ top: 0;
+ left: 0;
+ width: 100vw;
+ height: 100vh;
+ z-index: -1;
+}
diff --git a/frontend/crawlab-ui/public/favicon.ico b/frontend/crawlab-ui/public/favicon.ico
new file mode 100644
index 00000000..3fb253bd
Binary files /dev/null and b/frontend/crawlab-ui/public/favicon.ico differ
diff --git a/frontend/crawlab-ui/public/fonts/FontAwesome.otf b/frontend/crawlab-ui/public/fonts/FontAwesome.otf
new file mode 100644
index 00000000..401ec0f3
Binary files /dev/null and b/frontend/crawlab-ui/public/fonts/FontAwesome.otf differ
diff --git a/frontend/crawlab-ui/public/fonts/fontawesome-webfont.eot b/frontend/crawlab-ui/public/fonts/fontawesome-webfont.eot
new file mode 100644
index 00000000..e9f60ca9
Binary files /dev/null and b/frontend/crawlab-ui/public/fonts/fontawesome-webfont.eot differ
diff --git a/frontend/crawlab-ui/public/fonts/fontawesome-webfont.svg b/frontend/crawlab-ui/public/fonts/fontawesome-webfont.svg
new file mode 100644
index 00000000..855c845e
--- /dev/null
+++ b/frontend/crawlab-ui/public/fonts/fontawesome-webfont.svg
@@ -0,0 +1,2671 @@
+
+
+
diff --git a/frontend/crawlab-ui/public/fonts/fontawesome-webfont.ttf b/frontend/crawlab-ui/public/fonts/fontawesome-webfont.ttf
new file mode 100644
index 00000000..35acda2f
Binary files /dev/null and b/frontend/crawlab-ui/public/fonts/fontawesome-webfont.ttf differ
diff --git a/frontend/crawlab-ui/public/fonts/fontawesome-webfont.woff b/frontend/crawlab-ui/public/fonts/fontawesome-webfont.woff
new file mode 100644
index 00000000..400014a4
Binary files /dev/null and b/frontend/crawlab-ui/public/fonts/fontawesome-webfont.woff differ
diff --git a/frontend/crawlab-ui/public/fonts/fontawesome-webfont.woff2 b/frontend/crawlab-ui/public/fonts/fontawesome-webfont.woff2
new file mode 100644
index 00000000..4d13fc60
Binary files /dev/null and b/frontend/crawlab-ui/public/fonts/fontawesome-webfont.woff2 differ
diff --git a/frontend/crawlab-ui/public/js/login-canvas-three.js b/frontend/crawlab-ui/public/js/login-canvas-three.js
new file mode 100644
index 00000000..8577ea07
--- /dev/null
+++ b/frontend/crawlab-ui/public/js/login-canvas-three.js
@@ -0,0 +1,281 @@
+const THREE = window.THREE;
+
+class ThreeJSApp {
+ constructor() {
+ this.frameId = null;
+ this.fps = 60;
+ this.fpsInterval = 1000 / this.fps;
+ this.then = Date.now();
+ this.cameraRange = 3;
+ this.scene = new THREE.Scene();
+ this.renderer = new THREE.WebGLRenderer({ antialias: true });
+ this.camera = new THREE.PerspectiveCamera(
+ 35,
+ window.innerWidth / window.innerHeight,
+ 1,
+ 500
+ );
+
+ this.sceneGroup = new THREE.Object3D();
+ this.particularGroup = new THREE.Object3D();
+ this.modularGroup = new THREE.Object3D();
+
+ this.mouse = new THREE.Vector2();
+ this.INTERSECTED = null;
+ this.cameraValue = false;
+ this.uSpeed = 0.1;
+
+ this.initRenderer();
+ this.initScene();
+ this.initCamera();
+ this.initLights();
+ this.initObjects();
+ this.initRaycaster();
+ this.animate();
+ }
+
+ initRenderer() {
+ this.renderer.setSize(window.innerWidth, window.innerHeight);
+ this.renderer.shadowMap.enabled = false;
+ this.renderer.shadowMap.type = THREE.PCFSoftShadowMap;
+ let container = document.querySelector('#login-canvas');
+ container.appendChild(this.renderer.domElement);
+ }
+
+ initScene() {
+ const setColor = 0x000000;
+ this.scene.background = new THREE.Color(setColor);
+ this.scene.fog = new THREE.Fog(setColor, 2.5, 3.5);
+ this.scene.add(this.sceneGroup);
+ this.scene.add(this.modularGroup);
+ }
+
+ initCamera() {
+ this.camera.position.set(0, 0, this.cameraRange);
+ window.addEventListener('resize', () => this.onWindowResize(), false);
+ }
+
+ initLights() {
+ const light = new THREE.SpotLight(0xffffff, 1);
+ light.position.set(5, 5, 2);
+ light.castShadow = true;
+ light.shadow.mapSize.width = 10000;
+ light.shadow.mapSize.height = 10000;
+ light.penumbra = 0.5;
+
+ const lightBack = new THREE.PointLight(0x409eff, 1);
+ lightBack.position.set(0, -3, -1);
+
+ const rectLight = new THREE.RectAreaLight(0x409eff, 50, 1, 1);
+ rectLight.position.set(0, 0, 1);
+ rectLight.lookAt(0, 0, 0);
+
+ this.scene.add(light);
+ this.scene.add(lightBack);
+ this.scene.add(rectLight);
+ }
+
+ generateParticle(num, amp = 2) {
+ const material = new THREE.MeshPhysicalMaterial({
+ color: 0xffffff,
+ side: THREE.DoubleSide,
+ });
+ const geometry = new THREE.CircleGeometry(0.2, 5);
+
+ for (let i = 1; i < num; i++) {
+ const pScale = 0.001 + Math.abs(this.mathRandom(0.03));
+ const particular = new THREE.Mesh(geometry, material);
+ particular.position.set(
+ this.mathRandom(amp),
+ this.mathRandom(amp),
+ this.mathRandom(amp)
+ );
+ particular.rotation.set(
+ this.mathRandom(),
+ this.mathRandom(),
+ this.mathRandom()
+ );
+ particular.scale.set(pScale, pScale, pScale);
+ particular.speedValue = this.mathRandom(1);
+ this.particularGroup.add(particular);
+ }
+ }
+
+ mathRandom(num = 1) {
+ return -Math.random() * num + Math.random() * num;
+ }
+
+ initObjects() {
+ this.generateParticle(500, 2);
+ this.sceneGroup.add(this.particularGroup);
+
+ for (let i = 0; i < 10; i++) {
+ const geometry = new THREE.IcosahedronGeometry(1);
+ const material = new THREE.MeshStandardMaterial({
+ flatShading: THREE.SmoothShading,
+ color: 0x111111,
+ transparent: false,
+ opacity: 1,
+ wireframe: false,
+ });
+ const cube = new THREE.Mesh(geometry, material);
+ cube.speedRotation = Math.random() * 0.2;
+ cube.positionX = this.mathRandom();
+ cube.positionY = this.mathRandom();
+ cube.positionZ = this.mathRandom();
+ cube.castShadow = true;
+ cube.receiveShadow = true;
+
+ const newScaleValue = this.mathRandom(0.1);
+ cube.scale.set(newScaleValue, newScaleValue, newScaleValue);
+ cube.rotation.set(
+ this.mathRandom((180 * Math.PI) / 180),
+ this.mathRandom((180 * Math.PI) / 180),
+ this.mathRandom((180 * Math.PI) / 180)
+ );
+ cube.position.set(cube.positionX, cube.positionY, cube.positionZ);
+ this.modularGroup.add(cube);
+ }
+ }
+
+ initRaycaster() {
+ this.raycaster = new THREE.Raycaster();
+ }
+
+ addEventListeners() {
+ window.addEventListener(
+ 'mousedown',
+ event => this.onMouseDown(event),
+ false
+ );
+ window.addEventListener('mouseup', event => this.onMouseUp(event), false);
+ window.addEventListener(
+ 'mousemove',
+ event => this.onMouseMove(event),
+ false
+ );
+ }
+
+ onWindowResize() {
+ this.camera.aspect = window.innerWidth / window.innerHeight;
+ this.camera.updateProjectionMatrix();
+ this.renderer.setSize(window.innerWidth, window.innerHeight);
+ }
+
+ onMouseMove(event) {
+ event.preventDefault();
+ this.mouse.x = (event.clientX / window.innerWidth) * 2 - 1;
+ this.mouse.y = -(event.clientY / window.innerHeight) * 2 + 1;
+ }
+
+ onMouseDown(event) {
+ event.preventDefault();
+ this.onMouseMove(event);
+ this.raycaster.setFromCamera(this.mouse, this.camera);
+ const intersects = this.raycaster.intersectObjects(
+ this.modularGroup.children
+ );
+ if (intersects.length > 0) {
+ this.cameraValue = false;
+ if (this.INTERSECTED !== intersects[0].object) {
+ if (this.INTERSECTED) {
+ this.INTERSECTED.material.emissive.setHex(
+ this.INTERSECTED.currentHex
+ );
+ }
+ this.INTERSECTED = intersects[0].object;
+ this.INTERSECTED.currentHex =
+ this.INTERSECTED.material.emissive.getHex();
+ this.INTERSECTED.material.emissive.setHex(0xffff00);
+
+ TweenMax.to(this.camera.position, 1, {
+ x: this.INTERSECTED.position.x,
+ y: this.INTERSECTED.position.y,
+ z: this.INTERSECTED.position.z + 3,
+ ease: Power2.easeInOut,
+ });
+ } else {
+ if (this.INTERSECTED) {
+ this.INTERSECTED.material.emissive.setHex(
+ this.INTERSECTED.currentHex
+ );
+ }
+ this.INTERSECTED = null;
+ }
+ }
+ }
+
+ onMouseUp(event) {
+ // Add your onMouseUp logic if needed
+ }
+
+ animate() {
+ const time = performance.now() * 0.0003;
+ this.frameId = requestAnimationFrame(() => this.animate());
+
+ for (const object of this.particularGroup.children) {
+ object.rotation.x += object.speedValue / 10;
+ object.rotation.y += object.speedValue / 10;
+ object.rotation.z += object.speedValue / 10;
+ }
+
+ const ratio = 0.1;
+
+ for (const cube of this.modularGroup.children) {
+ cube.rotation.x += 0.008 * ratio;
+ cube.rotation.y += 0.005 * ratio;
+ cube.rotation.z += 0.003 * ratio;
+ cube.position.x = Math.sin(time * cube.positionZ) * cube.positionY;
+ cube.position.y = Math.cos(time * cube.positionX) * cube.positionZ;
+ cube.position.z = Math.sin(time * cube.positionY) * cube.positionX;
+ }
+
+ this.particularGroup.rotation.y += 0.005 * ratio;
+ this.modularGroup.rotation.y -=
+ (this.mouse.x * 4 + this.modularGroup.rotation.y) * this.uSpeed * ratio;
+ this.modularGroup.rotation.x -=
+ (-this.mouse.y * 4 + this.modularGroup.rotation.x) * this.uSpeed * ratio;
+ this.camera.lookAt(this.scene.position);
+
+ // this.logoGroup.rotation.x += 0.01;
+ // this.logoGroup.rotation.y += 0.01;
+
+ const now = Date.now();
+ const elapsed = now - this.then;
+ if (elapsed > this.fpsInterval) {
+ this.then = now - (elapsed % this.fpsInterval);
+ this.renderer.render(this.scene, this.camera);
+ }
+ }
+
+ stopAnimation() {
+ if (this.frameId) {
+ cancelAnimationFrame(this.frameId); // 使用frameId来停止动画
+ this.frameId = null; // 清空frameId
+ }
+ }
+
+ dispose() {
+ this.stopAnimation(); // 停止动画
+ this.scene.traverse(object => {
+ if (object.material) {
+ object.material.dispose();
+ }
+ if (object.geometry) {
+ object.geometry.dispose();
+ }
+ });
+ this.renderer.dispose();
+ this.renderer.domElement.remove();
+ }
+}
+
+window.initCanvas = function () {
+ window.threeJSApp = new ThreeJSApp();
+};
+window.resetCanvas = function () {
+ if (window.threeJSApp) {
+ window.threeJSApp.dispose();
+ window.threeJSApp = null;
+ }
+};
diff --git a/frontend/crawlab-ui/public/js/login-canvas.js b/frontend/crawlab-ui/public/js/login-canvas.js
new file mode 100644
index 00000000..cee41edf
--- /dev/null
+++ b/frontend/crawlab-ui/public/js/login-canvas.js
@@ -0,0 +1,292 @@
+function initCanvas() {
+ let canvas,
+ ctx,
+ circ,
+ nodes,
+ mouse,
+ SENSITIVITY,
+ SIBLINGS_LIMIT,
+ DENSITY,
+ NODES_QTY,
+ ANCHOR_LENGTH,
+ MOUSE_RADIUS,
+ TURBULENCE,
+ MOUSE_MOVING_TURBULENCE,
+ MOUSE_ANGLE_TURBULENCE,
+ MOUSE_MOVING_RADIUS,
+ BASE_BRIGHTNESS,
+ RADIUS_DEGRADE,
+ SAMPLE_SIZE;
+
+ let handle;
+
+ // how close next node must be to activate connection (in px)
+ // shorter distance == better connection (line width)
+ SENSITIVITY = 200;
+ // note that siblings limit is not 'accurate' as the node can actually have more connections than this value that's because the node accepts sibling nodes with no regard to their current connections this is acceptable because potential fix would not result in significant visual difference
+ // more siblings == bigger node
+ SIBLINGS_LIMIT = 10;
+ // default node margin
+ DENSITY = 100;
+ // total number of nodes used (incremented after creation)
+ NODES_QTY = 0;
+ // avoid nodes spreading
+ ANCHOR_LENGTH = 100;
+ // highlight radius
+ MOUSE_RADIUS = 200;
+ // turbulence of randomness
+ TURBULENCE = 3;
+ // turbulence of mouse moving
+ MOUSE_MOVING_TURBULENCE = 50;
+ // turbulence of mouse moving angle
+ MOUSE_ANGLE_TURBULENCE = 0.002;
+ // moving radius of mouse
+ MOUSE_MOVING_RADIUS = 600;
+ // base brightness
+ BASE_BRIGHTNESS = 0.12;
+ // radius degrade
+ RADIUS_DEGRADE = 0.4;
+ // sample size
+ SAMPLE_SIZE = 0.5;
+
+ circ = 2 * Math.PI;
+ nodes = [];
+
+ canvas = document.querySelector('#canvas');
+ if (!canvas) return;
+ resizeWindow();
+ ctx = canvas.getContext('2d');
+ if (!ctx) {
+ alert("Ooops! Your browser does not support canvas :'(");
+ }
+
+ function Mouse(x, y) {
+ this.anchorX = x;
+ this.anchorY = y;
+ this.x = x;
+ this.y = y - MOUSE_RADIUS / 2;
+ this.angle = 0;
+ }
+
+ Mouse.prototype.computePosition = function () {
+ // this.x = this.anchorX + MOUSE_MOVING_RADIUS / 2 * Math.sin(this.angle)
+ // this.y = this.anchorY - MOUSE_MOVING_RADIUS / 2 * Math.cos(this.angle)
+ };
+
+ Mouse.prototype.move = function () {
+ let vx = Math.random() * MOUSE_MOVING_TURBULENCE;
+ let vy = Math.random() * MOUSE_MOVING_TURBULENCE;
+ if (
+ this.x + vx + MOUSE_RADIUS / 2 > window.innerWidth ||
+ this.x + vx - MOUSE_RADIUS / 2 < 0
+ ) {
+ vx = -vx;
+ }
+ if (
+ this.y + vy + MOUSE_RADIUS / 2 > window.innerHeight ||
+ this.y + vy - MOUSE_RADIUS / 2 < 0
+ ) {
+ vy = -vy;
+ }
+ this.x += vx;
+ this.y += vy;
+ // this.angle += Math.random() * MOUSE_ANGLE_TURBULENCE * 2 * Math.PI
+ // this.angle -= Math.floor(this.angle / (2 * Math.PI)) * 2 * Math.PI
+ // this.computePosition()
+ };
+
+ function Node(x, y) {
+ this.anchorX = x;
+ this.anchorY = y;
+ this.x = Math.random() * (x - (x - ANCHOR_LENGTH)) + (x - ANCHOR_LENGTH);
+ this.y = Math.random() * (y - (y - ANCHOR_LENGTH)) + (y - ANCHOR_LENGTH);
+ this.vx = Math.random() * TURBULENCE - 1;
+ this.vy = Math.random() * TURBULENCE - 1;
+ this.energy = Math.random() * 100;
+ this.radius = Math.random();
+ this.siblings = [];
+ this.brightness = 0;
+ }
+
+ Node.prototype.drawNode = function () {
+ let color = 'rgba(64, 156, 255, ' + this.brightness + ')';
+ ctx.beginPath();
+ ctx.arc(
+ this.x,
+ this.y,
+ 2 * this.radius + (2 * this.siblings.length) / SIBLINGS_LIMIT / 1.5,
+ 0,
+ circ
+ );
+ ctx.fillStyle = color;
+ ctx.fill();
+ };
+
+ Node.prototype.drawConnections = function () {
+ for (let i = 0; i < this.siblings.length; i++) {
+ let color = 'rgba(64, 156, 255, ' + this.brightness + ')';
+ ctx.beginPath();
+ ctx.moveTo(this.x, this.y);
+ ctx.lineTo(this.siblings[i].x, this.siblings[i].y);
+ ctx.lineWidth = 1 - calcDistance(this, this.siblings[i]) / SENSITIVITY;
+ ctx.strokeStyle = color;
+ ctx.stroke();
+ }
+ };
+
+ Node.prototype.moveNode = function () {
+ this.energy -= 2;
+ if (this.energy < 1) {
+ this.energy = Math.random() * 100;
+ if (this.x - this.anchorX < -ANCHOR_LENGTH) {
+ this.vx = Math.random() * TURBULENCE;
+ } else if (this.x - this.anchorX > ANCHOR_LENGTH) {
+ this.vx = Math.random() * -TURBULENCE;
+ } else {
+ this.vx = Math.random() * 2 * TURBULENCE - TURBULENCE;
+ }
+ if (this.y - this.anchorY < -ANCHOR_LENGTH) {
+ this.vy = Math.random() * TURBULENCE;
+ } else if (this.y - this.anchorY > ANCHOR_LENGTH) {
+ this.vy = Math.random() * -TURBULENCE;
+ } else {
+ this.vy = Math.random() * 2 * TURBULENCE - TURBULENCE;
+ }
+ }
+ this.x += (this.vx * this.energy) / 100;
+ this.y += (this.vy * this.energy) / 100;
+ };
+
+ function Handle() {
+ this.isStopped = false;
+ }
+
+ Handle.prototype.stop = function () {
+ this.isStopped = true;
+ };
+
+ function initNodes() {
+ ctx.clearRect(0, 0, canvas.width, canvas.height);
+ nodes = [];
+ for (let i = DENSITY; i < canvas.width; i += DENSITY) {
+ for (let j = DENSITY; j < canvas.height; j += DENSITY) {
+ nodes.push(new Node(i, j));
+ NODES_QTY++;
+ }
+ }
+ }
+
+ function initMouse() {
+ mouse = new Mouse(canvas.width / 2, canvas.height / 2);
+ }
+
+ function initHandle() {
+ handle = new Handle();
+ }
+
+ function calcDistance(node1, node2) {
+ return Math.sqrt(
+ Math.pow(node1.x - node2.x, 2) + Math.pow(node1.y - node2.y, 2)
+ );
+ }
+
+ function findSiblings() {
+ let node1, node2, distance;
+ for (let i = 0; i < NODES_QTY; i++) {
+ node1 = nodes[i];
+ node1.siblings = [];
+ for (let j = 0; j < NODES_QTY; j++) {
+ node2 = nodes[j];
+ if (node1 !== node2) {
+ distance = calcDistance(node1, node2);
+ if (distance < SENSITIVITY) {
+ if (node1.siblings.length < SIBLINGS_LIMIT) {
+ node1.siblings.push(node2);
+ } else {
+ let node_sibling_distance = 0;
+ let max_distance = 0;
+ let s;
+ for (let k = 0; k < SIBLINGS_LIMIT; k++) {
+ node_sibling_distance = calcDistance(node1, node1.siblings[k]);
+ if (node_sibling_distance > max_distance) {
+ max_distance = node_sibling_distance;
+ s = k;
+ }
+ }
+ if (distance < max_distance) {
+ node1.siblings.splice(s, 1);
+ node1.siblings.push(node2);
+ }
+ }
+ }
+ }
+ }
+ }
+ }
+
+ function redrawScene() {
+ if (handle && handle.isStopped) {
+ return;
+ }
+ resizeWindow();
+ ctx.clearRect(0, 0, canvas.width, canvas.height);
+ findSiblings();
+ let i, node, distance;
+ for (i = 0; i < NODES_QTY; i++) {
+ node = nodes[i];
+ distance = calcDistance(
+ {
+ x: mouse.x,
+ y: mouse.y,
+ },
+ node
+ );
+ node.brightness =
+ (1 - Math.log((distance / MOUSE_RADIUS) * RADIUS_DEGRADE)) *
+ BASE_BRIGHTNESS;
+ }
+ for (i = 0; i < NODES_QTY; i++) {
+ node = nodes[i];
+ if (node.brightness) {
+ node.drawNode();
+ node.drawConnections();
+ }
+ node.moveNode();
+ }
+ // mouse.move()
+ setTimeout(() => {
+ requestAnimationFrame(redrawScene);
+ }, 50);
+ }
+
+ function initHandlers() {
+ document.addEventListener('resize', resizeWindow, { passive: true });
+ // canvas.addEventListener('mousemove', mousemoveHandler, false)
+ }
+
+ function resizeWindow() {
+ canvas.width = window.innerWidth;
+ canvas.height = window.innerHeight;
+ }
+
+ function mousemoveHandler(e) {
+ mouse.x = e.clientX;
+ mouse.y = e.clientY;
+ }
+
+ function init() {
+ initHandlers();
+ initNodes();
+ initMouse();
+ initHandle();
+ redrawScene();
+ }
+
+ function reset() {
+ handle.isStopped = true;
+ }
+
+ init();
+
+ window.disposeCanvas = reset;
+}
diff --git a/frontend/crawlab-ui/public/svg/logo-main.svg b/frontend/crawlab-ui/public/svg/logo-main.svg
new file mode 100644
index 00000000..aed3ae3a
--- /dev/null
+++ b/frontend/crawlab-ui/public/svg/logo-main.svg
@@ -0,0 +1,161 @@
+
+
+
diff --git a/frontend/crawlab-ui/public/three/loaders/SVGLoader.js b/frontend/crawlab-ui/public/three/loaders/SVGLoader.js
new file mode 100644
index 00000000..12a547e6
--- /dev/null
+++ b/frontend/crawlab-ui/public/three/loaders/SVGLoader.js
@@ -0,0 +1,2619 @@
+( function () {
+
+ class SVGLoader extends THREE.Loader {
+
+ constructor( manager ) {
+
+ super( manager ); // Default dots per inch
+
+ this.defaultDPI = 90; // Accepted units: 'mm', 'cm', 'in', 'pt', 'pc', 'px'
+
+ this.defaultUnit = 'px';
+
+ }
+
+ load( url, onLoad, onProgress, onError ) {
+
+ const scope = this;
+ const loader = new THREE.FileLoader( scope.manager );
+ loader.setPath( scope.path );
+ loader.setRequestHeader( scope.requestHeader );
+ loader.setWithCredentials( scope.withCredentials );
+ loader.load( url, function ( text ) {
+
+ try {
+
+ onLoad( scope.parse( text ) );
+
+ } catch ( e ) {
+
+ if ( onError ) {
+
+ onError( e );
+
+ } else {
+
+ console.error( e );
+
+ }
+
+ scope.manager.itemError( url );
+
+ }
+
+ }, onProgress, onError );
+
+ }
+
+ parse( text ) {
+
+ const scope = this;
+
+ function parseNode( node, style ) {
+
+ if ( node.nodeType !== 1 ) return;
+ const transform = getNodeTransform( node );
+ let traverseChildNodes = true;
+ let path = null;
+
+ switch ( node.nodeName ) {
+
+ case 'svg':
+ break;
+
+ case 'style':
+ parseCSSStylesheet( node );
+ break;
+
+ case 'g':
+ style = parseStyle( node, style );
+ break;
+
+ case 'path':
+ style = parseStyle( node, style );
+ if ( node.hasAttribute( 'd' ) ) path = parsePathNode( node );
+ break;
+
+ case 'rect':
+ style = parseStyle( node, style );
+ path = parseRectNode( node );
+ break;
+
+ case 'polygon':
+ style = parseStyle( node, style );
+ path = parsePolygonNode( node );
+ break;
+
+ case 'polyline':
+ style = parseStyle( node, style );
+ path = parsePolylineNode( node );
+ break;
+
+ case 'circle':
+ style = parseStyle( node, style );
+ path = parseCircleNode( node );
+ break;
+
+ case 'ellipse':
+ style = parseStyle( node, style );
+ path = parseEllipseNode( node );
+ break;
+
+ case 'line':
+ style = parseStyle( node, style );
+ path = parseLineNode( node );
+ break;
+
+ case 'defs':
+ traverseChildNodes = false;
+ break;
+
+ case 'use':
+ style = parseStyle( node, style );
+ const usedNodeId = node.href.baseVal.substring( 1 );
+ const usedNode = node.viewportElement.getElementById( usedNodeId );
+
+ if ( usedNode ) {
+
+ parseNode( usedNode, style );
+
+ } else {
+
+ console.warn( 'SVGLoader: \'use node\' references non-existent node id: ' + usedNodeId );
+
+ }
+
+ break;
+
+ default: // console.log( node );
+
+ }
+
+ if ( path ) {
+
+ if ( style.fill !== undefined && style.fill !== 'none' ) {
+
+ path.color.setStyle( style.fill );
+
+ }
+
+ transformPath( path, currentTransform );
+ paths.push( path );
+ path.userData = {
+ node: node,
+ style: style
+ };
+
+ }
+
+ if ( traverseChildNodes ) {
+
+ const nodes = node.childNodes;
+
+ for ( let i = 0; i < nodes.length; i ++ ) {
+
+ parseNode( nodes[ i ], style );
+
+ }
+
+ }
+
+ if ( transform ) {
+
+ transformStack.pop();
+
+ if ( transformStack.length > 0 ) {
+
+ currentTransform.copy( transformStack[ transformStack.length - 1 ] );
+
+ } else {
+
+ currentTransform.identity();
+
+ }
+
+ }
+
+ }
+
+ function parsePathNode( node ) {
+
+ const path = new THREE.ShapePath();
+ const point = new THREE.Vector2();
+ const control = new THREE.Vector2();
+ const firstPoint = new THREE.Vector2();
+ let isFirstPoint = true;
+ let doSetFirstPoint = false;
+ const d = node.getAttribute( 'd' ); // console.log( d );
+
+ const commands = d.match( /[a-df-z][^a-df-z]*/ig );
+
+ for ( let i = 0, l = commands.length; i < l; i ++ ) {
+
+ const command = commands[ i ];
+ const type = command.charAt( 0 );
+ const data = command.substr( 1 ).trim();
+
+ if ( isFirstPoint === true ) {
+
+ doSetFirstPoint = true;
+ isFirstPoint = false;
+
+ }
+
+ let numbers;
+
+ switch ( type ) {
+
+ case 'M':
+ numbers = parseFloats( data );
+
+ for ( let j = 0, jl = numbers.length; j < jl; j += 2 ) {
+
+ point.x = numbers[ j + 0 ];
+ point.y = numbers[ j + 1 ];
+ control.x = point.x;
+ control.y = point.y;
+
+ if ( j === 0 ) {
+
+ path.moveTo( point.x, point.y );
+
+ } else {
+
+ path.lineTo( point.x, point.y );
+
+ }
+
+ if ( j === 0 && doSetFirstPoint === true ) firstPoint.copy( point );
+
+ }
+
+ break;
+
+ case 'H':
+ numbers = parseFloats( data );
+
+ for ( let j = 0, jl = numbers.length; j < jl; j ++ ) {
+
+ point.x = numbers[ j ];
+ control.x = point.x;
+ control.y = point.y;
+ path.lineTo( point.x, point.y );
+ if ( j === 0 && doSetFirstPoint === true ) firstPoint.copy( point );
+
+ }
+
+ break;
+
+ case 'V':
+ numbers = parseFloats( data );
+
+ for ( let j = 0, jl = numbers.length; j < jl; j ++ ) {
+
+ point.y = numbers[ j ];
+ control.x = point.x;
+ control.y = point.y;
+ path.lineTo( point.x, point.y );
+ if ( j === 0 && doSetFirstPoint === true ) firstPoint.copy( point );
+
+ }
+
+ break;
+
+ case 'L':
+ numbers = parseFloats( data );
+
+ for ( let j = 0, jl = numbers.length; j < jl; j += 2 ) {
+
+ point.x = numbers[ j + 0 ];
+ point.y = numbers[ j + 1 ];
+ control.x = point.x;
+ control.y = point.y;
+ path.lineTo( point.x, point.y );
+ if ( j === 0 && doSetFirstPoint === true ) firstPoint.copy( point );
+
+ }
+
+ break;
+
+ case 'C':
+ numbers = parseFloats( data );
+
+ for ( let j = 0, jl = numbers.length; j < jl; j += 6 ) {
+
+ path.bezierCurveTo( numbers[ j + 0 ], numbers[ j + 1 ], numbers[ j + 2 ], numbers[ j + 3 ], numbers[ j + 4 ], numbers[ j + 5 ] );
+ control.x = numbers[ j + 2 ];
+ control.y = numbers[ j + 3 ];
+ point.x = numbers[ j + 4 ];
+ point.y = numbers[ j + 5 ];
+ if ( j === 0 && doSetFirstPoint === true ) firstPoint.copy( point );
+
+ }
+
+ break;
+
+ case 'S':
+ numbers = parseFloats( data );
+
+ for ( let j = 0, jl = numbers.length; j < jl; j += 4 ) {
+
+ path.bezierCurveTo( getReflection( point.x, control.x ), getReflection( point.y, control.y ), numbers[ j + 0 ], numbers[ j + 1 ], numbers[ j + 2 ], numbers[ j + 3 ] );
+ control.x = numbers[ j + 0 ];
+ control.y = numbers[ j + 1 ];
+ point.x = numbers[ j + 2 ];
+ point.y = numbers[ j + 3 ];
+ if ( j === 0 && doSetFirstPoint === true ) firstPoint.copy( point );
+
+ }
+
+ break;
+
+ case 'Q':
+ numbers = parseFloats( data );
+
+ for ( let j = 0, jl = numbers.length; j < jl; j += 4 ) {
+
+ path.quadraticCurveTo( numbers[ j + 0 ], numbers[ j + 1 ], numbers[ j + 2 ], numbers[ j + 3 ] );
+ control.x = numbers[ j + 0 ];
+ control.y = numbers[ j + 1 ];
+ point.x = numbers[ j + 2 ];
+ point.y = numbers[ j + 3 ];
+ if ( j === 0 && doSetFirstPoint === true ) firstPoint.copy( point );
+
+ }
+
+ break;
+
+ case 'T':
+ numbers = parseFloats( data );
+
+ for ( let j = 0, jl = numbers.length; j < jl; j += 2 ) {
+
+ const rx = getReflection( point.x, control.x );
+ const ry = getReflection( point.y, control.y );
+ path.quadraticCurveTo( rx, ry, numbers[ j + 0 ], numbers[ j + 1 ] );
+ control.x = rx;
+ control.y = ry;
+ point.x = numbers[ j + 0 ];
+ point.y = numbers[ j + 1 ];
+ if ( j === 0 && doSetFirstPoint === true ) firstPoint.copy( point );
+
+ }
+
+ break;
+
+ case 'A':
+ numbers = parseFloats( data, [ 3, 4 ], 7 );
+
+ for ( let j = 0, jl = numbers.length; j < jl; j += 7 ) {
+
+ // skip command if start point == end point
+ if ( numbers[ j + 5 ] == point.x && numbers[ j + 6 ] == point.y ) continue;
+ const start = point.clone();
+ point.x = numbers[ j + 5 ];
+ point.y = numbers[ j + 6 ];
+ control.x = point.x;
+ control.y = point.y;
+ parseArcCommand( path, numbers[ j ], numbers[ j + 1 ], numbers[ j + 2 ], numbers[ j + 3 ], numbers[ j + 4 ], start, point );
+ if ( j === 0 && doSetFirstPoint === true ) firstPoint.copy( point );
+
+ }
+
+ break;
+
+ case 'm':
+ numbers = parseFloats( data );
+
+ for ( let j = 0, jl = numbers.length; j < jl; j += 2 ) {
+
+ point.x += numbers[ j + 0 ];
+ point.y += numbers[ j + 1 ];
+ control.x = point.x;
+ control.y = point.y;
+
+ if ( j === 0 ) {
+
+ path.moveTo( point.x, point.y );
+
+ } else {
+
+ path.lineTo( point.x, point.y );
+
+ }
+
+ if ( j === 0 && doSetFirstPoint === true ) firstPoint.copy( point );
+
+ }
+
+ break;
+
+ case 'h':
+ numbers = parseFloats( data );
+
+ for ( let j = 0, jl = numbers.length; j < jl; j ++ ) {
+
+ point.x += numbers[ j ];
+ control.x = point.x;
+ control.y = point.y;
+ path.lineTo( point.x, point.y );
+ if ( j === 0 && doSetFirstPoint === true ) firstPoint.copy( point );
+
+ }
+
+ break;
+
+ case 'v':
+ numbers = parseFloats( data );
+
+ for ( let j = 0, jl = numbers.length; j < jl; j ++ ) {
+
+ point.y += numbers[ j ];
+ control.x = point.x;
+ control.y = point.y;
+ path.lineTo( point.x, point.y );
+ if ( j === 0 && doSetFirstPoint === true ) firstPoint.copy( point );
+
+ }
+
+ break;
+
+ case 'l':
+ numbers = parseFloats( data );
+
+ for ( let j = 0, jl = numbers.length; j < jl; j += 2 ) {
+
+ point.x += numbers[ j + 0 ];
+ point.y += numbers[ j + 1 ];
+ control.x = point.x;
+ control.y = point.y;
+ path.lineTo( point.x, point.y );
+ if ( j === 0 && doSetFirstPoint === true ) firstPoint.copy( point );
+
+ }
+
+ break;
+
+ case 'c':
+ numbers = parseFloats( data );
+
+ for ( let j = 0, jl = numbers.length; j < jl; j += 6 ) {
+
+ path.bezierCurveTo( point.x + numbers[ j + 0 ], point.y + numbers[ j + 1 ], point.x + numbers[ j + 2 ], point.y + numbers[ j + 3 ], point.x + numbers[ j + 4 ], point.y + numbers[ j + 5 ] );
+ control.x = point.x + numbers[ j + 2 ];
+ control.y = point.y + numbers[ j + 3 ];
+ point.x += numbers[ j + 4 ];
+ point.y += numbers[ j + 5 ];
+ if ( j === 0 && doSetFirstPoint === true ) firstPoint.copy( point );
+
+ }
+
+ break;
+
+ case 's':
+ numbers = parseFloats( data );
+
+ for ( let j = 0, jl = numbers.length; j < jl; j += 4 ) {
+
+ path.bezierCurveTo( getReflection( point.x, control.x ), getReflection( point.y, control.y ), point.x + numbers[ j + 0 ], point.y + numbers[ j + 1 ], point.x + numbers[ j + 2 ], point.y + numbers[ j + 3 ] );
+ control.x = point.x + numbers[ j + 0 ];
+ control.y = point.y + numbers[ j + 1 ];
+ point.x += numbers[ j + 2 ];
+ point.y += numbers[ j + 3 ];
+ if ( j === 0 && doSetFirstPoint === true ) firstPoint.copy( point );
+
+ }
+
+ break;
+
+ case 'q':
+ numbers = parseFloats( data );
+
+ for ( let j = 0, jl = numbers.length; j < jl; j += 4 ) {
+
+ path.quadraticCurveTo( point.x + numbers[ j + 0 ], point.y + numbers[ j + 1 ], point.x + numbers[ j + 2 ], point.y + numbers[ j + 3 ] );
+ control.x = point.x + numbers[ j + 0 ];
+ control.y = point.y + numbers[ j + 1 ];
+ point.x += numbers[ j + 2 ];
+ point.y += numbers[ j + 3 ];
+ if ( j === 0 && doSetFirstPoint === true ) firstPoint.copy( point );
+
+ }
+
+ break;
+
+ case 't':
+ numbers = parseFloats( data );
+
+ for ( let j = 0, jl = numbers.length; j < jl; j += 2 ) {
+
+ const rx = getReflection( point.x, control.x );
+ const ry = getReflection( point.y, control.y );
+ path.quadraticCurveTo( rx, ry, point.x + numbers[ j + 0 ], point.y + numbers[ j + 1 ] );
+ control.x = rx;
+ control.y = ry;
+ point.x = point.x + numbers[ j + 0 ];
+ point.y = point.y + numbers[ j + 1 ];
+ if ( j === 0 && doSetFirstPoint === true ) firstPoint.copy( point );
+
+ }
+
+ break;
+
+ case 'a':
+ numbers = parseFloats( data, [ 3, 4 ], 7 );
+
+ for ( let j = 0, jl = numbers.length; j < jl; j += 7 ) {
+
+ // skip command if no displacement
+ if ( numbers[ j + 5 ] == 0 && numbers[ j + 6 ] == 0 ) continue;
+ const start = point.clone();
+ point.x += numbers[ j + 5 ];
+ point.y += numbers[ j + 6 ];
+ control.x = point.x;
+ control.y = point.y;
+ parseArcCommand( path, numbers[ j ], numbers[ j + 1 ], numbers[ j + 2 ], numbers[ j + 3 ], numbers[ j + 4 ], start, point );
+ if ( j === 0 && doSetFirstPoint === true ) firstPoint.copy( point );
+
+ }
+
+ break;
+
+ case 'Z':
+ case 'z':
+ path.currentPath.autoClose = true;
+
+ if ( path.currentPath.curves.length > 0 ) {
+
+ // Reset point to beginning of THREE.Path
+ point.copy( firstPoint );
+ path.currentPath.currentPoint.copy( point );
+ isFirstPoint = true;
+
+ }
+
+ break;
+
+ default:
+ console.warn( command );
+
+ } // console.log( type, parseFloats( data ), parseFloats( data ).length )
+
+
+ doSetFirstPoint = false;
+
+ }
+
+ return path;
+
+ }
+
+ function parseCSSStylesheet( node ) {
+
+ if ( ! node.sheet || ! node.sheet.cssRules || ! node.sheet.cssRules.length ) return;
+
+ for ( let i = 0; i < node.sheet.cssRules.length; i ++ ) {
+
+ const stylesheet = node.sheet.cssRules[ i ];
+ if ( stylesheet.type !== 1 ) continue;
+ const selectorList = stylesheet.selectorText.split( /,/gm ).filter( Boolean ).map( i => i.trim() );
+
+ for ( let j = 0; j < selectorList.length; j ++ ) {
+
+ stylesheets[ selectorList[ j ] ] = Object.assign( stylesheets[ selectorList[ j ] ] || {}, stylesheet.style );
+
+ }
+
+ }
+
+ }
+ /**
+ * https://www.w3.org/TR/SVG/implnote.html#ArcImplementationNotes
+ * https://mortoray.com/2017/02/16/rendering-an-svg-elliptical-arc-as-bezier-curves/ Appendix: Endpoint to center arc conversion
+ * From
+ * rx ry x-axis-rotation large-arc-flag sweep-flag x y
+ * To
+ * aX, aY, xRadius, yRadius, aStartAngle, aEndAngle, aClockwise, aRotation
+ */
+
+
+ function parseArcCommand( path, rx, ry, x_axis_rotation, large_arc_flag, sweep_flag, start, end ) {
+
+ if ( rx == 0 || ry == 0 ) {
+
+ // draw a line if either of the radii == 0
+ path.lineTo( end.x, end.y );
+ return;
+
+ }
+
+ x_axis_rotation = x_axis_rotation * Math.PI / 180; // Ensure radii are positive
+
+ rx = Math.abs( rx );
+ ry = Math.abs( ry ); // Compute (x1', y1')
+
+ const dx2 = ( start.x - end.x ) / 2.0;
+ const dy2 = ( start.y - end.y ) / 2.0;
+ const x1p = Math.cos( x_axis_rotation ) * dx2 + Math.sin( x_axis_rotation ) * dy2;
+ const y1p = - Math.sin( x_axis_rotation ) * dx2 + Math.cos( x_axis_rotation ) * dy2; // Compute (cx', cy')
+
+ let rxs = rx * rx;
+ let rys = ry * ry;
+ const x1ps = x1p * x1p;
+ const y1ps = y1p * y1p; // Ensure radii are large enough
+
+ const cr = x1ps / rxs + y1ps / rys;
+
+ if ( cr > 1 ) {
+
+ // scale up rx,ry equally so cr == 1
+ const s = Math.sqrt( cr );
+ rx = s * rx;
+ ry = s * ry;
+ rxs = rx * rx;
+ rys = ry * ry;
+
+ }
+
+ const dq = rxs * y1ps + rys * x1ps;
+ const pq = ( rxs * rys - dq ) / dq;
+ let q = Math.sqrt( Math.max( 0, pq ) );
+ if ( large_arc_flag === sweep_flag ) q = - q;
+ const cxp = q * rx * y1p / ry;
+ const cyp = - q * ry * x1p / rx; // Step 3: Compute (cx, cy) from (cx', cy')
+
+ const cx = Math.cos( x_axis_rotation ) * cxp - Math.sin( x_axis_rotation ) * cyp + ( start.x + end.x ) / 2;
+ const cy = Math.sin( x_axis_rotation ) * cxp + Math.cos( x_axis_rotation ) * cyp + ( start.y + end.y ) / 2; // Step 4: Compute θ1 and Δθ
+
+ const theta = svgAngle( 1, 0, ( x1p - cxp ) / rx, ( y1p - cyp ) / ry );
+ const delta = svgAngle( ( x1p - cxp ) / rx, ( y1p - cyp ) / ry, ( - x1p - cxp ) / rx, ( - y1p - cyp ) / ry ) % ( Math.PI * 2 );
+ path.currentPath.absellipse( cx, cy, rx, ry, theta, theta + delta, sweep_flag === 0, x_axis_rotation );
+
+ }
+
+ function svgAngle( ux, uy, vx, vy ) {
+
+ const dot = ux * vx + uy * vy;
+ const len = Math.sqrt( ux * ux + uy * uy ) * Math.sqrt( vx * vx + vy * vy );
+ let ang = Math.acos( Math.max( - 1, Math.min( 1, dot / len ) ) ); // floating point precision, slightly over values appear
+
+ if ( ux * vy - uy * vx < 0 ) ang = - ang;
+ return ang;
+
+ }
+ /*
+ * According to https://www.w3.org/TR/SVG/shapes.html#RectElementRXAttribute
+ * rounded corner should be rendered to elliptical arc, but bezier curve does the job well enough
+ */
+
+
+ function parseRectNode( node ) {
+
+ const x = parseFloatWithUnits( node.getAttribute( 'x' ) || 0 );
+ const y = parseFloatWithUnits( node.getAttribute( 'y' ) || 0 );
+ const rx = parseFloatWithUnits( node.getAttribute( 'rx' ) || 0 );
+ const ry = parseFloatWithUnits( node.getAttribute( 'ry' ) || 0 );
+ const w = parseFloatWithUnits( node.getAttribute( 'width' ) );
+ const h = parseFloatWithUnits( node.getAttribute( 'height' ) );
+ const path = new THREE.ShapePath();
+ path.moveTo( x + 2 * rx, y );
+ path.lineTo( x + w - 2 * rx, y );
+ if ( rx !== 0 || ry !== 0 ) path.bezierCurveTo( x + w, y, x + w, y, x + w, y + 2 * ry );
+ path.lineTo( x + w, y + h - 2 * ry );
+ if ( rx !== 0 || ry !== 0 ) path.bezierCurveTo( x + w, y + h, x + w, y + h, x + w - 2 * rx, y + h );
+ path.lineTo( x + 2 * rx, y + h );
+
+ if ( rx !== 0 || ry !== 0 ) {
+
+ path.bezierCurveTo( x, y + h, x, y + h, x, y + h - 2 * ry );
+
+ }
+
+ path.lineTo( x, y + 2 * ry );
+
+ if ( rx !== 0 || ry !== 0 ) {
+
+ path.bezierCurveTo( x, y, x, y, x + 2 * rx, y );
+
+ }
+
+ return path;
+
+ }
+
+ function parsePolygonNode( node ) {
+
+ function iterator( match, a, b ) {
+
+ const x = parseFloatWithUnits( a );
+ const y = parseFloatWithUnits( b );
+
+ if ( index === 0 ) {
+
+ path.moveTo( x, y );
+
+ } else {
+
+ path.lineTo( x, y );
+
+ }
+
+ index ++;
+
+ }
+
+ const regex = /(-?[\d\.?]+)[,|\s](-?[\d\.?]+)/g;
+ const path = new THREE.ShapePath();
+ let index = 0;
+ node.getAttribute( 'points' ).replace( regex, iterator );
+ path.currentPath.autoClose = true;
+ return path;
+
+ }
+
+ function parsePolylineNode( node ) {
+
+ function iterator( match, a, b ) {
+
+ const x = parseFloatWithUnits( a );
+ const y = parseFloatWithUnits( b );
+
+ if ( index === 0 ) {
+
+ path.moveTo( x, y );
+
+ } else {
+
+ path.lineTo( x, y );
+
+ }
+
+ index ++;
+
+ }
+
+ const regex = /(-?[\d\.?]+)[,|\s](-?[\d\.?]+)/g;
+ const path = new THREE.ShapePath();
+ let index = 0;
+ node.getAttribute( 'points' ).replace( regex, iterator );
+ path.currentPath.autoClose = false;
+ return path;
+
+ }
+
+ function parseCircleNode( node ) {
+
+ const x = parseFloatWithUnits( node.getAttribute( 'cx' ) || 0 );
+ const y = parseFloatWithUnits( node.getAttribute( 'cy' ) || 0 );
+ const r = parseFloatWithUnits( node.getAttribute( 'r' ) || 0 );
+ const subpath = new THREE.Path();
+ subpath.absarc( x, y, r, 0, Math.PI * 2 );
+ const path = new THREE.ShapePath();
+ path.subPaths.push( subpath );
+ return path;
+
+ }
+
+ function parseEllipseNode( node ) {
+
+ const x = parseFloatWithUnits( node.getAttribute( 'cx' ) || 0 );
+ const y = parseFloatWithUnits( node.getAttribute( 'cy' ) || 0 );
+ const rx = parseFloatWithUnits( node.getAttribute( 'rx' ) || 0 );
+ const ry = parseFloatWithUnits( node.getAttribute( 'ry' ) || 0 );
+ const subpath = new THREE.Path();
+ subpath.absellipse( x, y, rx, ry, 0, Math.PI * 2 );
+ const path = new THREE.ShapePath();
+ path.subPaths.push( subpath );
+ return path;
+
+ }
+
+ function parseLineNode( node ) {
+
+ const x1 = parseFloatWithUnits( node.getAttribute( 'x1' ) || 0 );
+ const y1 = parseFloatWithUnits( node.getAttribute( 'y1' ) || 0 );
+ const x2 = parseFloatWithUnits( node.getAttribute( 'x2' ) || 0 );
+ const y2 = parseFloatWithUnits( node.getAttribute( 'y2' ) || 0 );
+ const path = new THREE.ShapePath();
+ path.moveTo( x1, y1 );
+ path.lineTo( x2, y2 );
+ path.currentPath.autoClose = false;
+ return path;
+
+ } //
+
+
+ function parseStyle( node, style ) {
+
+ style = Object.assign( {}, style ); // clone style
+
+ let stylesheetStyles = {};
+
+ if ( node.hasAttribute( 'class' ) ) {
+
+ const classSelectors = node.getAttribute( 'class' ).split( /\s/ ).filter( Boolean ).map( i => i.trim() );
+
+ for ( let i = 0; i < classSelectors.length; i ++ ) {
+
+ stylesheetStyles = Object.assign( stylesheetStyles, stylesheets[ '.' + classSelectors[ i ] ] );
+
+ }
+
+ }
+
+ if ( node.hasAttribute( 'id' ) ) {
+
+ stylesheetStyles = Object.assign( stylesheetStyles, stylesheets[ '#' + node.getAttribute( 'id' ) ] );
+
+ }
+
+ function addStyle( svgName, jsName, adjustFunction ) {
+
+ if ( adjustFunction === undefined ) adjustFunction = function copy( v ) {
+
+ if ( v.startsWith( 'url' ) ) console.warn( 'SVGLoader: url access in attributes is not implemented.' );
+ return v;
+
+ };
+
+ if ( node.hasAttribute( svgName ) ) style[ jsName ] = adjustFunction( node.getAttribute( svgName ) );
+ if ( stylesheetStyles[ svgName ] ) style[ jsName ] = adjustFunction( stylesheetStyles[ svgName ] );
+ if ( node.style && node.style[ svgName ] !== '' ) style[ jsName ] = adjustFunction( node.style[ svgName ] );
+
+ }
+
+ function clamp( v ) {
+
+ return Math.max( 0, Math.min( 1, parseFloatWithUnits( v ) ) );
+
+ }
+
+ function positive( v ) {
+
+ return Math.max( 0, parseFloatWithUnits( v ) );
+
+ }
+
+ addStyle( 'fill', 'fill' );
+ addStyle( 'fill-opacity', 'fillOpacity', clamp );
+ addStyle( 'opacity', 'opacity', clamp );
+ addStyle( 'stroke', 'stroke' );
+ addStyle( 'stroke-opacity', 'strokeOpacity', clamp );
+ addStyle( 'stroke-width', 'strokeWidth', positive );
+ addStyle( 'stroke-linejoin', 'strokeLineJoin' );
+ addStyle( 'stroke-linecap', 'strokeLineCap' );
+ addStyle( 'stroke-miterlimit', 'strokeMiterLimit', positive );
+ addStyle( 'visibility', 'visibility' );
+ return style;
+
+ } // http://www.w3.org/TR/SVG11/implnote.html#PathElementImplementationNotes
+
+
+ function getReflection( a, b ) {
+
+ return a - ( b - a );
+
+ } // from https://github.com/ppvg/svg-numbers (MIT License)
+
+
+ function parseFloats( input, flags, stride ) {
+
+ if ( typeof input !== 'string' ) {
+
+ throw new TypeError( 'Invalid input: ' + typeof input );
+
+ } // Character groups
+
+
+ const RE = {
+ SEPARATOR: /[ \t\r\n\,.\-+]/,
+ WHITESPACE: /[ \t\r\n]/,
+ DIGIT: /[\d]/,
+ SIGN: /[-+]/,
+ POINT: /\./,
+ COMMA: /,/,
+ EXP: /e/i,
+ FLAGS: /[01]/
+ }; // States
+
+ const SEP = 0;
+ const INT = 1;
+ const FLOAT = 2;
+ const EXP = 3;
+ let state = SEP;
+ let seenComma = true;
+ let number = '',
+ exponent = '';
+ const result = [];
+
+ function throwSyntaxError( current, i, partial ) {
+
+ const error = new SyntaxError( 'Unexpected character "' + current + '" at index ' + i + '.' );
+ error.partial = partial;
+ throw error;
+
+ }
+
+ function newNumber() {
+
+ if ( number !== '' ) {
+
+ if ( exponent === '' ) result.push( Number( number ) ); else result.push( Number( number ) * Math.pow( 10, Number( exponent ) ) );
+
+ }
+
+ number = '';
+ exponent = '';
+
+ }
+
+ let current;
+ const length = input.length;
+
+ for ( let i = 0; i < length; i ++ ) {
+
+ current = input[ i ]; // check for flags
+
+ if ( Array.isArray( flags ) && flags.includes( result.length % stride ) && RE.FLAGS.test( current ) ) {
+
+ state = INT;
+ number = current;
+ newNumber();
+ continue;
+
+ } // parse until next number
+
+
+ if ( state === SEP ) {
+
+ // eat whitespace
+ if ( RE.WHITESPACE.test( current ) ) {
+
+ continue;
+
+ } // start new number
+
+
+ if ( RE.DIGIT.test( current ) || RE.SIGN.test( current ) ) {
+
+ state = INT;
+ number = current;
+ continue;
+
+ }
+
+ if ( RE.POINT.test( current ) ) {
+
+ state = FLOAT;
+ number = current;
+ continue;
+
+ } // throw on double commas (e.g. "1, , 2")
+
+
+ if ( RE.COMMA.test( current ) ) {
+
+ if ( seenComma ) {
+
+ throwSyntaxError( current, i, result );
+
+ }
+
+ seenComma = true;
+
+ }
+
+ } // parse integer part
+
+
+ if ( state === INT ) {
+
+ if ( RE.DIGIT.test( current ) ) {
+
+ number += current;
+ continue;
+
+ }
+
+ if ( RE.POINT.test( current ) ) {
+
+ number += current;
+ state = FLOAT;
+ continue;
+
+ }
+
+ if ( RE.EXP.test( current ) ) {
+
+ state = EXP;
+ continue;
+
+ } // throw on double signs ("-+1"), but not on sign as separator ("-1-2")
+
+
+ if ( RE.SIGN.test( current ) && number.length === 1 && RE.SIGN.test( number[ 0 ] ) ) {
+
+ throwSyntaxError( current, i, result );
+
+ }
+
+ } // parse decimal part
+
+
+ if ( state === FLOAT ) {
+
+ if ( RE.DIGIT.test( current ) ) {
+
+ number += current;
+ continue;
+
+ }
+
+ if ( RE.EXP.test( current ) ) {
+
+ state = EXP;
+ continue;
+
+ } // throw on double decimal points (e.g. "1..2")
+
+
+ if ( RE.POINT.test( current ) && number[ number.length - 1 ] === '.' ) {
+
+ throwSyntaxError( current, i, result );
+
+ }
+
+ } // parse exponent part
+
+
+ if ( state === EXP ) {
+
+ if ( RE.DIGIT.test( current ) ) {
+
+ exponent += current;
+ continue;
+
+ }
+
+ if ( RE.SIGN.test( current ) ) {
+
+ if ( exponent === '' ) {
+
+ exponent += current;
+ continue;
+
+ }
+
+ if ( exponent.length === 1 && RE.SIGN.test( exponent ) ) {
+
+ throwSyntaxError( current, i, result );
+
+ }
+
+ }
+
+ } // end of number
+
+
+ if ( RE.WHITESPACE.test( current ) ) {
+
+ newNumber();
+ state = SEP;
+ seenComma = false;
+
+ } else if ( RE.COMMA.test( current ) ) {
+
+ newNumber();
+ state = SEP;
+ seenComma = true;
+
+ } else if ( RE.SIGN.test( current ) ) {
+
+ newNumber();
+ state = INT;
+ number = current;
+
+ } else if ( RE.POINT.test( current ) ) {
+
+ newNumber();
+ state = FLOAT;
+ number = current;
+
+ } else {
+
+ throwSyntaxError( current, i, result );
+
+ }
+
+ } // add the last number found (if any)
+
+
+ newNumber();
+ return result;
+
+ } // Units
+
+
+ const units = [ 'mm', 'cm', 'in', 'pt', 'pc', 'px' ]; // Conversion: [ fromUnit ][ toUnit ] (-1 means dpi dependent)
+
+ const unitConversion = {
+ 'mm': {
+ 'mm': 1,
+ 'cm': 0.1,
+ 'in': 1 / 25.4,
+ 'pt': 72 / 25.4,
+ 'pc': 6 / 25.4,
+ 'px': - 1
+ },
+ 'cm': {
+ 'mm': 10,
+ 'cm': 1,
+ 'in': 1 / 2.54,
+ 'pt': 72 / 2.54,
+ 'pc': 6 / 2.54,
+ 'px': - 1
+ },
+ 'in': {
+ 'mm': 25.4,
+ 'cm': 2.54,
+ 'in': 1,
+ 'pt': 72,
+ 'pc': 6,
+ 'px': - 1
+ },
+ 'pt': {
+ 'mm': 25.4 / 72,
+ 'cm': 2.54 / 72,
+ 'in': 1 / 72,
+ 'pt': 1,
+ 'pc': 6 / 72,
+ 'px': - 1
+ },
+ 'pc': {
+ 'mm': 25.4 / 6,
+ 'cm': 2.54 / 6,
+ 'in': 1 / 6,
+ 'pt': 72 / 6,
+ 'pc': 1,
+ 'px': - 1
+ },
+ 'px': {
+ 'px': 1
+ }
+ };
+
+ function parseFloatWithUnits( string ) {
+
+ let theUnit = 'px';
+
+ if ( typeof string === 'string' || string instanceof String ) {
+
+ for ( let i = 0, n = units.length; i < n; i ++ ) {
+
+ const u = units[ i ];
+
+ if ( string.endsWith( u ) ) {
+
+ theUnit = u;
+ string = string.substring( 0, string.length - u.length );
+ break;
+
+ }
+
+ }
+
+ }
+
+ let scale = undefined;
+
+ if ( theUnit === 'px' && scope.defaultUnit !== 'px' ) {
+
+ // Conversion scale from pixels to inches, then to default units
+ scale = unitConversion[ 'in' ][ scope.defaultUnit ] / scope.defaultDPI;
+
+ } else {
+
+ scale = unitConversion[ theUnit ][ scope.defaultUnit ];
+
+ if ( scale < 0 ) {
+
+ // Conversion scale to pixels
+ scale = unitConversion[ theUnit ][ 'in' ] * scope.defaultDPI;
+
+ }
+
+ }
+
+ return scale * parseFloat( string );
+
+ } // Transforms
+
+
+ function getNodeTransform( node ) {
+
+ if ( ! ( node.hasAttribute( 'transform' ) || node.nodeName === 'use' && ( node.hasAttribute( 'x' ) || node.hasAttribute( 'y' ) ) ) ) {
+
+ return null;
+
+ }
+
+ const transform = parseNodeTransform( node );
+
+ if ( transformStack.length > 0 ) {
+
+ transform.premultiply( transformStack[ transformStack.length - 1 ] );
+
+ }
+
+ currentTransform.copy( transform );
+ transformStack.push( transform );
+ return transform;
+
+ }
+
+ function parseNodeTransform( node ) {
+
+ const transform = new THREE.Matrix3();
+ const currentTransform = tempTransform0;
+
+ if ( node.nodeName === 'use' && ( node.hasAttribute( 'x' ) || node.hasAttribute( 'y' ) ) ) {
+
+ const tx = parseFloatWithUnits( node.getAttribute( 'x' ) );
+ const ty = parseFloatWithUnits( node.getAttribute( 'y' ) );
+ transform.translate( tx, ty );
+
+ }
+
+ if ( node.hasAttribute( 'transform' ) ) {
+
+ const transformsTexts = node.getAttribute( 'transform' ).split( ')' );
+
+ for ( let tIndex = transformsTexts.length - 1; tIndex >= 0; tIndex -- ) {
+
+ const transformText = transformsTexts[ tIndex ].trim();
+ if ( transformText === '' ) continue;
+ const openParPos = transformText.indexOf( '(' );
+ const closeParPos = transformText.length;
+
+ if ( openParPos > 0 && openParPos < closeParPos ) {
+
+ const transformType = transformText.substr( 0, openParPos );
+ const array = parseFloats( transformText.substr( openParPos + 1, closeParPos - openParPos - 1 ) );
+ currentTransform.identity();
+
+ switch ( transformType ) {
+
+ case 'translate':
+ if ( array.length >= 1 ) {
+
+ const tx = array[ 0 ];
+ let ty = tx;
+
+ if ( array.length >= 2 ) {
+
+ ty = array[ 1 ];
+
+ }
+
+ currentTransform.translate( tx, ty );
+
+ }
+
+ break;
+
+ case 'rotate':
+ if ( array.length >= 1 ) {
+
+ let angle = 0;
+ let cx = 0;
+ let cy = 0; // Angle
+
+ angle = - array[ 0 ] * Math.PI / 180;
+
+ if ( array.length >= 3 ) {
+
+ // Center x, y
+ cx = array[ 1 ];
+ cy = array[ 2 ];
+
+ } // Rotate around center (cx, cy)
+
+
+ tempTransform1.identity().translate( - cx, - cy );
+ tempTransform2.identity().rotate( angle );
+ tempTransform3.multiplyMatrices( tempTransform2, tempTransform1 );
+ tempTransform1.identity().translate( cx, cy );
+ currentTransform.multiplyMatrices( tempTransform1, tempTransform3 );
+
+ }
+
+ break;
+
+ case 'scale':
+ if ( array.length >= 1 ) {
+
+ const scaleX = array[ 0 ];
+ let scaleY = scaleX;
+
+ if ( array.length >= 2 ) {
+
+ scaleY = array[ 1 ];
+
+ }
+
+ currentTransform.scale( scaleX, scaleY );
+
+ }
+
+ break;
+
+ case 'skewX':
+ if ( array.length === 1 ) {
+
+ currentTransform.set( 1, Math.tan( array[ 0 ] * Math.PI / 180 ), 0, 0, 1, 0, 0, 0, 1 );
+
+ }
+
+ break;
+
+ case 'skewY':
+ if ( array.length === 1 ) {
+
+ currentTransform.set( 1, 0, 0, Math.tan( array[ 0 ] * Math.PI / 180 ), 1, 0, 0, 0, 1 );
+
+ }
+
+ break;
+
+ case 'matrix':
+ if ( array.length === 6 ) {
+
+ currentTransform.set( array[ 0 ], array[ 2 ], array[ 4 ], array[ 1 ], array[ 3 ], array[ 5 ], 0, 0, 1 );
+
+ }
+
+ break;
+
+ }
+
+ }
+
+ transform.premultiply( currentTransform );
+
+ }
+
+ }
+
+ return transform;
+
+ }
+
+ function transformPath( path, m ) {
+
+ function transfVec2( v2 ) {
+
+ tempV3.set( v2.x, v2.y, 1 ).applyMatrix3( m );
+ v2.set( tempV3.x, tempV3.y );
+
+ }
+
+ const isRotated = isTransformRotated( m );
+ const subPaths = path.subPaths;
+
+ for ( let i = 0, n = subPaths.length; i < n; i ++ ) {
+
+ const subPath = subPaths[ i ];
+ const curves = subPath.curves;
+
+ for ( let j = 0; j < curves.length; j ++ ) {
+
+ const curve = curves[ j ];
+
+ if ( curve.isLineCurve ) {
+
+ transfVec2( curve.v1 );
+ transfVec2( curve.v2 );
+
+ } else if ( curve.isCubicBezierCurve ) {
+
+ transfVec2( curve.v0 );
+ transfVec2( curve.v1 );
+ transfVec2( curve.v2 );
+ transfVec2( curve.v3 );
+
+ } else if ( curve.isQuadraticBezierCurve ) {
+
+ transfVec2( curve.v0 );
+ transfVec2( curve.v1 );
+ transfVec2( curve.v2 );
+
+ } else if ( curve.isEllipseCurve ) {
+
+ if ( isRotated ) {
+
+ console.warn( 'SVGLoader: Elliptic arc or ellipse rotation or skewing is not implemented.' );
+
+ }
+
+ tempV2.set( curve.aX, curve.aY );
+ transfVec2( tempV2 );
+ curve.aX = tempV2.x;
+ curve.aY = tempV2.y;
+ curve.xRadius *= getTransformScaleX( m );
+ curve.yRadius *= getTransformScaleY( m );
+
+ }
+
+ }
+
+ }
+
+ }
+
+ function isTransformRotated( m ) {
+
+ return m.elements[ 1 ] !== 0 || m.elements[ 3 ] !== 0;
+
+ }
+
+ function getTransformScaleX( m ) {
+
+ const te = m.elements;
+ return Math.sqrt( te[ 0 ] * te[ 0 ] + te[ 1 ] * te[ 1 ] );
+
+ }
+
+ function getTransformScaleY( m ) {
+
+ const te = m.elements;
+ return Math.sqrt( te[ 3 ] * te[ 3 ] + te[ 4 ] * te[ 4 ] );
+
+ } //
+
+
+ const paths = [];
+ const stylesheets = {};
+ const transformStack = [];
+ const tempTransform0 = new THREE.Matrix3();
+ const tempTransform1 = new THREE.Matrix3();
+ const tempTransform2 = new THREE.Matrix3();
+ const tempTransform3 = new THREE.Matrix3();
+ const tempV2 = new THREE.Vector2();
+ const tempV3 = new THREE.Vector3();
+ const currentTransform = new THREE.Matrix3();
+ const xml = new DOMParser().parseFromString( text, 'image/svg+xml' ); // application/xml
+
+ parseNode( xml.documentElement, {
+ fill: '#000',
+ fillOpacity: 1,
+ strokeOpacity: 1,
+ strokeWidth: 1,
+ strokeLineJoin: 'miter',
+ strokeLineCap: 'butt',
+ strokeMiterLimit: 4
+ } );
+ const data = {
+ paths: paths,
+ xml: xml.documentElement
+ }; // console.log( paths );
+
+ return data;
+
+ }
+
+ static createShapes( shapePath ) {
+
+ // Param shapePath: a shapepath as returned by the parse function of this class
+ // Returns THREE.Shape object
+ const BIGNUMBER = 999999999;
+ const IntersectionLocationType = {
+ ORIGIN: 0,
+ DESTINATION: 1,
+ BETWEEN: 2,
+ LEFT: 3,
+ RIGHT: 4,
+ BEHIND: 5,
+ BEYOND: 6
+ };
+ const classifyResult = {
+ loc: IntersectionLocationType.ORIGIN,
+ t: 0
+ };
+
+ function findEdgeIntersection( a0, a1, b0, b1 ) {
+
+ const x1 = a0.x;
+ const x2 = a1.x;
+ const x3 = b0.x;
+ const x4 = b1.x;
+ const y1 = a0.y;
+ const y2 = a1.y;
+ const y3 = b0.y;
+ const y4 = b1.y;
+ const nom1 = ( x4 - x3 ) * ( y1 - y3 ) - ( y4 - y3 ) * ( x1 - x3 );
+ const nom2 = ( x2 - x1 ) * ( y1 - y3 ) - ( y2 - y1 ) * ( x1 - x3 );
+ const denom = ( y4 - y3 ) * ( x2 - x1 ) - ( x4 - x3 ) * ( y2 - y1 );
+ const t1 = nom1 / denom;
+ const t2 = nom2 / denom;
+
+ if ( denom === 0 && nom1 !== 0 || t1 <= 0 || t1 >= 1 || t2 < 0 || t2 > 1 ) {
+
+ //1. lines are parallel or edges don't intersect
+ return null;
+
+ } else if ( nom1 === 0 && denom === 0 ) {
+
+ //2. lines are colinear
+ //check if endpoints of edge2 (b0-b1) lies on edge1 (a0-a1)
+ for ( let i = 0; i < 2; i ++ ) {
+
+ classifyPoint( i === 0 ? b0 : b1, a0, a1 ); //find position of this endpoints relatively to edge1
+
+ if ( classifyResult.loc == IntersectionLocationType.ORIGIN ) {
+
+ const point = i === 0 ? b0 : b1;
+ return {
+ x: point.x,
+ y: point.y,
+ t: classifyResult.t
+ };
+
+ } else if ( classifyResult.loc == IntersectionLocationType.BETWEEN ) {
+
+ const x = + ( x1 + classifyResult.t * ( x2 - x1 ) ).toPrecision( 10 );
+ const y = + ( y1 + classifyResult.t * ( y2 - y1 ) ).toPrecision( 10 );
+ return {
+ x: x,
+ y: y,
+ t: classifyResult.t
+ };
+
+ }
+
+ }
+
+ return null;
+
+ } else {
+
+ //3. edges intersect
+ for ( let i = 0; i < 2; i ++ ) {
+
+ classifyPoint( i === 0 ? b0 : b1, a0, a1 );
+
+ if ( classifyResult.loc == IntersectionLocationType.ORIGIN ) {
+
+ const point = i === 0 ? b0 : b1;
+ return {
+ x: point.x,
+ y: point.y,
+ t: classifyResult.t
+ };
+
+ }
+
+ }
+
+ const x = + ( x1 + t1 * ( x2 - x1 ) ).toPrecision( 10 );
+ const y = + ( y1 + t1 * ( y2 - y1 ) ).toPrecision( 10 );
+ return {
+ x: x,
+ y: y,
+ t: t1
+ };
+
+ }
+
+ }
+
+ function classifyPoint( p, edgeStart, edgeEnd ) {
+
+ const ax = edgeEnd.x - edgeStart.x;
+ const ay = edgeEnd.y - edgeStart.y;
+ const bx = p.x - edgeStart.x;
+ const by = p.y - edgeStart.y;
+ const sa = ax * by - bx * ay;
+
+ if ( p.x === edgeStart.x && p.y === edgeStart.y ) {
+
+ classifyResult.loc = IntersectionLocationType.ORIGIN;
+ classifyResult.t = 0;
+ return;
+
+ }
+
+ if ( p.x === edgeEnd.x && p.y === edgeEnd.y ) {
+
+ classifyResult.loc = IntersectionLocationType.DESTINATION;
+ classifyResult.t = 1;
+ return;
+
+ }
+
+ if ( sa < - Number.EPSILON ) {
+
+ classifyResult.loc = IntersectionLocationType.LEFT;
+ return;
+
+ }
+
+ if ( sa > Number.EPSILON ) {
+
+ classifyResult.loc = IntersectionLocationType.RIGHT;
+ return;
+
+ }
+
+ if ( ax * bx < 0 || ay * by < 0 ) {
+
+ classifyResult.loc = IntersectionLocationType.BEHIND;
+ return;
+
+ }
+
+ if ( Math.sqrt( ax * ax + ay * ay ) < Math.sqrt( bx * bx + by * by ) ) {
+
+ classifyResult.loc = IntersectionLocationType.BEYOND;
+ return;
+
+ }
+
+ let t;
+
+ if ( ax !== 0 ) {
+
+ t = bx / ax;
+
+ } else {
+
+ t = by / ay;
+
+ }
+
+ classifyResult.loc = IntersectionLocationType.BETWEEN;
+ classifyResult.t = t;
+
+ }
+
+ function getIntersections( path1, path2 ) {
+
+ const intersectionsRaw = [];
+ const intersections = [];
+
+ for ( let index = 1; index < path1.length; index ++ ) {
+
+ const path1EdgeStart = path1[ index - 1 ];
+ const path1EdgeEnd = path1[ index ];
+
+ for ( let index2 = 1; index2 < path2.length; index2 ++ ) {
+
+ const path2EdgeStart = path2[ index2 - 1 ];
+ const path2EdgeEnd = path2[ index2 ];
+ const intersection = findEdgeIntersection( path1EdgeStart, path1EdgeEnd, path2EdgeStart, path2EdgeEnd );
+
+ if ( intersection !== null && intersectionsRaw.find( i => i.t <= intersection.t + Number.EPSILON && i.t >= intersection.t - Number.EPSILON ) === undefined ) {
+
+ intersectionsRaw.push( intersection );
+ intersections.push( new THREE.Vector2( intersection.x, intersection.y ) );
+
+ }
+
+ }
+
+ }
+
+ return intersections;
+
+ }
+
+ function getScanlineIntersections( scanline, boundingBox, paths ) {
+
+ const center = new THREE.Vector2();
+ boundingBox.getCenter( center );
+ const allIntersections = [];
+ paths.forEach( path => {
+
+ // check if the center of the bounding box is in the bounding box of the paths.
+ // this is a pruning method to limit the search of intersections in paths that can't envelop of the current path.
+ // if a path envelops another path. The center of that oter path, has to be inside the bounding box of the enveloping path.
+ if ( path.boundingBox.containsPoint( center ) ) {
+
+ const intersections = getIntersections( scanline, path.points );
+ intersections.forEach( p => {
+
+ allIntersections.push( {
+ identifier: path.identifier,
+ isCW: path.isCW,
+ point: p
+ } );
+
+ } );
+
+ }
+
+ } );
+ allIntersections.sort( ( i1, i2 ) => {
+
+ return i1.point.x - i2.point.x;
+
+ } );
+ return allIntersections;
+
+ }
+
+ function isHoleTo( simplePath, allPaths, scanlineMinX, scanlineMaxX, _fillRule ) {
+
+ if ( _fillRule === null || _fillRule === undefined || _fillRule === '' ) {
+
+ _fillRule = 'nonzero';
+
+ }
+
+ const centerBoundingBox = new THREE.Vector2();
+ simplePath.boundingBox.getCenter( centerBoundingBox );
+ const scanline = [ new THREE.Vector2( scanlineMinX, centerBoundingBox.y ), new THREE.Vector2( scanlineMaxX, centerBoundingBox.y ) ];
+ const scanlineIntersections = getScanlineIntersections( scanline, simplePath.boundingBox, allPaths );
+ scanlineIntersections.sort( ( i1, i2 ) => {
+
+ return i1.point.x - i2.point.x;
+
+ } );
+ const baseIntersections = [];
+ const otherIntersections = [];
+ scanlineIntersections.forEach( i => {
+
+ if ( i.identifier === simplePath.identifier ) {
+
+ baseIntersections.push( i );
+
+ } else {
+
+ otherIntersections.push( i );
+
+ }
+
+ } );
+ const firstXOfPath = baseIntersections[ 0 ].point.x; // build up the path hierarchy
+
+ const stack = [];
+ let i = 0;
+
+ while ( i < otherIntersections.length && otherIntersections[ i ].point.x < firstXOfPath ) {
+
+ if ( stack.length > 0 && stack[ stack.length - 1 ] === otherIntersections[ i ].identifier ) {
+
+ stack.pop();
+
+ } else {
+
+ stack.push( otherIntersections[ i ].identifier );
+
+ }
+
+ i ++;
+
+ }
+
+ stack.push( simplePath.identifier );
+
+ if ( _fillRule === 'evenodd' ) {
+
+ const isHole = stack.length % 2 === 0 ? true : false;
+ const isHoleFor = stack[ stack.length - 2 ];
+ return {
+ identifier: simplePath.identifier,
+ isHole: isHole,
+ for: isHoleFor
+ };
+
+ } else if ( _fillRule === 'nonzero' ) {
+
+ // check if path is a hole by counting the amount of paths with alternating rotations it has to cross.
+ let isHole = true;
+ let isHoleFor = null;
+ let lastCWValue = null;
+
+ for ( let i = 0; i < stack.length; i ++ ) {
+
+ const identifier = stack[ i ];
+
+ if ( isHole ) {
+
+ lastCWValue = allPaths[ identifier ].isCW;
+ isHole = false;
+ isHoleFor = identifier;
+
+ } else if ( lastCWValue !== allPaths[ identifier ].isCW ) {
+
+ lastCWValue = allPaths[ identifier ].isCW;
+ isHole = true;
+
+ }
+
+ }
+
+ return {
+ identifier: simplePath.identifier,
+ isHole: isHole,
+ for: isHoleFor
+ };
+
+ } else {
+
+ console.warn( 'fill-rule: "' + _fillRule + '" is currently not implemented.' );
+
+ }
+
+ } // check for self intersecting paths
+ // TODO
+ // check intersecting paths
+ // TODO
+ // prepare paths for hole detection
+
+
+ let identifier = 0;
+ let scanlineMinX = BIGNUMBER;
+ let scanlineMaxX = - BIGNUMBER;
+ let simplePaths = shapePath.subPaths.map( p => {
+
+ const points = p.getPoints();
+ let maxY = - BIGNUMBER;
+ let minY = BIGNUMBER;
+ let maxX = - BIGNUMBER;
+ let minX = BIGNUMBER; //points.forEach(p => p.y *= -1);
+
+ for ( let i = 0; i < points.length; i ++ ) {
+
+ const p = points[ i ];
+
+ if ( p.y > maxY ) {
+
+ maxY = p.y;
+
+ }
+
+ if ( p.y < minY ) {
+
+ minY = p.y;
+
+ }
+
+ if ( p.x > maxX ) {
+
+ maxX = p.x;
+
+ }
+
+ if ( p.x < minX ) {
+
+ minX = p.x;
+
+ }
+
+ } //
+
+
+ if ( scanlineMaxX <= maxX ) {
+
+ scanlineMaxX = maxX + 1;
+
+ }
+
+ if ( scanlineMinX >= minX ) {
+
+ scanlineMinX = minX - 1;
+
+ }
+
+ return {
+ points: points,
+ isCW: THREE.ShapeUtils.isClockWise( points ),
+ identifier: identifier ++,
+ boundingBox: new THREE.Box2( new THREE.Vector2( minX, minY ), new THREE.Vector2( maxX, maxY ) )
+ };
+
+ } );
+ simplePaths = simplePaths.filter( sp => sp.points.length > 0 ); // check if path is solid or a hole
+
+ const isAHole = simplePaths.map( p => isHoleTo( p, simplePaths, scanlineMinX, scanlineMaxX, shapePath.userData.style.fillRule ) );
+ const shapesToReturn = [];
+ simplePaths.forEach( p => {
+
+ const amIAHole = isAHole[ p.identifier ];
+
+ if ( ! amIAHole.isHole ) {
+
+ const shape = new THREE.Shape( p.points );
+ const holes = isAHole.filter( h => h.isHole && h.for === p.identifier );
+ holes.forEach( h => {
+
+ const path = simplePaths[ h.identifier ];
+ shape.holes.push( new THREE.Path( path.points ) );
+
+ } );
+ shapesToReturn.push( shape );
+
+ }
+
+ } );
+ return shapesToReturn;
+
+ }
+
+ static getStrokeStyle( width, color, lineJoin, lineCap, miterLimit ) {
+
+ // Param width: Stroke width
+ // Param color: As returned by THREE.Color.getStyle()
+ // Param lineJoin: One of "round", "bevel", "miter" or "miter-limit"
+ // Param lineCap: One of "round", "square" or "butt"
+ // Param miterLimit: Maximum join length, in multiples of the "width" parameter (join is truncated if it exceeds that distance)
+ // Returns style object
+ width = width !== undefined ? width : 1;
+ color = color !== undefined ? color : '#000';
+ lineJoin = lineJoin !== undefined ? lineJoin : 'miter';
+ lineCap = lineCap !== undefined ? lineCap : 'butt';
+ miterLimit = miterLimit !== undefined ? miterLimit : 4;
+ return {
+ strokeColor: color,
+ strokeWidth: width,
+ strokeLineJoin: lineJoin,
+ strokeLineCap: lineCap,
+ strokeMiterLimit: miterLimit
+ };
+
+ }
+
+ static pointsToStroke( points, style, arcDivisions, minDistance ) {
+
+ // Generates a stroke with some witdh around the given path.
+ // The path can be open or closed (last point equals to first point)
+ // Param points: Array of Vector2D (the path). Minimum 2 points.
+ // Param style: Object with SVG properties as returned by SVGLoader.getStrokeStyle(), or SVGLoader.parse() in the path.userData.style object
+ // Params arcDivisions: Arc divisions for round joins and endcaps. (Optional)
+ // Param minDistance: Points closer to this distance will be merged. (Optional)
+ // Returns THREE.BufferGeometry with stroke triangles (In plane z = 0). UV coordinates are generated ('u' along path. 'v' across it, from left to right)
+ const vertices = [];
+ const normals = [];
+ const uvs = [];
+
+ if ( SVGLoader.pointsToStrokeWithBuffers( points, style, arcDivisions, minDistance, vertices, normals, uvs ) === 0 ) {
+
+ return null;
+
+ }
+
+ const geometry = new THREE.BufferGeometry();
+ geometry.setAttribute( 'position', new THREE.Float32BufferAttribute( vertices, 3 ) );
+ geometry.setAttribute( 'normal', new THREE.Float32BufferAttribute( normals, 3 ) );
+ geometry.setAttribute( 'uv', new THREE.Float32BufferAttribute( uvs, 2 ) );
+ return geometry;
+
+ }
+
+ static pointsToStrokeWithBuffers( points, style, arcDivisions, minDistance, vertices, normals, uvs, vertexOffset ) {
+
+ // This function can be called to update existing arrays or buffers.
+ // Accepts same parameters as pointsToStroke, plus the buffers and optional offset.
+ // Param vertexOffset: Offset vertices to start writing in the buffers (3 elements/vertex for vertices and normals, and 2 elements/vertex for uvs)
+ // Returns number of written vertices / normals / uvs pairs
+ // if 'vertices' parameter is undefined no triangles will be generated, but the returned vertices count will still be valid (useful to preallocate the buffers)
+ // 'normals' and 'uvs' buffers are optional
+ const tempV2_1 = new THREE.Vector2();
+ const tempV2_2 = new THREE.Vector2();
+ const tempV2_3 = new THREE.Vector2();
+ const tempV2_4 = new THREE.Vector2();
+ const tempV2_5 = new THREE.Vector2();
+ const tempV2_6 = new THREE.Vector2();
+ const tempV2_7 = new THREE.Vector2();
+ const lastPointL = new THREE.Vector2();
+ const lastPointR = new THREE.Vector2();
+ const point0L = new THREE.Vector2();
+ const point0R = new THREE.Vector2();
+ const currentPointL = new THREE.Vector2();
+ const currentPointR = new THREE.Vector2();
+ const nextPointL = new THREE.Vector2();
+ const nextPointR = new THREE.Vector2();
+ const innerPoint = new THREE.Vector2();
+ const outerPoint = new THREE.Vector2();
+ arcDivisions = arcDivisions !== undefined ? arcDivisions : 12;
+ minDistance = minDistance !== undefined ? minDistance : 0.001;
+ vertexOffset = vertexOffset !== undefined ? vertexOffset : 0; // First ensure there are no duplicated points
+
+ points = removeDuplicatedPoints( points );
+ const numPoints = points.length;
+ if ( numPoints < 2 ) return 0;
+ const isClosed = points[ 0 ].equals( points[ numPoints - 1 ] );
+ let currentPoint;
+ let previousPoint = points[ 0 ];
+ let nextPoint;
+ const strokeWidth2 = style.strokeWidth / 2;
+ const deltaU = 1 / ( numPoints - 1 );
+ let u0 = 0,
+ u1;
+ let innerSideModified;
+ let joinIsOnLeftSide;
+ let isMiter;
+ let initialJoinIsOnLeftSide = false;
+ let numVertices = 0;
+ let currentCoordinate = vertexOffset * 3;
+ let currentCoordinateUV = vertexOffset * 2; // Get initial left and right stroke points
+
+ getNormal( points[ 0 ], points[ 1 ], tempV2_1 ).multiplyScalar( strokeWidth2 );
+ lastPointL.copy( points[ 0 ] ).sub( tempV2_1 );
+ lastPointR.copy( points[ 0 ] ).add( tempV2_1 );
+ point0L.copy( lastPointL );
+ point0R.copy( lastPointR );
+
+ for ( let iPoint = 1; iPoint < numPoints; iPoint ++ ) {
+
+ currentPoint = points[ iPoint ]; // Get next point
+
+ if ( iPoint === numPoints - 1 ) {
+
+ if ( isClosed ) {
+
+ // Skip duplicated initial point
+ nextPoint = points[ 1 ];
+
+ } else nextPoint = undefined;
+
+ } else {
+
+ nextPoint = points[ iPoint + 1 ];
+
+ } // Normal of previous segment in tempV2_1
+
+
+ const normal1 = tempV2_1;
+ getNormal( previousPoint, currentPoint, normal1 );
+ tempV2_3.copy( normal1 ).multiplyScalar( strokeWidth2 );
+ currentPointL.copy( currentPoint ).sub( tempV2_3 );
+ currentPointR.copy( currentPoint ).add( tempV2_3 );
+ u1 = u0 + deltaU;
+ innerSideModified = false;
+
+ if ( nextPoint !== undefined ) {
+
+ // Normal of next segment in tempV2_2
+ getNormal( currentPoint, nextPoint, tempV2_2 );
+ tempV2_3.copy( tempV2_2 ).multiplyScalar( strokeWidth2 );
+ nextPointL.copy( currentPoint ).sub( tempV2_3 );
+ nextPointR.copy( currentPoint ).add( tempV2_3 );
+ joinIsOnLeftSide = true;
+ tempV2_3.subVectors( nextPoint, previousPoint );
+
+ if ( normal1.dot( tempV2_3 ) < 0 ) {
+
+ joinIsOnLeftSide = false;
+
+ }
+
+ if ( iPoint === 1 ) initialJoinIsOnLeftSide = joinIsOnLeftSide;
+ tempV2_3.subVectors( nextPoint, currentPoint );
+ tempV2_3.normalize();
+ const dot = Math.abs( normal1.dot( tempV2_3 ) ); // If path is straight, don't create join
+
+ if ( dot !== 0 ) {
+
+ // Compute inner and outer segment intersections
+ const miterSide = strokeWidth2 / dot;
+ tempV2_3.multiplyScalar( - miterSide );
+ tempV2_4.subVectors( currentPoint, previousPoint );
+ tempV2_5.copy( tempV2_4 ).setLength( miterSide ).add( tempV2_3 );
+ innerPoint.copy( tempV2_5 ).negate();
+ const miterLength2 = tempV2_5.length();
+ const segmentLengthPrev = tempV2_4.length();
+ tempV2_4.divideScalar( segmentLengthPrev );
+ tempV2_6.subVectors( nextPoint, currentPoint );
+ const segmentLengthNext = tempV2_6.length();
+ tempV2_6.divideScalar( segmentLengthNext ); // Check that previous and next segments doesn't overlap with the innerPoint of intersection
+
+ if ( tempV2_4.dot( innerPoint ) < segmentLengthPrev && tempV2_6.dot( innerPoint ) < segmentLengthNext ) {
+
+ innerSideModified = true;
+
+ }
+
+ outerPoint.copy( tempV2_5 ).add( currentPoint );
+ innerPoint.add( currentPoint );
+ isMiter = false;
+
+ if ( innerSideModified ) {
+
+ if ( joinIsOnLeftSide ) {
+
+ nextPointR.copy( innerPoint );
+ currentPointR.copy( innerPoint );
+
+ } else {
+
+ nextPointL.copy( innerPoint );
+ currentPointL.copy( innerPoint );
+
+ }
+
+ } else {
+
+ // The segment triangles are generated here if there was overlapping
+ makeSegmentTriangles();
+
+ }
+
+ switch ( style.strokeLineJoin ) {
+
+ case 'bevel':
+ makeSegmentWithBevelJoin( joinIsOnLeftSide, innerSideModified, u1 );
+ break;
+
+ case 'round':
+ // Segment triangles
+ createSegmentTrianglesWithMiddleSection( joinIsOnLeftSide, innerSideModified ); // Join triangles
+
+ if ( joinIsOnLeftSide ) {
+
+ makeCircularSector( currentPoint, currentPointL, nextPointL, u1, 0 );
+
+ } else {
+
+ makeCircularSector( currentPoint, nextPointR, currentPointR, u1, 1 );
+
+ }
+
+ break;
+
+ case 'miter':
+ case 'miter-clip':
+ default:
+ const miterFraction = strokeWidth2 * style.strokeMiterLimit / miterLength2;
+
+ if ( miterFraction < 1 ) {
+
+ // The join miter length exceeds the miter limit
+ if ( style.strokeLineJoin !== 'miter-clip' ) {
+
+ makeSegmentWithBevelJoin( joinIsOnLeftSide, innerSideModified, u1 );
+ break;
+
+ } else {
+
+ // Segment triangles
+ createSegmentTrianglesWithMiddleSection( joinIsOnLeftSide, innerSideModified ); // Miter-clip join triangles
+
+ if ( joinIsOnLeftSide ) {
+
+ tempV2_6.subVectors( outerPoint, currentPointL ).multiplyScalar( miterFraction ).add( currentPointL );
+ tempV2_7.subVectors( outerPoint, nextPointL ).multiplyScalar( miterFraction ).add( nextPointL );
+ addVertex( currentPointL, u1, 0 );
+ addVertex( tempV2_6, u1, 0 );
+ addVertex( currentPoint, u1, 0.5 );
+ addVertex( currentPoint, u1, 0.5 );
+ addVertex( tempV2_6, u1, 0 );
+ addVertex( tempV2_7, u1, 0 );
+ addVertex( currentPoint, u1, 0.5 );
+ addVertex( tempV2_7, u1, 0 );
+ addVertex( nextPointL, u1, 0 );
+
+ } else {
+
+ tempV2_6.subVectors( outerPoint, currentPointR ).multiplyScalar( miterFraction ).add( currentPointR );
+ tempV2_7.subVectors( outerPoint, nextPointR ).multiplyScalar( miterFraction ).add( nextPointR );
+ addVertex( currentPointR, u1, 1 );
+ addVertex( tempV2_6, u1, 1 );
+ addVertex( currentPoint, u1, 0.5 );
+ addVertex( currentPoint, u1, 0.5 );
+ addVertex( tempV2_6, u1, 1 );
+ addVertex( tempV2_7, u1, 1 );
+ addVertex( currentPoint, u1, 0.5 );
+ addVertex( tempV2_7, u1, 1 );
+ addVertex( nextPointR, u1, 1 );
+
+ }
+
+ }
+
+ } else {
+
+ // Miter join segment triangles
+ if ( innerSideModified ) {
+
+ // Optimized segment + join triangles
+ if ( joinIsOnLeftSide ) {
+
+ addVertex( lastPointR, u0, 1 );
+ addVertex( lastPointL, u0, 0 );
+ addVertex( outerPoint, u1, 0 );
+ addVertex( lastPointR, u0, 1 );
+ addVertex( outerPoint, u1, 0 );
+ addVertex( innerPoint, u1, 1 );
+
+ } else {
+
+ addVertex( lastPointR, u0, 1 );
+ addVertex( lastPointL, u0, 0 );
+ addVertex( outerPoint, u1, 1 );
+ addVertex( lastPointL, u0, 0 );
+ addVertex( innerPoint, u1, 0 );
+ addVertex( outerPoint, u1, 1 );
+
+ }
+
+ if ( joinIsOnLeftSide ) {
+
+ nextPointL.copy( outerPoint );
+
+ } else {
+
+ nextPointR.copy( outerPoint );
+
+ }
+
+ } else {
+
+ // Add extra miter join triangles
+ if ( joinIsOnLeftSide ) {
+
+ addVertex( currentPointL, u1, 0 );
+ addVertex( outerPoint, u1, 0 );
+ addVertex( currentPoint, u1, 0.5 );
+ addVertex( currentPoint, u1, 0.5 );
+ addVertex( outerPoint, u1, 0 );
+ addVertex( nextPointL, u1, 0 );
+
+ } else {
+
+ addVertex( currentPointR, u1, 1 );
+ addVertex( outerPoint, u1, 1 );
+ addVertex( currentPoint, u1, 0.5 );
+ addVertex( currentPoint, u1, 0.5 );
+ addVertex( outerPoint, u1, 1 );
+ addVertex( nextPointR, u1, 1 );
+
+ }
+
+ }
+
+ isMiter = true;
+
+ }
+
+ break;
+
+ }
+
+ } else {
+
+ // The segment triangles are generated here when two consecutive points are collinear
+ makeSegmentTriangles();
+
+ }
+
+ } else {
+
+ // The segment triangles are generated here if it is the ending segment
+ makeSegmentTriangles();
+
+ }
+
+ if ( ! isClosed && iPoint === numPoints - 1 ) {
+
+ // Start line endcap
+ addCapGeometry( points[ 0 ], point0L, point0R, joinIsOnLeftSide, true, u0 );
+
+ } // Increment loop variables
+
+
+ u0 = u1;
+ previousPoint = currentPoint;
+ lastPointL.copy( nextPointL );
+ lastPointR.copy( nextPointR );
+
+ }
+
+ if ( ! isClosed ) {
+
+ // Ending line endcap
+ addCapGeometry( currentPoint, currentPointL, currentPointR, joinIsOnLeftSide, false, u1 );
+
+ } else if ( innerSideModified && vertices ) {
+
+ // Modify path first segment vertices to adjust to the segments inner and outer intersections
+ let lastOuter = outerPoint;
+ let lastInner = innerPoint;
+
+ if ( initialJoinIsOnLeftSide !== joinIsOnLeftSide ) {
+
+ lastOuter = innerPoint;
+ lastInner = outerPoint;
+
+ }
+
+ if ( joinIsOnLeftSide ) {
+
+ if ( isMiter || initialJoinIsOnLeftSide ) {
+
+ lastInner.toArray( vertices, 0 * 3 );
+ lastInner.toArray( vertices, 3 * 3 );
+
+ if ( isMiter ) {
+
+ lastOuter.toArray( vertices, 1 * 3 );
+
+ }
+
+ }
+
+ } else {
+
+ if ( isMiter || ! initialJoinIsOnLeftSide ) {
+
+ lastInner.toArray( vertices, 1 * 3 );
+ lastInner.toArray( vertices, 3 * 3 );
+
+ if ( isMiter ) {
+
+ lastOuter.toArray( vertices, 0 * 3 );
+
+ }
+
+ }
+
+ }
+
+ }
+
+ return numVertices; // -- End of algorithm
+ // -- Functions
+
+ function getNormal( p1, p2, result ) {
+
+ result.subVectors( p2, p1 );
+ return result.set( - result.y, result.x ).normalize();
+
+ }
+
+ function addVertex( position, u, v ) {
+
+ if ( vertices ) {
+
+ vertices[ currentCoordinate ] = position.x;
+ vertices[ currentCoordinate + 1 ] = position.y;
+ vertices[ currentCoordinate + 2 ] = 0;
+
+ if ( normals ) {
+
+ normals[ currentCoordinate ] = 0;
+ normals[ currentCoordinate + 1 ] = 0;
+ normals[ currentCoordinate + 2 ] = 1;
+
+ }
+
+ currentCoordinate += 3;
+
+ if ( uvs ) {
+
+ uvs[ currentCoordinateUV ] = u;
+ uvs[ currentCoordinateUV + 1 ] = v;
+ currentCoordinateUV += 2;
+
+ }
+
+ }
+
+ numVertices += 3;
+
+ }
+
+ function makeCircularSector( center, p1, p2, u, v ) {
+
+ // param p1, p2: Points in the circle arc.
+ // p1 and p2 are in clockwise direction.
+ tempV2_1.copy( p1 ).sub( center ).normalize();
+ tempV2_2.copy( p2 ).sub( center ).normalize();
+ let angle = Math.PI;
+ const dot = tempV2_1.dot( tempV2_2 );
+ if ( Math.abs( dot ) < 1 ) angle = Math.abs( Math.acos( dot ) );
+ angle /= arcDivisions;
+ tempV2_3.copy( p1 );
+
+ for ( let i = 0, il = arcDivisions - 1; i < il; i ++ ) {
+
+ tempV2_4.copy( tempV2_3 ).rotateAround( center, angle );
+ addVertex( tempV2_3, u, v );
+ addVertex( tempV2_4, u, v );
+ addVertex( center, u, 0.5 );
+ tempV2_3.copy( tempV2_4 );
+
+ }
+
+ addVertex( tempV2_4, u, v );
+ addVertex( p2, u, v );
+ addVertex( center, u, 0.5 );
+
+ }
+
+ function makeSegmentTriangles() {
+
+ addVertex( lastPointR, u0, 1 );
+ addVertex( lastPointL, u0, 0 );
+ addVertex( currentPointL, u1, 0 );
+ addVertex( lastPointR, u0, 1 );
+ addVertex( currentPointL, u1, 1 );
+ addVertex( currentPointR, u1, 0 );
+
+ }
+
+ function makeSegmentWithBevelJoin( joinIsOnLeftSide, innerSideModified, u ) {
+
+ if ( innerSideModified ) {
+
+ // Optimized segment + bevel triangles
+ if ( joinIsOnLeftSide ) {
+
+ // THREE.Path segments triangles
+ addVertex( lastPointR, u0, 1 );
+ addVertex( lastPointL, u0, 0 );
+ addVertex( currentPointL, u1, 0 );
+ addVertex( lastPointR, u0, 1 );
+ addVertex( currentPointL, u1, 0 );
+ addVertex( innerPoint, u1, 1 ); // Bevel join triangle
+
+ addVertex( currentPointL, u, 0 );
+ addVertex( nextPointL, u, 0 );
+ addVertex( innerPoint, u, 0.5 );
+
+ } else {
+
+ // THREE.Path segments triangles
+ addVertex( lastPointR, u0, 1 );
+ addVertex( lastPointL, u0, 0 );
+ addVertex( currentPointR, u1, 1 );
+ addVertex( lastPointL, u0, 0 );
+ addVertex( innerPoint, u1, 0 );
+ addVertex( currentPointR, u1, 1 ); // Bevel join triangle
+
+ addVertex( currentPointR, u, 1 );
+ addVertex( nextPointR, u, 0 );
+ addVertex( innerPoint, u, 0.5 );
+
+ }
+
+ } else {
+
+ // Bevel join triangle. The segment triangles are done in the main loop
+ if ( joinIsOnLeftSide ) {
+
+ addVertex( currentPointL, u, 0 );
+ addVertex( nextPointL, u, 0 );
+ addVertex( currentPoint, u, 0.5 );
+
+ } else {
+
+ addVertex( currentPointR, u, 1 );
+ addVertex( nextPointR, u, 0 );
+ addVertex( currentPoint, u, 0.5 );
+
+ }
+
+ }
+
+ }
+
+ function createSegmentTrianglesWithMiddleSection( joinIsOnLeftSide, innerSideModified ) {
+
+ if ( innerSideModified ) {
+
+ if ( joinIsOnLeftSide ) {
+
+ addVertex( lastPointR, u0, 1 );
+ addVertex( lastPointL, u0, 0 );
+ addVertex( currentPointL, u1, 0 );
+ addVertex( lastPointR, u0, 1 );
+ addVertex( currentPointL, u1, 0 );
+ addVertex( innerPoint, u1, 1 );
+ addVertex( currentPointL, u0, 0 );
+ addVertex( currentPoint, u1, 0.5 );
+ addVertex( innerPoint, u1, 1 );
+ addVertex( currentPoint, u1, 0.5 );
+ addVertex( nextPointL, u0, 0 );
+ addVertex( innerPoint, u1, 1 );
+
+ } else {
+
+ addVertex( lastPointR, u0, 1 );
+ addVertex( lastPointL, u0, 0 );
+ addVertex( currentPointR, u1, 1 );
+ addVertex( lastPointL, u0, 0 );
+ addVertex( innerPoint, u1, 0 );
+ addVertex( currentPointR, u1, 1 );
+ addVertex( currentPointR, u0, 1 );
+ addVertex( innerPoint, u1, 0 );
+ addVertex( currentPoint, u1, 0.5 );
+ addVertex( currentPoint, u1, 0.5 );
+ addVertex( innerPoint, u1, 0 );
+ addVertex( nextPointR, u0, 1 );
+
+ }
+
+ }
+
+ }
+
+ function addCapGeometry( center, p1, p2, joinIsOnLeftSide, start, u ) {
+
+ // param center: End point of the path
+ // param p1, p2: Left and right cap points
+ switch ( style.strokeLineCap ) {
+
+ case 'round':
+ if ( start ) {
+
+ makeCircularSector( center, p2, p1, u, 0.5 );
+
+ } else {
+
+ makeCircularSector( center, p1, p2, u, 0.5 );
+
+ }
+
+ break;
+
+ case 'square':
+ if ( start ) {
+
+ tempV2_1.subVectors( p1, center );
+ tempV2_2.set( tempV2_1.y, - tempV2_1.x );
+ tempV2_3.addVectors( tempV2_1, tempV2_2 ).add( center );
+ tempV2_4.subVectors( tempV2_2, tempV2_1 ).add( center ); // Modify already existing vertices
+
+ if ( joinIsOnLeftSide ) {
+
+ tempV2_3.toArray( vertices, 1 * 3 );
+ tempV2_4.toArray( vertices, 0 * 3 );
+ tempV2_4.toArray( vertices, 3 * 3 );
+
+ } else {
+
+ tempV2_3.toArray( vertices, 1 * 3 );
+ tempV2_3.toArray( vertices, 3 * 3 );
+ tempV2_4.toArray( vertices, 0 * 3 );
+
+ }
+
+ } else {
+
+ tempV2_1.subVectors( p2, center );
+ tempV2_2.set( tempV2_1.y, - tempV2_1.x );
+ tempV2_3.addVectors( tempV2_1, tempV2_2 ).add( center );
+ tempV2_4.subVectors( tempV2_2, tempV2_1 ).add( center );
+ const vl = vertices.length; // Modify already existing vertices
+
+ if ( joinIsOnLeftSide ) {
+
+ tempV2_3.toArray( vertices, vl - 1 * 3 );
+ tempV2_4.toArray( vertices, vl - 2 * 3 );
+ tempV2_4.toArray( vertices, vl - 4 * 3 );
+
+ } else {
+
+ tempV2_3.toArray( vertices, vl - 2 * 3 );
+ tempV2_4.toArray( vertices, vl - 1 * 3 );
+ tempV2_4.toArray( vertices, vl - 4 * 3 );
+
+ }
+
+ }
+
+ break;
+
+ case 'butt':
+ default:
+ // Nothing to do here
+ break;
+
+ }
+
+ }
+
+ function removeDuplicatedPoints( points ) {
+
+ // Creates a new array if necessary with duplicated points removed.
+ // This does not remove duplicated initial and ending points of a closed path.
+ let dupPoints = false;
+
+ for ( let i = 1, n = points.length - 1; i < n; i ++ ) {
+
+ if ( points[ i ].distanceTo( points[ i + 1 ] ) < minDistance ) {
+
+ dupPoints = true;
+ break;
+
+ }
+
+ }
+
+ if ( ! dupPoints ) return points;
+ const newPoints = [];
+ newPoints.push( points[ 0 ] );
+
+ for ( let i = 1, n = points.length - 1; i < n; i ++ ) {
+
+ if ( points[ i ].distanceTo( points[ i + 1 ] ) >= minDistance ) {
+
+ newPoints.push( points[ i ] );
+
+ }
+
+ }
+
+ newPoints.push( points[ points.length - 1 ] );
+ return newPoints;
+
+ }
+
+ }
+
+ }
+
+ THREE.SVGLoader = SVGLoader;
+
+} )();
diff --git a/frontend/apps/crawlab/public/three/three.min.js b/frontend/crawlab-ui/public/three/three.min.js
similarity index 100%
rename from frontend/apps/crawlab/public/three/three.min.js
rename to frontend/crawlab-ui/public/three/three.min.js
diff --git a/frontend/crawlab-ui/scripts/gen-dts.js b/frontend/crawlab-ui/scripts/gen-dts.js
new file mode 100644
index 00000000..a6d10255
--- /dev/null
+++ b/frontend/crawlab-ui/scripts/gen-dts.js
@@ -0,0 +1,33 @@
+import fs from 'fs';
+import { spawn } from 'child_process';
+import { resolve, join, dirname } from 'path';
+import { fileURLToPath } from 'url';
+import { log } from './utils.js';
+
+const __dirname = dirname(fileURLToPath(import.meta.url));
+
+function main() {
+ const typingsPath = resolve(join(__dirname, '..', 'typings'));
+ if (fs.existsSync(typingsPath)) {
+ fs.rmSync(typingsPath, { recursive: true });
+ }
+
+ const child = spawn(
+ 'vue-tsc',
+ '--declaration --emitDeclarationOnly --declarationDir ./typings --rootDir ./src'.split(
+ ' '
+ ),
+ { shell: true }
+ );
+ child.stdout.on('data', data => {
+ log(`${data}`, 'info');
+ });
+ child.stderr.on('data', data => {
+ log(`${data}`, 'error');
+ });
+ child.on('close', code => {
+ log(`child process exited with code ${code}`, 'info');
+ });
+}
+
+main();
diff --git a/frontend/crawlab-ui/scripts/gen-index.js b/frontend/crawlab-ui/scripts/gen-index.js
new file mode 100644
index 00000000..5367937f
--- /dev/null
+++ b/frontend/crawlab-ui/scripts/gen-index.js
@@ -0,0 +1,137 @@
+import path from 'path';
+import fs from 'fs';
+import os from 'os';
+
+const EXPORT_MODULES = ['components', 'views', 'directives', 'layouts'];
+
+const COMPONENT_PREFIX = 'Cl';
+const INDEX_COMP_NAME = 'index';
+
+function isWindows() {
+ return os.platform() === 'win32';
+}
+
+function getModulePath(moduleName) {
+ let modulePath = path.resolve(`./src/${moduleName}`);
+ if (isWindows()) {
+ modulePath = modulePath.replace(/\\/g, '/');
+ }
+ return modulePath;
+}
+
+function readFileAndModify(filePath, componentName) {
+ const fileContent = fs.readFileSync(filePath, 'utf8');
+ let newFileContent = '';
+ newFileContent = addComponentName(fileContent, componentName);
+ if (newFileContent !== fileContent) {
+ fs.writeFileSync(filePath, newFileContent);
+ }
+}
+
+function processFile(filePath, moduleName) {
+ const fileName = path.basename(filePath);
+ const relPath = `.${filePath.replace(getModulePath(moduleName), '')}`;
+
+ // skip index.ts
+ if (fileName.split('.')[0] === INDEX_COMP_NAME) {
+ return;
+ }
+
+ if (filePath.endsWith('.vue')) {
+ const compName = fileName.replace('.vue', '');
+ const importLine = `import ${compName} from '${relPath}';`;
+ const exportLine = `${compName} as ${COMPONENT_PREFIX}${compName},`;
+
+ readFileAndModify(filePath, compName);
+ return { importLine, exportLine };
+ } else if (
+ !filePath.endsWith('.d.ts') &&
+ (filePath.endsWith('.ts') || filePath.endsWith('.tsx')) &&
+ fileName !== INDEX_COMP_NAME
+ ) {
+ let compName = fileName.replace(/.tsx?$/, '');
+ compName += compName === 'export' ? '_' : '';
+
+ let importLine;
+ if (compName.startsWith('use')) {
+ importLine = `import ${compName} from '${relPath.replace(/.tsx?$/, '')}';`;
+ } else {
+ importLine = `import * as ${compName} from '${relPath.replace(/.tsx?$/, '')}';`;
+ }
+ const exportLine = `${compName} as ${compName},`;
+ return { importLine, exportLine };
+ }
+}
+
+function addComponentName(content, componentName) {
+ const setupScriptTagRegex = /(
diff --git a/frontend/crawlab-ui/src/admin/baidu.ts b/frontend/crawlab-ui/src/admin/baidu.ts
new file mode 100644
index 00000000..b097991d
--- /dev/null
+++ b/frontend/crawlab-ui/src/admin/baidu.ts
@@ -0,0 +1,12 @@
+// baidu tongji
+export const initBaiduTonji = () => {
+ if (localStorage.getItem('useStats') !== '0') {
+ window._hmt = window._hmt || [];
+ (function () {
+ const hm = document.createElement('script');
+ hm.src = 'https://hm.baidu.com/hm.js?c35e3a563a06caee2524902c81975add';
+ const s = document.getElementsByTagName('script')[0];
+ s?.parentNode?.insertBefore(hm, s);
+ })();
+ }
+};
diff --git a/frontend/crawlab-ui/src/admin/clarity.ts b/frontend/crawlab-ui/src/admin/clarity.ts
new file mode 100644
index 00000000..c0c5324c
--- /dev/null
+++ b/frontend/crawlab-ui/src/admin/clarity.ts
@@ -0,0 +1,6 @@
+// baidu tongji
+export const initClarity = () => {
+ if (localStorage.getItem('useStats') !== '0') {
+ import('@/assets/js/clarity.js');
+ }
+};
diff --git a/frontend/crawlab-ui/src/admin/umeng.ts b/frontend/crawlab-ui/src/admin/umeng.ts
new file mode 100644
index 00000000..e4aeece3
--- /dev/null
+++ b/frontend/crawlab-ui/src/admin/umeng.ts
@@ -0,0 +1,18 @@
+import useRequest from '@/services/request';
+
+const { get } = useRequest();
+export const getEventParamsWrapped = (
+ eventParams?: TrackEventParams
+): TrackEventParamsWrapped => {
+ if (!eventParams) return {};
+ const res: TrackEventParamsWrapped = {};
+ Object.keys(eventParams).forEach(key => {
+ const value = eventParams[key];
+ if (typeof value === 'function') {
+ res[key] = value();
+ } else {
+ res[key] = value;
+ }
+ });
+ return res;
+};
diff --git a/frontend/crawlab-ui/src/assets/favicon.ico b/frontend/crawlab-ui/src/assets/favicon.ico
new file mode 100644
index 00000000..12b5c475
Binary files /dev/null and b/frontend/crawlab-ui/src/assets/favicon.ico differ
diff --git a/frontend/crawlab-ui/src/assets/js/clarity.js b/frontend/crawlab-ui/src/assets/js/clarity.js
new file mode 100644
index 00000000..ae014ea6
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/js/clarity.js
@@ -0,0 +1,12 @@
+(function (c, l, a, r, i, t, y) {
+ c[a] =
+ c[a] ||
+ function () {
+ (c[a].q = c[a].q || []).push(arguments);
+ };
+ t = l.createElement(r);
+ t.async = 1;
+ t.src = 'https://www.clarity.ms/tag/' + i;
+ y = l.getElementsByTagName(r)[0];
+ y.parentNode.insertBefore(t, y);
+})(window, document, 'clarity', 'script', 'htjm67dldb');
diff --git a/frontend/crawlab-ui/src/assets/js/svg/163.js b/frontend/crawlab-ui/src/assets/js/svg/163.js
new file mode 100644
index 00000000..42b084b6
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/js/svg/163.js
@@ -0,0 +1 @@
+module.exports = ``
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/js/svg/aol.js b/frontend/crawlab-ui/src/assets/js/svg/aol.js
new file mode 100644
index 00000000..52911441
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/js/svg/aol.js
@@ -0,0 +1 @@
+module.exports = ``
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/js/svg/arrow-clockwise.js b/frontend/crawlab-ui/src/assets/js/svg/arrow-clockwise.js
new file mode 100644
index 00000000..e8682613
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/js/svg/arrow-clockwise.js
@@ -0,0 +1 @@
+module.exports = ``
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/js/svg/arrow-counterclockwise.js b/frontend/crawlab-ui/src/assets/js/svg/arrow-counterclockwise.js
new file mode 100644
index 00000000..40e163ca
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/js/svg/arrow-counterclockwise.js
@@ -0,0 +1 @@
+module.exports = ``
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/js/svg/cassandra.js b/frontend/crawlab-ui/src/assets/js/svg/cassandra.js
new file mode 100644
index 00000000..fd18e5c4
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/js/svg/cassandra.js
@@ -0,0 +1 @@
+module.exports = ``
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/js/svg/chat-square-quote.js b/frontend/crawlab-ui/src/assets/js/svg/chat-square-quote.js
new file mode 100644
index 00000000..9672cb1e
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/js/svg/chat-square-quote.js
@@ -0,0 +1 @@
+module.exports = ``
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/js/svg/cheerio.js b/frontend/crawlab-ui/src/assets/js/svg/cheerio.js
new file mode 100644
index 00000000..e3a17738
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/js/svg/cheerio.js
@@ -0,0 +1 @@
+module.exports = ``
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/js/svg/chevron-down.js b/frontend/crawlab-ui/src/assets/js/svg/chevron-down.js
new file mode 100644
index 00000000..cdf26337
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/js/svg/chevron-down.js
@@ -0,0 +1 @@
+module.exports = ``
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/js/svg/chromium.js b/frontend/crawlab-ui/src/assets/js/svg/chromium.js
new file mode 100644
index 00000000..c7914350
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/js/svg/chromium.js
@@ -0,0 +1 @@
+module.exports = ``
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/js/svg/clickhouse.js b/frontend/crawlab-ui/src/assets/js/svg/clickhouse.js
new file mode 100644
index 00000000..4c5cec2b
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/js/svg/clickhouse.js
@@ -0,0 +1 @@
+module.exports = ``
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/js/svg/code.js b/frontend/crawlab-ui/src/assets/js/svg/code.js
new file mode 100644
index 00000000..241ee55b
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/js/svg/code.js
@@ -0,0 +1 @@
+module.exports = ``
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/js/svg/colly.js b/frontend/crawlab-ui/src/assets/js/svg/colly.js
new file mode 100644
index 00000000..55f34b2f
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/js/svg/colly.js
@@ -0,0 +1 @@
+module.exports = ``
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/js/svg/crawlee.js b/frontend/crawlab-ui/src/assets/js/svg/crawlee.js
new file mode 100644
index 00000000..3ae89f66
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/js/svg/crawlee.js
@@ -0,0 +1 @@
+module.exports = ``
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/js/svg/db2.js b/frontend/crawlab-ui/src/assets/js/svg/db2.js
new file mode 100644
index 00000000..8b449e5a
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/js/svg/db2.js
@@ -0,0 +1 @@
+module.exports = ``
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/js/svg/devops.js b/frontend/crawlab-ui/src/assets/js/svg/devops.js
new file mode 100644
index 00000000..8fec2c3a
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/js/svg/devops.js
@@ -0,0 +1 @@
+module.exports = ``
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/js/svg/dingtalk.js b/frontend/crawlab-ui/src/assets/js/svg/dingtalk.js
new file mode 100644
index 00000000..9d21de29
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/js/svg/dingtalk.js
@@ -0,0 +1 @@
+module.exports = ``
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/js/svg/drission-page.js b/frontend/crawlab-ui/src/assets/js/svg/drission-page.js
new file mode 100644
index 00000000..e81fc5f9
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/js/svg/drission-page.js
@@ -0,0 +1 @@
+module.exports = ``
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/js/svg/elasticsearch.js b/frontend/crawlab-ui/src/assets/js/svg/elasticsearch.js
new file mode 100644
index 00000000..8dd05fef
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/js/svg/elasticsearch.js
@@ -0,0 +1 @@
+module.exports = ``
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/js/svg/gmail.js b/frontend/crawlab-ui/src/assets/js/svg/gmail.js
new file mode 100644
index 00000000..1a118a4e
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/js/svg/gmail.js
@@ -0,0 +1 @@
+module.exports = ``
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/js/svg/go.js b/frontend/crawlab-ui/src/assets/js/svg/go.js
new file mode 100644
index 00000000..23d289d6
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/js/svg/go.js
@@ -0,0 +1 @@
+module.exports = ``
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/js/svg/hive.js b/frontend/crawlab-ui/src/assets/js/svg/hive.js
new file mode 100644
index 00000000..678fcde6
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/js/svg/hive.js
@@ -0,0 +1 @@
+module.exports = ``
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/js/svg/image-broken.js b/frontend/crawlab-ui/src/assets/js/svg/image-broken.js
new file mode 100644
index 00000000..c1a59669
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/js/svg/image-broken.js
@@ -0,0 +1 @@
+module.exports = ``
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/js/svg/image.js b/frontend/crawlab-ui/src/assets/js/svg/image.js
new file mode 100644
index 00000000..54da2a52
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/js/svg/image.js
@@ -0,0 +1 @@
+module.exports = ``
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/js/svg/journal-code.js b/frontend/crawlab-ui/src/assets/js/svg/journal-code.js
new file mode 100644
index 00000000..20fe0f60
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/js/svg/journal-code.js
@@ -0,0 +1 @@
+module.exports = ``
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/js/svg/journal-text.js b/frontend/crawlab-ui/src/assets/js/svg/journal-text.js
new file mode 100644
index 00000000..c51a94dd
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/js/svg/journal-text.js
@@ -0,0 +1 @@
+module.exports = ``
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/js/svg/justify.js b/frontend/crawlab-ui/src/assets/js/svg/justify.js
new file mode 100644
index 00000000..7bcda738
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/js/svg/justify.js
@@ -0,0 +1 @@
+module.exports = ``
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/js/svg/kafka.js b/frontend/crawlab-ui/src/assets/js/svg/kafka.js
new file mode 100644
index 00000000..975507c2
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/js/svg/kafka.js
@@ -0,0 +1 @@
+module.exports = ``
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/js/svg/lark.js b/frontend/crawlab-ui/src/assets/js/svg/lark.js
new file mode 100644
index 00000000..ac8d3295
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/js/svg/lark.js
@@ -0,0 +1 @@
+module.exports = ``
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/js/svg/link.js b/frontend/crawlab-ui/src/assets/js/svg/link.js
new file mode 100644
index 00000000..1a259416
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/js/svg/link.js
@@ -0,0 +1 @@
+module.exports = ``
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/js/svg/list-ol.js b/frontend/crawlab-ui/src/assets/js/svg/list-ol.js
new file mode 100644
index 00000000..d8576034
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/js/svg/list-ol.js
@@ -0,0 +1 @@
+module.exports = ``
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/js/svg/list-ul.js b/frontend/crawlab-ui/src/assets/js/svg/list-ul.js
new file mode 100644
index 00000000..049bfe54
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/js/svg/list-ul.js
@@ -0,0 +1 @@
+module.exports = ``
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/js/svg/logo-icon-white.js b/frontend/crawlab-ui/src/assets/js/svg/logo-icon-white.js
new file mode 100644
index 00000000..d462f7de
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/js/svg/logo-icon-white.js
@@ -0,0 +1 @@
+module.exports = ``
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/js/svg/logo-icon.js b/frontend/crawlab-ui/src/assets/js/svg/logo-icon.js
new file mode 100644
index 00000000..3a8e2cb5
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/js/svg/logo-icon.js
@@ -0,0 +1 @@
+module.exports = ``
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/js/svg/logo-main.js b/frontend/crawlab-ui/src/assets/js/svg/logo-main.js
new file mode 100644
index 00000000..ec95b766
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/js/svg/logo-main.js
@@ -0,0 +1 @@
+module.exports = ``
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/js/svg/logo-white.js b/frontend/crawlab-ui/src/assets/js/svg/logo-white.js
new file mode 100644
index 00000000..3abddd30
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/js/svg/logo-white.js
@@ -0,0 +1 @@
+module.exports = ``
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/js/svg/logo.js b/frontend/crawlab-ui/src/assets/js/svg/logo.js
new file mode 100644
index 00000000..33bffbda
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/js/svg/logo.js
@@ -0,0 +1 @@
+module.exports = ``
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/js/svg/maven.js b/frontend/crawlab-ui/src/assets/js/svg/maven.js
new file mode 100644
index 00000000..581d39e2
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/js/svg/maven.js
@@ -0,0 +1 @@
+module.exports = ``
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/js/svg/mongodb.js b/frontend/crawlab-ui/src/assets/js/svg/mongodb.js
new file mode 100644
index 00000000..c3ac619b
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/js/svg/mongodb.js
@@ -0,0 +1 @@
+module.exports = ``
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/js/svg/ms_teams.js b/frontend/crawlab-ui/src/assets/js/svg/ms_teams.js
new file mode 100644
index 00000000..6b203990
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/js/svg/ms_teams.js
@@ -0,0 +1 @@
+module.exports = ``
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/js/svg/mssql.js b/frontend/crawlab-ui/src/assets/js/svg/mssql.js
new file mode 100644
index 00000000..cb1dc46f
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/js/svg/mssql.js
@@ -0,0 +1 @@
+module.exports = ``
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/js/svg/mysql.js b/frontend/crawlab-ui/src/assets/js/svg/mysql.js
new file mode 100644
index 00000000..83eaa5b6
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/js/svg/mysql.js
@@ -0,0 +1 @@
+module.exports = ``
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/js/svg/netease.js b/frontend/crawlab-ui/src/assets/js/svg/netease.js
new file mode 100644
index 00000000..76279464
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/js/svg/netease.js
@@ -0,0 +1 @@
+module.exports = ``
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/js/svg/oracle.js b/frontend/crawlab-ui/src/assets/js/svg/oracle.js
new file mode 100644
index 00000000..91d340ed
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/js/svg/oracle.js
@@ -0,0 +1 @@
+module.exports = ``
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/js/svg/outlook.js b/frontend/crawlab-ui/src/assets/js/svg/outlook.js
new file mode 100644
index 00000000..0543f60a
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/js/svg/outlook.js
@@ -0,0 +1 @@
+module.exports = ``
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/js/svg/pencil-fill.js b/frontend/crawlab-ui/src/assets/js/svg/pencil-fill.js
new file mode 100644
index 00000000..29ba3ee7
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/js/svg/pencil-fill.js
@@ -0,0 +1 @@
+module.exports = ``
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/js/svg/playwright.js b/frontend/crawlab-ui/src/assets/js/svg/playwright.js
new file mode 100644
index 00000000..3f1bae3f
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/js/svg/playwright.js
@@ -0,0 +1 @@
+module.exports = ``
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/js/svg/plus.js b/frontend/crawlab-ui/src/assets/js/svg/plus.js
new file mode 100644
index 00000000..4b40c702
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/js/svg/plus.js
@@ -0,0 +1 @@
+module.exports = ``
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/js/svg/postgres.js b/frontend/crawlab-ui/src/assets/js/svg/postgres.js
new file mode 100644
index 00000000..cb42ac04
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/js/svg/postgres.js
@@ -0,0 +1 @@
+module.exports = ``
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/js/svg/proton.js b/frontend/crawlab-ui/src/assets/js/svg/proton.js
new file mode 100644
index 00000000..0570d764
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/js/svg/proton.js
@@ -0,0 +1 @@
+module.exports = ``
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/js/svg/puppeteer.js b/frontend/crawlab-ui/src/assets/js/svg/puppeteer.js
new file mode 100644
index 00000000..4bc06544
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/js/svg/puppeteer.js
@@ -0,0 +1 @@
+module.exports = ``
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/js/svg/redis.js b/frontend/crawlab-ui/src/assets/js/svg/redis.js
new file mode 100644
index 00000000..ff2dc637
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/js/svg/redis.js
@@ -0,0 +1 @@
+module.exports = ``
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/js/svg/scrapy.js b/frontend/crawlab-ui/src/assets/js/svg/scrapy.js
new file mode 100644
index 00000000..58ccb03d
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/js/svg/scrapy.js
@@ -0,0 +1 @@
+module.exports = ``
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/js/svg/selenium.js b/frontend/crawlab-ui/src/assets/js/svg/selenium.js
new file mode 100644
index 00000000..41e807c8
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/js/svg/selenium.js
@@ -0,0 +1 @@
+module.exports = ``
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/js/svg/snowflake.js b/frontend/crawlab-ui/src/assets/js/svg/snowflake.js
new file mode 100644
index 00000000..19789b14
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/js/svg/snowflake.js
@@ -0,0 +1 @@
+module.exports = ``
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/js/svg/table.js b/frontend/crawlab-ui/src/assets/js/svg/table.js
new file mode 100644
index 00000000..6f369d6f
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/js/svg/table.js
@@ -0,0 +1 @@
+module.exports = ``
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/js/svg/text-center.js b/frontend/crawlab-ui/src/assets/js/svg/text-center.js
new file mode 100644
index 00000000..7ea8b4dc
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/js/svg/text-center.js
@@ -0,0 +1 @@
+module.exports = ``
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/js/svg/text-left.js b/frontend/crawlab-ui/src/assets/js/svg/text-left.js
new file mode 100644
index 00000000..dafd817c
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/js/svg/text-left.js
@@ -0,0 +1 @@
+module.exports = ``
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/js/svg/text-paragraph.js b/frontend/crawlab-ui/src/assets/js/svg/text-paragraph.js
new file mode 100644
index 00000000..948e3149
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/js/svg/text-paragraph.js
@@ -0,0 +1 @@
+module.exports = ``
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/js/svg/text-right.js b/frontend/crawlab-ui/src/assets/js/svg/text-right.js
new file mode 100644
index 00000000..5fccaf33
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/js/svg/text-right.js
@@ -0,0 +1 @@
+module.exports = ``
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/js/svg/type-bold.js b/frontend/crawlab-ui/src/assets/js/svg/type-bold.js
new file mode 100644
index 00000000..59635c7d
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/js/svg/type-bold.js
@@ -0,0 +1 @@
+module.exports = ``
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/js/svg/type-h1.js b/frontend/crawlab-ui/src/assets/js/svg/type-h1.js
new file mode 100644
index 00000000..e147764f
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/js/svg/type-h1.js
@@ -0,0 +1 @@
+module.exports = ``
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/js/svg/type-h2.js b/frontend/crawlab-ui/src/assets/js/svg/type-h2.js
new file mode 100644
index 00000000..e712c2a0
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/js/svg/type-h2.js
@@ -0,0 +1 @@
+module.exports = ``
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/js/svg/type-h3.js b/frontend/crawlab-ui/src/assets/js/svg/type-h3.js
new file mode 100644
index 00000000..18b72e5d
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/js/svg/type-h3.js
@@ -0,0 +1 @@
+module.exports = ``
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/js/svg/type-italic.js b/frontend/crawlab-ui/src/assets/js/svg/type-italic.js
new file mode 100644
index 00000000..84655a30
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/js/svg/type-italic.js
@@ -0,0 +1 @@
+module.exports = ``
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/js/svg/type-strikethrough.js b/frontend/crawlab-ui/src/assets/js/svg/type-strikethrough.js
new file mode 100644
index 00000000..1ea1244e
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/js/svg/type-strikethrough.js
@@ -0,0 +1 @@
+module.exports = ``
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/js/svg/type-underline.js b/frontend/crawlab-ui/src/assets/js/svg/type-underline.js
new file mode 100644
index 00000000..17463475
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/js/svg/type-underline.js
@@ -0,0 +1 @@
+module.exports = ``
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/js/svg/webmagic.js b/frontend/crawlab-ui/src/assets/js/svg/webmagic.js
new file mode 100644
index 00000000..88c04a08
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/js/svg/webmagic.js
@@ -0,0 +1 @@
+module.exports = ``
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/js/svg/wechat_work.js b/frontend/crawlab-ui/src/assets/js/svg/wechat_work.js
new file mode 100644
index 00000000..ff1d7702
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/js/svg/wechat_work.js
@@ -0,0 +1 @@
+module.exports = ``
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/js/svg/xxl.js b/frontend/crawlab-ui/src/assets/js/svg/xxl.js
new file mode 100644
index 00000000..2325f6a4
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/js/svg/xxl.js
@@ -0,0 +1 @@
+module.exports = ``
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/js/svg/zoho.js b/frontend/crawlab-ui/src/assets/js/svg/zoho.js
new file mode 100644
index 00000000..b512965a
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/js/svg/zoho.js
@@ -0,0 +1 @@
+module.exports = ``
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/lexical/images/icons/LICENSE.md b/frontend/crawlab-ui/src/assets/lexical/images/icons/LICENSE.md
new file mode 100644
index 00000000..ce74f6ab
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/lexical/images/icons/LICENSE.md
@@ -0,0 +1,5 @@
+Bootstrap Icons
+https://icons.getbootstrap.com
+
+Licensed under MIT license
+https://github.com/twbs/icons/blob/main/LICENSE.md
diff --git a/frontend/crawlab-ui/src/assets/lexical/images/icons/arrow-clockwise.svg b/frontend/crawlab-ui/src/assets/lexical/images/icons/arrow-clockwise.svg
new file mode 100644
index 00000000..b072eb09
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/lexical/images/icons/arrow-clockwise.svg
@@ -0,0 +1,4 @@
+
+
+
+
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/lexical/images/icons/arrow-counterclockwise.svg b/frontend/crawlab-ui/src/assets/lexical/images/icons/arrow-counterclockwise.svg
new file mode 100644
index 00000000..b0b23b9b
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/lexical/images/icons/arrow-counterclockwise.svg
@@ -0,0 +1,4 @@
+
+
+
+
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/lexical/images/icons/chat-square-quote.svg b/frontend/crawlab-ui/src/assets/lexical/images/icons/chat-square-quote.svg
new file mode 100644
index 00000000..40893f49
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/lexical/images/icons/chat-square-quote.svg
@@ -0,0 +1,4 @@
+
+
+
+
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/lexical/images/icons/chevron-down.svg b/frontend/crawlab-ui/src/assets/lexical/images/icons/chevron-down.svg
new file mode 100644
index 00000000..1f0b8bc7
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/lexical/images/icons/chevron-down.svg
@@ -0,0 +1,3 @@
+
+
+
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/lexical/images/icons/code.svg b/frontend/crawlab-ui/src/assets/lexical/images/icons/code.svg
new file mode 100644
index 00000000..079f5c67
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/lexical/images/icons/code.svg
@@ -0,0 +1,3 @@
+
+
+
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/lexical/images/icons/image.svg b/frontend/crawlab-ui/src/assets/lexical/images/icons/image.svg
new file mode 100644
index 00000000..579f7b8b
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/lexical/images/icons/image.svg
@@ -0,0 +1 @@
+
diff --git a/frontend/crawlab-ui/src/assets/lexical/images/icons/journal-code.svg b/frontend/crawlab-ui/src/assets/lexical/images/icons/journal-code.svg
new file mode 100644
index 00000000..82098b9c
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/lexical/images/icons/journal-code.svg
@@ -0,0 +1,5 @@
+
+
+
+
+
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/lexical/images/icons/journal-text.svg b/frontend/crawlab-ui/src/assets/lexical/images/icons/journal-text.svg
new file mode 100644
index 00000000..9b66f43a
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/lexical/images/icons/journal-text.svg
@@ -0,0 +1,5 @@
+
+
+
+
+
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/lexical/images/icons/justify.svg b/frontend/crawlab-ui/src/assets/lexical/images/icons/justify.svg
new file mode 100644
index 00000000..009bd721
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/lexical/images/icons/justify.svg
@@ -0,0 +1,3 @@
+
+
+
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/lexical/images/icons/link.svg b/frontend/crawlab-ui/src/assets/lexical/images/icons/link.svg
new file mode 100644
index 00000000..df35bc8a
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/lexical/images/icons/link.svg
@@ -0,0 +1,4 @@
+
+
+
+
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/lexical/images/icons/list-ol.svg b/frontend/crawlab-ui/src/assets/lexical/images/icons/list-ol.svg
new file mode 100644
index 00000000..5782568d
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/lexical/images/icons/list-ol.svg
@@ -0,0 +1,4 @@
+
+
+
+
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/lexical/images/icons/list-ul.svg b/frontend/crawlab-ui/src/assets/lexical/images/icons/list-ul.svg
new file mode 100644
index 00000000..217d1539
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/lexical/images/icons/list-ul.svg
@@ -0,0 +1,3 @@
+
+
+
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/lexical/images/icons/pencil-fill.svg b/frontend/crawlab-ui/src/assets/lexical/images/icons/pencil-fill.svg
new file mode 100644
index 00000000..59d2830c
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/lexical/images/icons/pencil-fill.svg
@@ -0,0 +1,3 @@
+
+
+
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/lexical/images/icons/plus.svg b/frontend/crawlab-ui/src/assets/lexical/images/icons/plus.svg
new file mode 100644
index 00000000..eb15d6ba
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/lexical/images/icons/plus.svg
@@ -0,0 +1 @@
+
diff --git a/frontend/crawlab-ui/src/assets/lexical/images/icons/table.svg b/frontend/crawlab-ui/src/assets/lexical/images/icons/table.svg
new file mode 100644
index 00000000..667ede67
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/lexical/images/icons/table.svg
@@ -0,0 +1 @@
+
diff --git a/frontend/crawlab-ui/src/assets/lexical/images/icons/text-center.svg b/frontend/crawlab-ui/src/assets/lexical/images/icons/text-center.svg
new file mode 100644
index 00000000..2887a99f
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/lexical/images/icons/text-center.svg
@@ -0,0 +1,3 @@
+
+
+
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/lexical/images/icons/text-left.svg b/frontend/crawlab-ui/src/assets/lexical/images/icons/text-left.svg
new file mode 100644
index 00000000..04526116
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/lexical/images/icons/text-left.svg
@@ -0,0 +1,3 @@
+
+
+
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/lexical/images/icons/text-paragraph.svg b/frontend/crawlab-ui/src/assets/lexical/images/icons/text-paragraph.svg
new file mode 100644
index 00000000..9779beab
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/lexical/images/icons/text-paragraph.svg
@@ -0,0 +1,3 @@
+
+
+
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/lexical/images/icons/text-right.svg b/frontend/crawlab-ui/src/assets/lexical/images/icons/text-right.svg
new file mode 100644
index 00000000..34686b0f
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/lexical/images/icons/text-right.svg
@@ -0,0 +1,3 @@
+
+
+
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/lexical/images/icons/type-bold.svg b/frontend/crawlab-ui/src/assets/lexical/images/icons/type-bold.svg
new file mode 100644
index 00000000..276d133c
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/lexical/images/icons/type-bold.svg
@@ -0,0 +1,3 @@
+
+
+
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/lexical/images/icons/type-h1.svg b/frontend/crawlab-ui/src/assets/lexical/images/icons/type-h1.svg
new file mode 100644
index 00000000..4c891817
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/lexical/images/icons/type-h1.svg
@@ -0,0 +1,3 @@
+
+
+
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/lexical/images/icons/type-h2.svg b/frontend/crawlab-ui/src/assets/lexical/images/icons/type-h2.svg
new file mode 100644
index 00000000..b6ab7650
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/lexical/images/icons/type-h2.svg
@@ -0,0 +1,3 @@
+
+
+
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/lexical/images/icons/type-h3.svg b/frontend/crawlab-ui/src/assets/lexical/images/icons/type-h3.svg
new file mode 100644
index 00000000..154c293f
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/lexical/images/icons/type-h3.svg
@@ -0,0 +1,3 @@
+
+
+
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/lexical/images/icons/type-italic.svg b/frontend/crawlab-ui/src/assets/lexical/images/icons/type-italic.svg
new file mode 100644
index 00000000..3ac6b09f
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/lexical/images/icons/type-italic.svg
@@ -0,0 +1,3 @@
+
+
+
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/lexical/images/icons/type-strikethrough.svg b/frontend/crawlab-ui/src/assets/lexical/images/icons/type-strikethrough.svg
new file mode 100644
index 00000000..1c940e42
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/lexical/images/icons/type-strikethrough.svg
@@ -0,0 +1,3 @@
+
+
+
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/lexical/images/icons/type-underline.svg b/frontend/crawlab-ui/src/assets/lexical/images/icons/type-underline.svg
new file mode 100644
index 00000000..c299b8bf
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/lexical/images/icons/type-underline.svg
@@ -0,0 +1,3 @@
+
+
+
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/lexical/images/image-broken.svg b/frontend/crawlab-ui/src/assets/lexical/images/image-broken.svg
new file mode 100644
index 00000000..bb4c600a
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/lexical/images/image-broken.svg
@@ -0,0 +1,3 @@
+
+
+
diff --git a/frontend/crawlab-ui/src/assets/logo.svg b/frontend/crawlab-ui/src/assets/logo.svg
new file mode 100644
index 00000000..c52872b4
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/logo.svg
@@ -0,0 +1,16 @@
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/assets/svg/icons/163.svg b/frontend/crawlab-ui/src/assets/svg/icons/163.svg
new file mode 100644
index 00000000..75a0fcbf
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/svg/icons/163.svg
@@ -0,0 +1 @@
+
diff --git a/frontend/crawlab-ui/src/assets/svg/icons/anthropic.svg b/frontend/crawlab-ui/src/assets/svg/icons/anthropic.svg
new file mode 100644
index 00000000..5b81844c
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/svg/icons/anthropic.svg
@@ -0,0 +1 @@
+Anthropic
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/svg/icons/aol.svg b/frontend/crawlab-ui/src/assets/svg/icons/aol.svg
new file mode 100644
index 00000000..eff387e8
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/svg/icons/aol.svg
@@ -0,0 +1,57 @@
+
+
+
+
+
+ image/svg+xml
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/assets/svg/icons/azure.svg b/frontend/crawlab-ui/src/assets/svg/icons/azure.svg
new file mode 100644
index 00000000..ed50209c
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/svg/icons/azure.svg
@@ -0,0 +1 @@
+Azure
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/svg/icons/azure/devops.svg b/frontend/crawlab-ui/src/assets/svg/icons/azure/devops.svg
new file mode 100644
index 00000000..3d4a462f
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/svg/icons/azure/devops.svg
@@ -0,0 +1 @@
+Icon-devops-261
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/svg/icons/cassandra.svg b/frontend/crawlab-ui/src/assets/svg/icons/cassandra.svg
new file mode 100644
index 00000000..743f4709
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/svg/icons/cassandra.svg
@@ -0,0 +1 @@
+
diff --git a/frontend/crawlab-ui/src/assets/svg/icons/cheerio.svg b/frontend/crawlab-ui/src/assets/svg/icons/cheerio.svg
new file mode 100644
index 00000000..72212d66
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/svg/icons/cheerio.svg
@@ -0,0 +1,4 @@
+
+
+
+
diff --git a/frontend/crawlab-ui/src/assets/svg/icons/chromium.svg b/frontend/crawlab-ui/src/assets/svg/icons/chromium.svg
new file mode 100644
index 00000000..5eb9f6ad
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/svg/icons/chromium.svg
@@ -0,0 +1,120 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/assets/svg/icons/clickhouse.svg b/frontend/crawlab-ui/src/assets/svg/icons/clickhouse.svg
new file mode 100644
index 00000000..f2144b5d
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/svg/icons/clickhouse.svg
@@ -0,0 +1 @@
+
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/svg/icons/colly.svg b/frontend/crawlab-ui/src/assets/svg/icons/colly.svg
new file mode 100644
index 00000000..f89aea5f
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/svg/icons/colly.svg
@@ -0,0 +1,12 @@
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/assets/svg/icons/crawlee.svg b/frontend/crawlab-ui/src/assets/svg/icons/crawlee.svg
new file mode 100644
index 00000000..d34fe6a1
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/svg/icons/crawlee.svg
@@ -0,0 +1,65 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/assets/svg/icons/db2.svg b/frontend/crawlab-ui/src/assets/svg/icons/db2.svg
new file mode 100644
index 00000000..41c158bf
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/svg/icons/db2.svg
@@ -0,0 +1 @@
+
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/svg/icons/deepseek.svg b/frontend/crawlab-ui/src/assets/svg/icons/deepseek.svg
new file mode 100644
index 00000000..3fc23024
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/svg/icons/deepseek.svg
@@ -0,0 +1 @@
+DeepSeek
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/svg/icons/dingtalk.svg b/frontend/crawlab-ui/src/assets/svg/icons/dingtalk.svg
new file mode 100644
index 00000000..d14c9273
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/svg/icons/dingtalk.svg
@@ -0,0 +1,5 @@
+
+
+
+
+
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/svg/icons/drission-page.svg b/frontend/crawlab-ui/src/assets/svg/icons/drission-page.svg
new file mode 100644
index 00000000..6ca7090e
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/svg/icons/drission-page.svg
@@ -0,0 +1,31 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/assets/svg/icons/elasticsearch.svg b/frontend/crawlab-ui/src/assets/svg/icons/elasticsearch.svg
new file mode 100644
index 00000000..9fc689ef
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/svg/icons/elasticsearch.svg
@@ -0,0 +1,12 @@
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/assets/svg/icons/gemini.svg b/frontend/crawlab-ui/src/assets/svg/icons/gemini.svg
new file mode 100644
index 00000000..878eb627
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/svg/icons/gemini.svg
@@ -0,0 +1 @@
+Gemini
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/svg/icons/gmail.svg b/frontend/crawlab-ui/src/assets/svg/icons/gmail.svg
new file mode 100644
index 00000000..40b7175c
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/svg/icons/gmail.svg
@@ -0,0 +1,7 @@
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/svg/icons/go.svg b/frontend/crawlab-ui/src/assets/svg/icons/go.svg
new file mode 100644
index 00000000..c60a234a
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/svg/icons/go.svg
@@ -0,0 +1,9 @@
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/assets/svg/icons/grok.svg b/frontend/crawlab-ui/src/assets/svg/icons/grok.svg
new file mode 100644
index 00000000..efb1a618
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/svg/icons/grok.svg
@@ -0,0 +1 @@
+Grok
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/svg/icons/hive.svg b/frontend/crawlab-ui/src/assets/svg/icons/hive.svg
new file mode 100644
index 00000000..e68b9094
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/svg/icons/hive.svg
@@ -0,0 +1 @@
+
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/svg/icons/kafka.svg b/frontend/crawlab-ui/src/assets/svg/icons/kafka.svg
new file mode 100644
index 00000000..403ed158
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/svg/icons/kafka.svg
@@ -0,0 +1,6 @@
+
+
+
diff --git a/frontend/crawlab-ui/src/assets/svg/icons/lark.svg b/frontend/crawlab-ui/src/assets/svg/icons/lark.svg
new file mode 100644
index 00000000..b07867c7
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/svg/icons/lark.svg
@@ -0,0 +1,6 @@
+
+
+
+
+
+
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/svg/icons/maven.svg b/frontend/crawlab-ui/src/assets/svg/icons/maven.svg
new file mode 100644
index 00000000..6a5e1b02
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/svg/icons/maven.svg
@@ -0,0 +1,2 @@
+
+file_type_maven
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/svg/icons/mistral.svg b/frontend/crawlab-ui/src/assets/svg/icons/mistral.svg
new file mode 100644
index 00000000..8e03e244
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/svg/icons/mistral.svg
@@ -0,0 +1 @@
+Mistral
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/svg/icons/mongodb.svg b/frontend/crawlab-ui/src/assets/svg/icons/mongodb.svg
new file mode 100644
index 00000000..8ca38b61
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/svg/icons/mongodb.svg
@@ -0,0 +1,6 @@
+
+
+
diff --git a/frontend/crawlab-ui/src/assets/svg/icons/ms_teams.svg b/frontend/crawlab-ui/src/assets/svg/icons/ms_teams.svg
new file mode 100644
index 00000000..3409e6cf
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/svg/icons/ms_teams.svg
@@ -0,0 +1,22 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/assets/svg/icons/mssql.svg b/frontend/crawlab-ui/src/assets/svg/icons/mssql.svg
new file mode 100644
index 00000000..7a137dfb
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/svg/icons/mssql.svg
@@ -0,0 +1,12 @@
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/assets/svg/icons/mysql.svg b/frontend/crawlab-ui/src/assets/svg/icons/mysql.svg
new file mode 100644
index 00000000..44fd4dbf
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/svg/icons/mysql.svg
@@ -0,0 +1,6 @@
+
+
+
diff --git a/frontend/crawlab-ui/src/assets/svg/icons/netease.svg b/frontend/crawlab-ui/src/assets/svg/icons/netease.svg
new file mode 100644
index 00000000..fcdea27f
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/svg/icons/netease.svg
@@ -0,0 +1,2 @@
+
+
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/svg/icons/openai.svg b/frontend/crawlab-ui/src/assets/svg/icons/openai.svg
new file mode 100644
index 00000000..50d94d6c
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/svg/icons/openai.svg
@@ -0,0 +1 @@
+OpenAI
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/svg/icons/oracle.svg b/frontend/crawlab-ui/src/assets/svg/icons/oracle.svg
new file mode 100644
index 00000000..3f4e051f
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/svg/icons/oracle.svg
@@ -0,0 +1,4 @@
+
+
+
+
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/svg/icons/outlook.svg b/frontend/crawlab-ui/src/assets/svg/icons/outlook.svg
new file mode 100644
index 00000000..2cde22d4
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/svg/icons/outlook.svg
@@ -0,0 +1,35 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/svg/icons/playwright.svg b/frontend/crawlab-ui/src/assets/svg/icons/playwright.svg
new file mode 100644
index 00000000..3dcb4ac0
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/svg/icons/playwright.svg
@@ -0,0 +1 @@
+Playwright Streamline Icon: https://streamlinehq.com
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/svg/icons/postgres.svg b/frontend/crawlab-ui/src/assets/svg/icons/postgres.svg
new file mode 100644
index 00000000..8bed16d4
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/svg/icons/postgres.svg
@@ -0,0 +1,18 @@
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/assets/svg/icons/proton.svg b/frontend/crawlab-ui/src/assets/svg/icons/proton.svg
new file mode 100644
index 00000000..dd39b390
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/svg/icons/proton.svg
@@ -0,0 +1 @@
+
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/svg/icons/puppeteer.svg b/frontend/crawlab-ui/src/assets/svg/icons/puppeteer.svg
new file mode 100644
index 00000000..dd734b18
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/svg/icons/puppeteer.svg
@@ -0,0 +1,18 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/svg/icons/qwen.svg b/frontend/crawlab-ui/src/assets/svg/icons/qwen.svg
new file mode 100644
index 00000000..e1199f82
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/svg/icons/qwen.svg
@@ -0,0 +1 @@
+Qwen
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/svg/icons/redis.svg b/frontend/crawlab-ui/src/assets/svg/icons/redis.svg
new file mode 100644
index 00000000..6b8d45ae
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/svg/icons/redis.svg
@@ -0,0 +1,6 @@
+
+
+
diff --git a/frontend/crawlab-ui/src/assets/svg/icons/scrapy.svg b/frontend/crawlab-ui/src/assets/svg/icons/scrapy.svg
new file mode 100644
index 00000000..224fee19
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/svg/icons/scrapy.svg
@@ -0,0 +1,13 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/assets/svg/icons/selenium.svg b/frontend/crawlab-ui/src/assets/svg/icons/selenium.svg
new file mode 100644
index 00000000..e9d6043d
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/svg/icons/selenium.svg
@@ -0,0 +1,8 @@
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/svg/icons/snowflake.svg b/frontend/crawlab-ui/src/assets/svg/icons/snowflake.svg
new file mode 100644
index 00000000..f491c273
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/svg/icons/snowflake.svg
@@ -0,0 +1 @@
+
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/svg/icons/webmagic.svg b/frontend/crawlab-ui/src/assets/svg/icons/webmagic.svg
new file mode 100644
index 00000000..2b286d1a
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/svg/icons/webmagic.svg
@@ -0,0 +1,37 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/assets/svg/icons/wechat_work.svg b/frontend/crawlab-ui/src/assets/svg/icons/wechat_work.svg
new file mode 100644
index 00000000..714dd1a6
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/svg/icons/wechat_work.svg
@@ -0,0 +1 @@
+
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/svg/icons/xxl.svg b/frontend/crawlab-ui/src/assets/svg/icons/xxl.svg
new file mode 100644
index 00000000..96728a5c
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/svg/icons/xxl.svg
@@ -0,0 +1,107 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/assets/svg/icons/zoho.svg b/frontend/crawlab-ui/src/assets/svg/icons/zoho.svg
new file mode 100644
index 00000000..d5c9087a
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/svg/icons/zoho.svg
@@ -0,0 +1 @@
+
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/assets/svg/logo-icon-white.svg b/frontend/crawlab-ui/src/assets/svg/logo-icon-white.svg
new file mode 100644
index 00000000..997c9db6
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/svg/logo-icon-white.svg
@@ -0,0 +1,107 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/assets/svg/logo-icon.svg b/frontend/crawlab-ui/src/assets/svg/logo-icon.svg
new file mode 100644
index 00000000..e095990a
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/svg/logo-icon.svg
@@ -0,0 +1,107 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/assets/svg/logo-main.svg b/frontend/crawlab-ui/src/assets/svg/logo-main.svg
new file mode 100644
index 00000000..aed3ae3a
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/svg/logo-main.svg
@@ -0,0 +1,161 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/assets/svg/logo-white.svg b/frontend/crawlab-ui/src/assets/svg/logo-white.svg
new file mode 100644
index 00000000..afc7439a
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/svg/logo-white.svg
@@ -0,0 +1,161 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/assets/svg/logo.ts b/frontend/crawlab-ui/src/assets/svg/logo.ts
new file mode 100644
index 00000000..c57b7d0e
--- /dev/null
+++ b/frontend/crawlab-ui/src/assets/svg/logo.ts
@@ -0,0 +1 @@
+export default ``;
diff --git a/frontend/crawlab-ui/src/components/core/ai/AssistantConsole.vue b/frontend/crawlab-ui/src/components/core/ai/AssistantConsole.vue
new file mode 100644
index 00000000..a5970ff8
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/ai/AssistantConsole.vue
@@ -0,0 +1,342 @@
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/ai/useAssistantConsole.ts b/frontend/crawlab-ui/src/components/core/ai/useAssistantConsole.ts
new file mode 100644
index 00000000..b7898462
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/ai/useAssistantConsole.ts
@@ -0,0 +1,456 @@
+import { computed, ref, watch, reactive, onMounted } from 'vue';
+import { useI18n } from 'vue-i18n';
+import useRequest from '@/services/request';
+import { getRequestBaseUrl } from '@/utils';
+import { debounce } from 'lodash';
+import { ElMessage } from 'element-plus';
+import { AxiosError } from 'axios';
+import { ClChatInput } from '@/components';
+
+const useAssistantConsole = () => {
+ const { t } = useI18n();
+ const { get } = useRequest();
+
+ // Refs
+ const messageListRef = ref<{ scrollToBottom: () => Promise } | null>(
+ null
+ );
+ const chatInputRef = ref | null>(null);
+
+ // State management
+ const currentConversation = ref(null);
+ const currentConversationId = ref('');
+ const conversations = ref([]);
+ const chatHistory = reactive([]);
+ const isGenerating = ref(false);
+ const streamError = ref('');
+ const isLoadingConversations = ref(false);
+ const isLoadingMessages = ref(false);
+ const historyDialogVisible = ref(false);
+ const configDialogVisible = ref(false);
+ const abortController = ref(null);
+ const availableProviders = ref([]);
+
+ // Computed properties
+ const currentConversationTitle = computed(() => {
+ if (!currentConversationId.value) return t('components.ai.chatbot.newChat');
+ return (
+ currentConversation.value?.title || t('components.ai.chatbot.newChat')
+ );
+ });
+
+ const chatbotConfig = ref({
+ systemPrompt:
+ 'You are a helpful AI assistant for Crawlab, a web crawling and data extraction platform.',
+ temperature: 0.7,
+ maxTokens: 1000,
+ });
+
+ // Load conversations
+ const loadConversations = async () => {
+ isLoadingConversations.value = true;
+ try {
+ const res = await get('/ai/chat/conversations', {
+ page: 1,
+ size: 500,
+ sort: '-last_message_at',
+ });
+ conversations.value = res.data || [];
+ } catch (error) {
+ console.error('Failed to load conversations:', error);
+ } finally {
+ isLoadingConversations.value = false;
+ }
+ };
+
+ // Load messages for a conversation
+ const loadConversationMessages = debounce(async (conversationId: string) => {
+ if (!conversationId) return;
+
+ isLoadingMessages.value = true;
+ try {
+ const res = await get(
+ `/ai/chat/conversations/${conversationId}/messages`
+ );
+ const messages = (res.data || []).map((msg: any) => {
+ const message: ChatMessage = {
+ ...msg,
+ timestamp: new Date(msg.created_ts || Date.now()),
+ };
+ return message;
+ });
+
+ messages.sort(
+ (a: ChatMessage, b: ChatMessage) =>
+ (a.timestamp?.getTime() || 0) - (b.timestamp?.getTime() || 0)
+ );
+
+ chatHistory.splice(0, chatHistory.length, ...messages);
+ currentConversationId.value = conversationId;
+ } catch (error) {
+ console.error('Failed to load conversation messages:', error);
+ } finally {
+ isLoadingMessages.value = false;
+ }
+ });
+
+ // Load current conversation details
+ const loadCurrentConversation = debounce(async (conversationId: string) => {
+ if (!conversationId) {
+ currentConversation.value = null;
+ return;
+ }
+ try {
+ const res = await get(`/ai/chat/conversations/${conversationId}`);
+ currentConversation.value = res.data;
+ } catch (error) {
+ if ((error as AxiosError)?.response?.status === 404) {
+ currentConversationId.value = '';
+ return;
+ }
+ console.error('Failed to load conversation details:', error);
+ currentConversation.value = null;
+ }
+ });
+
+ // Load LLM providers
+ const loadLLMProviders = debounce(async () => {
+ try {
+ const res = await get('/ai/llm/providers', { available: true });
+ availableProviders.value = res.data || [];
+
+ if (!availableProviders.value.length) {
+ resetChatbotConfig();
+ }
+
+ if (!chatbotConfig.value.provider || !chatbotConfig.value.model) {
+ if (availableProviders.value.length > 0) {
+ chatbotConfig.value.provider = availableProviders.value[0].key!;
+ chatbotConfig.value.model = availableProviders.value[0].models![0];
+ localStorage.setItem(
+ 'chatbotConfig',
+ JSON.stringify(chatbotConfig.value)
+ );
+ }
+ }
+ } catch (error) {
+ console.error('Failed to load LLM providers:', error);
+ }
+ });
+
+ // Config management
+ const loadChatbotConfig = () => {
+ const storedConfig = localStorage.getItem('chatbotConfig');
+ if (storedConfig) {
+ try {
+ const parsedConfig = JSON.parse(storedConfig);
+ chatbotConfig.value = { ...chatbotConfig.value, ...parsedConfig };
+ } catch (e) {
+ console.error('Failed to parse stored chatbot config', e);
+ }
+ }
+ };
+
+ const saveChatbotConfig = (config: ChatbotConfig) => {
+ configDialogVisible.value = false;
+ chatbotConfig.value = { ...chatbotConfig.value, ...config };
+ localStorage.setItem('chatbotConfig', JSON.stringify(chatbotConfig.value));
+ ElMessage.success(t('common.message.success.save'));
+ };
+
+ const resetChatbotConfig = () => {
+ chatbotConfig.value = {};
+ localStorage.removeItem('chatbotConfig');
+ };
+
+ // Message handling
+ const extractErrorMessage = (errorData: string): string => {
+ try {
+ const parsed = JSON.parse(errorData);
+ if (parsed.error_detail?.message) return parsed.error_detail.message;
+ if (parsed.error)
+ return typeof parsed.error === 'string'
+ ? parsed.error
+ : JSON.stringify(parsed.error);
+ if (parsed.text?.startsWith('Error:')) return parsed.text;
+ if (typeof parsed === 'object') return JSON.stringify(parsed, null, 2);
+ return errorData;
+ } catch (e) {
+ return errorData;
+ }
+ };
+
+ // Message stream handling
+ const createChatRequest = (message: string): ChatRequest => {
+ const { provider, model, systemPrompt, temperature, maxTokens } =
+ chatbotConfig.value;
+
+ if (!provider || !model) {
+ throw new Error(
+ 'Please select a provider and model before sending a message'
+ );
+ }
+
+ return {
+ provider,
+ model,
+ query: message,
+ system_prompt: systemPrompt,
+ temperature,
+ max_tokens: maxTokens,
+ conversation_id: currentConversationId.value,
+ };
+ };
+
+ const getStreamHeaders = (): HeadersInit => {
+ const token = localStorage.getItem('token');
+ return {
+ 'Content-Type': 'application/json',
+ ...(token ? { Authorization: token } : {}),
+ };
+ };
+
+ const handleStreamChunk = (
+ chunk: ChatbotStreamMessage,
+ currentMessage: ChatMessage
+ ): void => {
+ // Handle initial conversation ID update
+ if (chunk.is_initial) {
+ currentConversationId.value = chunk.conversation_id!;
+ return;
+ }
+
+ // Update conversation title
+ if (chunk.conversation_title) {
+ if (!currentConversation.value) {
+ currentConversation.value = {};
+ }
+ currentConversation.value.title = chunk.conversation_title;
+ }
+
+ // Update usage
+ if (chunk.usage) {
+ if (!currentMessage.usage) {
+ currentMessage.usage = {
+ prompt_tokens: 0,
+ completion_tokens: 0,
+ total_tokens: 0,
+ };
+ }
+ currentMessage.usage.prompt_tokens! += chunk.usage.prompt_tokens || 0;
+ currentMessage.usage.completion_tokens! +=
+ chunk.usage.completion_tokens || 0;
+ currentMessage.usage.total_tokens! += chunk.usage.total_tokens || 0;
+ }
+
+ // Initialize contents array if needed
+ if (!currentMessage.contents?.length) {
+ currentMessage.contents = [];
+ }
+
+ // Find content index
+ const contentKey = chunk.key || '';
+ const contentIndex = currentMessage.contents.findIndex(
+ c => c.key === contentKey
+ );
+
+ if (contentIndex >= 0) {
+ // Update existing content
+ const content = currentMessage.contents[contentIndex];
+ if (chunk.type === 'text') {
+ content.content += chunk.content || '';
+ if (chunk.is_text_done) {
+ content.isStreaming = false;
+ }
+ } else if (chunk.type === 'action') {
+ currentMessage.contents[contentIndex] = {
+ ...content,
+ ...chunk,
+ };
+ }
+ } else {
+ // Add new content
+ if (chunk.type === 'text') {
+ // Create new content object
+ const newContent: ChatMessageContent = {
+ ...chunk,
+ content: chunk.content || '',
+ isStreaming: true,
+ };
+ // Add new content to the message
+ currentMessage.contents.push(newContent);
+ } else if (chunk.type === 'action') {
+ // Create new action content object
+ currentMessage.contents.push({
+ ...chunk,
+ });
+ }
+ }
+ };
+
+ const processStreamData = async (
+ reader: ReadableStreamDefaultReader,
+ responseIndex: number,
+ onMessageUpdate: (index: number) => void
+ ): Promise => {
+ const decoder = new TextDecoder();
+ let buffer = '';
+
+ while (true) {
+ const { value, done } = await reader.read();
+ const currentMessage = chatHistory[responseIndex];
+
+ if (done) {
+ if (currentMessage) {
+ currentMessage.isStreaming = false;
+ onMessageUpdate(responseIndex);
+ }
+ return;
+ }
+
+ buffer += decoder.decode(value, { stream: true });
+ const lines = buffer.split('\n');
+ buffer = lines.pop() || '';
+
+ for (const line of lines) {
+ if (line.startsWith('data:')) {
+ const eventData = line.slice(5).trim();
+ if (eventData === '') continue;
+
+ try {
+ const chunk: ChatbotStreamMessage = JSON.parse(eventData);
+ handleStreamChunk(chunk, currentMessage);
+
+ if (chunk.is_done) {
+ if (chatHistory[responseIndex]) {
+ chatHistory[responseIndex].isStreaming = false;
+ onMessageUpdate(responseIndex);
+ }
+ return;
+ }
+ } catch (e) {
+ console.error('Error parsing event data:', e);
+ }
+ } else if (line.startsWith('event: error')) {
+ const errorLine = lines.find(l => l.startsWith('data:'));
+ if (errorLine) {
+ const errorData = errorLine.slice(5).trim();
+ throw new Error(extractErrorMessage(errorData));
+ }
+ }
+ }
+ }
+ };
+
+ const sendStreamingRequest = async (
+ message: string,
+ responseIndex: number,
+ onMessageUpdate: (index: number) => void
+ ): Promise => {
+ try {
+ const chatRequest = createChatRequest(message);
+ const baseUrl = getRequestBaseUrl();
+ const url = `${baseUrl}/ai/chat/stream`;
+
+ const response = await fetch(url, {
+ method: 'POST',
+ headers: getStreamHeaders(),
+ body: JSON.stringify(chatRequest),
+ signal: abortController.value?.signal,
+ });
+
+ if (!response.ok) {
+ const text = await response.text();
+ throw new Error(extractErrorMessage(text));
+ }
+
+ await processStreamData(
+ response.body!.getReader(),
+ responseIndex,
+ onMessageUpdate
+ );
+ } catch (error) {
+ if (error instanceof Error) {
+ throw error;
+ }
+ throw new Error('An error occurred while processing the stream');
+ }
+ };
+
+ // Conversation management
+ const selectConversation = async (conversationId: string) => {
+ if (currentConversationId.value === conversationId) return;
+
+ chatInputRef.value?.focus();
+ currentConversationId.value = conversationId;
+ streamError.value = '';
+ await loadConversationMessages(conversationId);
+ };
+
+ const createNewConversation = () => {
+ currentConversationId.value = '';
+ localStorage.removeItem('currentConversationId');
+ streamError.value = '';
+ chatHistory.splice(0, chatHistory.length);
+ chatInputRef.value?.focus();
+ };
+
+ // Watch conversation ID changes
+ watch(currentConversationId, async newId => {
+ if (newId) {
+ localStorage.setItem('currentConversationId', newId);
+ await loadCurrentConversation(newId);
+ } else {
+ localStorage.removeItem('currentConversationId');
+ currentConversation.value = null;
+ }
+
+ if (newId && !currentConversationId.value) {
+ await loadConversations();
+ currentConversationId.value = newId;
+ }
+ });
+
+ onMounted(() => {
+ setTimeout(() => {
+ chatInputRef.value?.focus();
+ }, 200);
+ });
+
+ return {
+ // Refs
+ messageListRef,
+ chatInputRef,
+
+ // State
+ currentConversation,
+ currentConversationId,
+ conversations,
+ chatHistory,
+ isGenerating,
+ streamError,
+ isLoadingConversations,
+ isLoadingMessages,
+ historyDialogVisible,
+ configDialogVisible,
+ abortController,
+ availableProviders,
+ chatbotConfig,
+ currentConversationTitle,
+
+ // Methods
+ loadConversations,
+ loadConversationMessages,
+ loadCurrentConversation,
+ loadLLMProviders,
+ loadChatbotConfig,
+ saveChatbotConfig,
+ resetChatbotConfig,
+ selectConversation,
+ createNewConversation,
+ sendStreamingRequest,
+ extractErrorMessage,
+ };
+};
+
+export default useAssistantConsole;
diff --git a/frontend/crawlab-ui/src/components/core/database/CreateEditDatabaseDialog.vue b/frontend/crawlab-ui/src/components/core/database/CreateEditDatabaseDialog.vue
new file mode 100644
index 00000000..a4f57c00
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/database/CreateEditDatabaseDialog.vue
@@ -0,0 +1,31 @@
+
+
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/database/CreateEditDatabaseTableDialog.vue b/frontend/crawlab-ui/src/components/core/database/CreateEditDatabaseTableDialog.vue
new file mode 100644
index 00000000..e08f6350
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/database/CreateEditDatabaseTableDialog.vue
@@ -0,0 +1,82 @@
+
+
+
+
+ (activeTabName = id)"
+ />
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/database/DatabaseDataSource.vue b/frontend/crawlab-ui/src/components/core/database/DatabaseDataSource.vue
new file mode 100644
index 00000000..2a85c063
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/database/DatabaseDataSource.vue
@@ -0,0 +1,125 @@
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/database/DatabaseDatabaseDetail.vue b/frontend/crawlab-ui/src/components/core/database/DatabaseDatabaseDetail.vue
new file mode 100644
index 00000000..92bfee2c
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/database/DatabaseDatabaseDetail.vue
@@ -0,0 +1,88 @@
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/database/DatabaseForm.vue b/frontend/crawlab-ui/src/components/core/database/DatabaseForm.vue
new file mode 100644
index 00000000..bbc12155
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/database/DatabaseForm.vue
@@ -0,0 +1,219 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ {{ op.label }}
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/database/DatabaseSidebar.vue b/frontend/crawlab-ui/src/components/core/database/DatabaseSidebar.vue
new file mode 100644
index 00000000..4f27edf4
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/database/DatabaseSidebar.vue
@@ -0,0 +1,935 @@
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/database/DatabaseStatus.vue b/frontend/crawlab-ui/src/components/core/database/DatabaseStatus.vue
new file mode 100644
index 00000000..5fa990cc
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/database/DatabaseStatus.vue
@@ -0,0 +1,77 @@
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/database/DatabaseTableDetail.vue b/frontend/crawlab-ui/src/components/core/database/DatabaseTableDetail.vue
new file mode 100644
index 00000000..8b1c9889
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/database/DatabaseTableDetail.vue
@@ -0,0 +1,277 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+ {
+ internalTable = val;
+ }
+ "
+ />
+
+
+ {
+ internalTable = val;
+ }
+ "
+ />
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/database/nav/DatabaseNavTabs.vue b/frontend/crawlab-ui/src/components/core/database/nav/DatabaseNavTabs.vue
new file mode 100644
index 00000000..7ab56bb6
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/database/nav/DatabaseNavTabs.vue
@@ -0,0 +1,65 @@
+
+
+
+ (activeTabName = key)"
+ >
+
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/database/tables/DatabaseTableDetailColumns.vue b/frontend/crawlab-ui/src/components/core/database/tables/DatabaseTableDetailColumns.vue
new file mode 100644
index 00000000..16f3ddf6
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/database/tables/DatabaseTableDetailColumns.vue
@@ -0,0 +1,357 @@
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/database/tables/DatabaseTableDetailData.vue b/frontend/crawlab-ui/src/components/core/database/tables/DatabaseTableDetailData.vue
new file mode 100644
index 00000000..31bca8c3
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/database/tables/DatabaseTableDetailData.vue
@@ -0,0 +1,375 @@
+
+
+
+ selectedRows = selection"
+ >
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/database/tables/DatabaseTableDetailIndexes.vue b/frontend/crawlab-ui/src/components/core/database/tables/DatabaseTableDetailIndexes.vue
new file mode 100644
index 00000000..c4422292
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/database/tables/DatabaseTableDetailIndexes.vue
@@ -0,0 +1,427 @@
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/database/useDatabase.ts b/frontend/crawlab-ui/src/components/core/database/useDatabase.ts
new file mode 100644
index 00000000..aff9f80f
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/database/useDatabase.ts
@@ -0,0 +1,118 @@
+import { computed } from 'vue';
+import { Store } from 'vuex';
+import { ElMessage, ElMessageBox } from 'element-plus';
+import useDatabaseService from '@/services/database/databaseService';
+import {
+ EMPTY_OBJECT_ID,
+ getDefaultFormComponentData,
+ plainClone,
+ translate,
+} from '@/utils';
+import useForm from '@/components/ui/form/useForm';
+import { databaseDefaults } from '@/utils/database';
+
+// i18n
+const t = translate;
+
+// form component data
+const formComponentData = getDefaultFormComponentData();
+
+export const useDatabase = (store: Store) => {
+ // store
+ const ns = 'database' as ListStoreNamespace;
+ const { database: state } = store.state as RootStoreState;
+
+ // form rules
+ const formRules: FormRules = {};
+
+ const availableDatabaseNames: DatabaseDataSource[] = [
+ 'mongo',
+ 'mysql',
+ 'postgres',
+ 'mssql',
+ 'elasticsearch',
+ ];
+
+ // type options
+ const dataSourceOptions = computed[]>(() =>
+ databaseDefaults.map(item => {
+ const disabled = !availableDatabaseNames.includes(item.data_source!);
+ const label = !disabled
+ ? item.name
+ : `${item.name} (${t('common.status.currentlyUnavailable')})`;
+ const value = item.data_source;
+ return {
+ label,
+ value,
+ disabled,
+ };
+ })
+ );
+ const getTypeOptionsWithDefault = (): SelectOption[] => {
+ return [
+ { label: t('components.database.dataSources.default'), value: undefined },
+ ...dataSourceOptions.value,
+ ];
+ };
+
+ // on change password function
+ const onChangePasswordFunc = async (id?: string) => {
+ if (!id) return;
+
+ const { value } = await ElMessageBox.prompt(
+ t('components.user.messageBox.prompt.changePassword'),
+ t('components.user.form.changePassword'),
+ {
+ inputType: 'password',
+ inputPlaceholder: t('components.user.form.newPassword'),
+ }
+ );
+
+ await store.dispatch(`${ns}/changePassword`, { id, password: value });
+ ElMessage.success(t('common.message.success.save'));
+ };
+
+ // on hosts add
+ const onHostsAdd = (index: number) => {
+ const form = plainClone(state.form);
+ if (!form.hosts) form.hosts = [];
+ form.hosts?.splice(index + 1, 0, '');
+ store.commit(`${ns}/setForm`, { ...form });
+ };
+
+ // on hosts delete
+ const onHostsDelete = (index: number) => {
+ const form = plainClone(state.form);
+ form.hosts?.splice(index, 1);
+ if (form.hosts?.length === 0) form.hosts?.push('');
+ store.commit(`${ns}/setForm`, { ...form });
+ };
+
+ const allListSelectOptions = computed(
+ () =>
+ store.getters[`${ns}/allListSelectOptions`]?.map((op: SelectOption) => {
+ if (op.value === EMPTY_OBJECT_ID) {
+ return { ...op, label: t('components.database.default.name') };
+ }
+ return op;
+ }) || []
+ );
+
+ return {
+ ...useForm(
+ ns,
+ store,
+ useDatabaseService(store),
+ formComponentData
+ ),
+ formRules,
+ dataSourceOptions,
+ allListSelectOptions,
+ getTypeOptionsWithDefault,
+ onChangePasswordFunc,
+ onHostsAdd,
+ onHostsDelete,
+ };
+};
+
+export default useDatabase;
diff --git a/frontend/crawlab-ui/src/components/core/dependency/DependencyConfigDialog.vue b/frontend/crawlab-ui/src/components/core/dependency/DependencyConfigDialog.vue
new file mode 100644
index 00000000..2edcd5fe
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/dependency/DependencyConfigDialog.vue
@@ -0,0 +1,109 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/dependency/DependencyInstallDialog.vue b/frontend/crawlab-ui/src/components/core/dependency/DependencyInstallDialog.vue
new file mode 100644
index 00000000..63c70be3
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/dependency/DependencyInstallDialog.vue
@@ -0,0 +1,143 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ {{ n.name }}
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/dependency/DependencyLogsDialog.vue b/frontend/crawlab-ui/src/components/core/dependency/DependencyLogsDialog.vue
new file mode 100644
index 00000000..b421efdb
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/dependency/DependencyLogsDialog.vue
@@ -0,0 +1,109 @@
+
+
+
+
+
+
+ {{ t('common.tabs.logs') }} - {{ activeTargetName }}
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/dependency/DependencySetupDialog.vue b/frontend/crawlab-ui/src/components/core/dependency/DependencySetupDialog.vue
new file mode 100644
index 00000000..fc19339f
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/dependency/DependencySetupDialog.vue
@@ -0,0 +1,243 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ {{ n.name }}
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ {{
+ t('views.env.deps.config.alert.browser.nodeSetupRequired.content')
+ }}
+
+
+ {
+ store.commit(`${ns}/setLang`, 'node');
+ store.commit(`${ns}/setRepoTabName`, 'nodes');
+ store.commit(`${ns}/hideDialog`);
+ }
+ "
+ />
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/dependency/DependencyStatusTag.vue b/frontend/crawlab-ui/src/components/core/dependency/DependencyStatusTag.vue
new file mode 100644
index 00000000..fed49593
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/dependency/DependencyStatusTag.vue
@@ -0,0 +1,87 @@
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/dependency/DependencyUninstallDialog.vue b/frontend/crawlab-ui/src/components/core/dependency/DependencyUninstallDialog.vue
new file mode 100644
index 00000000..01d1bea7
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/dependency/DependencyUninstallDialog.vue
@@ -0,0 +1,126 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ {{ n.name }}
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/dependency/DependencyVersions.vue b/frontend/crawlab-ui/src/components/core/dependency/DependencyVersions.vue
new file mode 100644
index 00000000..689b88d2
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/dependency/DependencyVersions.vue
@@ -0,0 +1,107 @@
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/environment/CreateEditEnvironmentDialog.vue b/frontend/crawlab-ui/src/components/core/environment/CreateEditEnvironmentDialog.vue
new file mode 100644
index 00000000..b9f77634
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/environment/CreateEditEnvironmentDialog.vue
@@ -0,0 +1,30 @@
+
+
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/environment/EnvironmentForm.vue b/frontend/crawlab-ui/src/components/core/environment/EnvironmentForm.vue
new file mode 100644
index 00000000..e775049e
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/environment/EnvironmentForm.vue
@@ -0,0 +1,71 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/environment/useEnvironment.ts b/frontend/crawlab-ui/src/components/core/environment/useEnvironment.ts
new file mode 100644
index 00000000..4378ff60
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/environment/useEnvironment.ts
@@ -0,0 +1,26 @@
+import { Store } from 'vuex';
+import useEnvironmentService from '@/services/environment/environmentService';
+import { getDefaultFormComponentData, translate } from '@/utils';
+import useForm from '@/components/ui/form/useForm';
+
+// i18n
+const t = translate;
+
+// form component data
+const formComponentData = getDefaultFormComponentData();
+
+export const useEnvironment: any = (store: Store) => {
+ // store
+ const ns = 'environment' as ListStoreNamespace;
+
+ return {
+ ...useForm(
+ ns,
+ store,
+ useEnvironmentService(store),
+ formComponentData
+ ),
+ };
+};
+
+export default useEnvironment;
diff --git a/frontend/crawlab-ui/src/components/core/git/CreateEditGitDialog.vue b/frontend/crawlab-ui/src/components/core/git/CreateEditGitDialog.vue
new file mode 100644
index 00000000..10684bac
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/git/CreateEditGitDialog.vue
@@ -0,0 +1,32 @@
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/git/CreateGitBranchDialog.vue b/frontend/crawlab-ui/src/components/core/git/CreateGitBranchDialog.vue
new file mode 100644
index 00000000..927fd69d
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/git/CreateGitBranchDialog.vue
@@ -0,0 +1,115 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/git/CreateGitSpiderDialog.vue b/frontend/crawlab-ui/src/components/core/git/CreateGitSpiderDialog.vue
new file mode 100644
index 00000000..b6d01933
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/git/CreateGitSpiderDialog.vue
@@ -0,0 +1,135 @@
+
+
+
+
+
+
+
+
+
+
+
+ {{ label }}
+
+
+ {{ value }}
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/git/GitBranchSelect.vue b/frontend/crawlab-ui/src/components/core/git/GitBranchSelect.vue
new file mode 100644
index 00000000..2e8f38af
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/git/GitBranchSelect.vue
@@ -0,0 +1,322 @@
+
+
+
+
+
+
+
+
+
+
+ {{ label }}
+
+
+
+
+
+
+
+
+ {{ op.label }}
+
+
+ {{ op.branch?.remote_track }}
+
+
+ onDeleteBranch(op.value as string, event)
+ "
+ />
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ -
+
+
+
+ {{ t('components.git.branches.pull') }}
+
+
+
+ -
+
+
+
+ {{ t('components.git.branches.commit') }}
+
+
+
+ -
+
+
+
+ {{ t('components.git.branches.push') }}
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/git/GitCloneLogsDialog.vue b/frontend/crawlab-ui/src/components/core/git/GitCloneLogsDialog.vue
new file mode 100644
index 00000000..e133a17f
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/git/GitCloneLogsDialog.vue
@@ -0,0 +1,58 @@
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/git/GitFileDiffDialog.vue b/frontend/crawlab-ui/src/components/core/git/GitFileDiffDialog.vue
new file mode 100644
index 00000000..6b62f393
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/git/GitFileDiffDialog.vue
@@ -0,0 +1,93 @@
+
+
+
+
+
+
+
+
+
+ {{ state.activeFilePath }}
+
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/git/GitFileStatus.vue b/frontend/crawlab-ui/src/components/core/git/GitFileStatus.vue
new file mode 100644
index 00000000..92126037
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/git/GitFileStatus.vue
@@ -0,0 +1,65 @@
+
+
+
+
+
+ {{ fileStatus.name }}
+
+
+ {{ fileStatus.path }}
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/git/GitForm.vue b/frontend/crawlab-ui/src/components/core/git/GitForm.vue
new file mode 100644
index 00000000..6e325aea
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/git/GitForm.vue
@@ -0,0 +1,201 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/git/GitLogsBox.vue b/frontend/crawlab-ui/src/components/core/git/GitLogsBox.vue
new file mode 100644
index 00000000..5a091601
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/git/GitLogsBox.vue
@@ -0,0 +1,33 @@
+
+
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/git/GitPath.vue b/frontend/crawlab-ui/src/components/core/git/GitPath.vue
new file mode 100644
index 00000000..201f38ca
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/git/GitPath.vue
@@ -0,0 +1,17 @@
+
+
+
+ {{ path }}
+
+ {{ FILE_ROOT }} ({{ t('components.file.rootDirectory') }})
+
+
diff --git a/frontend/crawlab-ui/src/components/core/git/GitRepo.vue b/frontend/crawlab-ui/src/components/core/git/GitRepo.vue
new file mode 100644
index 00000000..80041760
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/git/GitRepo.vue
@@ -0,0 +1,31 @@
+
+
+
+
+
+
+ {{ gitRootPath }}
+ ~ ({{ t('components.file.rootDirectory') }})
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/git/GitStatus.vue b/frontend/crawlab-ui/src/components/core/git/GitStatus.vue
new file mode 100644
index 00000000..38569c87
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/git/GitStatus.vue
@@ -0,0 +1,168 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/git/UploadGitFilesDialog.vue b/frontend/crawlab-ui/src/components/core/git/UploadGitFilesDialog.vue
new file mode 100644
index 00000000..2202cfe1
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/git/UploadGitFilesDialog.vue
@@ -0,0 +1,24 @@
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/git/useGit.ts b/frontend/crawlab-ui/src/components/core/git/useGit.ts
new file mode 100644
index 00000000..b35d3a78
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/git/useGit.ts
@@ -0,0 +1,34 @@
+import { Store } from 'vuex';
+import useForm from '@/components/ui/form/useForm';
+import useGitService from '@/services/git/gitService';
+import { getDefaultFormComponentData } from '@/utils/form';
+
+// form component data
+const formComponentData = getDefaultFormComponentData();
+
+const getGitIcon = (row: Git): { icon: Icon; color?: string; name: string } => {
+ if (row.url?.includes('github')) {
+ return { icon: ['fab', 'github'], color: '#0d1117', name: 'github' };
+ } else if (row.url?.includes('bitbucket')) {
+ return { icon: ['fab', 'bitbucket'], color: '#0052cc', name: 'bitbucket' };
+ } else if (row.url?.includes('gitlab')) {
+ return { icon: ['fab', 'gitlab'], color: '#E24329', name: 'gitlab' };
+ } else if (row.url?.includes('amazonaws')) {
+ return { icon: ['fab', 'aws'], color: '#232f3e', name: 'aws' };
+ } else {
+ return {
+ icon: ['fab', 'git'],
+ color: 'var(--cl-info-medium-dark-color)',
+ name: 'git',
+ };
+ }
+};
+
+const useGit = (store: Store) => {
+ return {
+ ...useForm('git', store, useGitService(store), formComponentData),
+ getGitIcon,
+ };
+};
+
+export default useGit;
diff --git a/frontend/crawlab-ui/src/components/core/metric/CurrentMetrics.vue b/frontend/crawlab-ui/src/components/core/metric/CurrentMetrics.vue
new file mode 100644
index 00000000..f7937f15
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/metric/CurrentMetrics.vue
@@ -0,0 +1,193 @@
+
+
+
+
+
+ emit('click', event)"
+ >
+
+
+
+
+ {{ Math.round(metric?.cpu_usage_percent || 0) }}%
+
+
+
+
+ emit('click', event)"
+ >
+
+
+
+
+ {{ formatBytes(metric?.used_memory) }} ({{
+ Math.round(metric?.used_memory_percent || 0)
+ }}%)
+
+
+
+
+
+ {{ formatBytes(metric?.total_memory) }}
+
+
+
+
+ emit('click', event)"
+ >
+
+
+
+
+ {{ formatBytes(metric?.used_disk) }} ({{
+ Math.round(metric?.used_disk_percent || 0)
+ }}%)
+
+
+
+
+
+ {{ formatBytes(metric?.total_disk) }}
+
+
+
+
+ emit('click', event)"
+ >
+
+
+
+
+ {{ formatBytes(metric?.used_memory) }}
+
+
+
+
+ emit('click', event)"
+ >
+
+
+
+
+ {{ formatBytes(metric?.used_disk) }}
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/metric/MetricMonitoringDetail.vue b/frontend/crawlab-ui/src/components/core/metric/MetricMonitoringDetail.vue
new file mode 100644
index 00000000..940646d2
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/metric/MetricMonitoringDetail.vue
@@ -0,0 +1,379 @@
+
+
+
+
+
+
+ emit('time-range-change', value)"
+ >
+
+
+
+
+
+
+
+ emit('metric-groups-change', value)"
+ >
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ {{ getMetricGroupLabel(name) }}
+
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/node/CreateEditNodeDialog.vue b/frontend/crawlab-ui/src/components/core/node/CreateEditNodeDialog.vue
new file mode 100644
index 00000000..687b3cb1
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/node/CreateEditNodeDialog.vue
@@ -0,0 +1,34 @@
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/node/NodeActive.vue b/frontend/crawlab-ui/src/components/core/node/NodeActive.vue
new file mode 100644
index 00000000..ebe85b36
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/node/NodeActive.vue
@@ -0,0 +1,38 @@
+
+
+
+
+
+ {{ label }}
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/node/NodeForm.vue b/frontend/crawlab-ui/src/components/core/node/NodeForm.vue
new file mode 100644
index 00000000..81247986
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/node/NodeForm.vue
@@ -0,0 +1,152 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ {{ t('common.mode.unlimited') }}
+
+
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/node/NodeRunners.vue b/frontend/crawlab-ui/src/components/core/node/NodeRunners.vue
new file mode 100644
index 00000000..3af79876
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/node/NodeRunners.vue
@@ -0,0 +1,92 @@
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/node/NodeStatus.vue b/frontend/crawlab-ui/src/components/core/node/NodeStatus.vue
new file mode 100644
index 00000000..5bc80175
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/node/NodeStatus.vue
@@ -0,0 +1,58 @@
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/node/NodeTag.vue b/frontend/crawlab-ui/src/components/core/node/NodeTag.vue
new file mode 100644
index 00000000..846db936
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/node/NodeTag.vue
@@ -0,0 +1,154 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/node/NodeType.vue b/frontend/crawlab-ui/src/components/core/node/NodeType.vue
new file mode 100644
index 00000000..6052912c
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/node/NodeType.vue
@@ -0,0 +1,84 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/node/useNode.ts b/frontend/crawlab-ui/src/components/core/node/useNode.ts
new file mode 100644
index 00000000..98fcbeb8
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/node/useNode.ts
@@ -0,0 +1,37 @@
+import { Store } from 'vuex';
+import useForm from '@/components/ui/form/useForm';
+import useNodeService from '@/services/node/nodeService';
+import { getDefaultFormComponentData } from '@/utils/form';
+import { computed } from 'vue';
+
+type Node = CNode;
+
+// form component data
+const formComponentData = getDefaultFormComponentData();
+
+const useNode = (store: Store) => {
+ // store
+ const ns = 'node';
+ const { node: state } = store.state;
+
+ // form rules
+ const formRules: FormRules = {};
+
+ const activeNodesSorted = computed(() => {
+ return state.allList
+ .filter(n => n.active)
+ .sort((a, b) => {
+ if (a.is_master) return -1;
+ if (b.is_master) return 1;
+ return a.name!.localeCompare(b.name!);
+ });
+ });
+
+ return {
+ ...useForm(ns, store, useNodeService(store), formComponentData),
+ formRules,
+ activeNodesSorted,
+ };
+};
+
+export default useNode;
diff --git a/frontend/crawlab-ui/src/components/core/notification/alert/CreateEditNotificationAlertDialog.vue b/frontend/crawlab-ui/src/components/core/notification/alert/CreateEditNotificationAlertDialog.vue
new file mode 100644
index 00000000..948639d6
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/notification/alert/CreateEditNotificationAlertDialog.vue
@@ -0,0 +1,32 @@
+
+
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/notification/alert/NotificationAlertForm.vue b/frontend/crawlab-ui/src/components/core/notification/alert/NotificationAlertForm.vue
new file mode 100644
index 00000000..be79a6ae
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/notification/alert/NotificationAlertForm.vue
@@ -0,0 +1,228 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ {{ t('views.notification.alerts.form.hasMetricTarget') }}
+
+
+
+
+
+
+
+
+
+
+
+
+
+ {
+ form.target_value = parseFloat(val);
+ }
+ "
+ >
+
+ {{ getMetricUnitLabel(form.metric_name || '') }}
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/notification/alert/useNotificationAlert.ts b/frontend/crawlab-ui/src/components/core/notification/alert/useNotificationAlert.ts
new file mode 100644
index 00000000..4404adc9
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/notification/alert/useNotificationAlert.ts
@@ -0,0 +1,99 @@
+import { computed } from 'vue';
+import { Store } from 'vuex';
+import { useRoute } from 'vue-router';
+import {
+ EMPTY_OBJECT_ID,
+ getDefaultFormComponentData,
+ translate,
+} from '@/utils';
+import useNotificationAlertService from '@/services/notification/useNotificationAlertService';
+import useForm from '@/components/ui/form/useForm';
+import { getAllMetricGroups } from '@/utils/metric';
+
+const t = translate;
+
+// form component data
+const formComponentData = getDefaultFormComponentData();
+
+const useNotificationAlert = (store: Store) => {
+ const { notificationAlert: state, node: nodeState } =
+ store.state as RootStoreState;
+
+ // route
+ const route = useRoute();
+
+ // notification id
+ const id = computed(() => route.params.id);
+
+ const form = computed(() => state.form);
+
+ const metricTargetOptions = computed(
+ () =>
+ [
+ { label: t('common.mode.all'), value: EMPTY_OBJECT_ID },
+ ...nodeState.allList
+ .filter(node => node.active)
+ .map(node => ({
+ label: node.name,
+ value: node._id,
+ })),
+ ] as SelectOption[]
+ );
+
+ const metricNameOptions = computed(() => {
+ const options = [] as SelectOption[];
+ getAllMetricGroups().forEach(group => {
+ group.metrics.forEach(metric => {
+ options.push({
+ label: t(`components.metric.metrics.${metric}`),
+ value: metric as string,
+ });
+ });
+ });
+ return options;
+ });
+
+ const operatorOptions = computed[]>(
+ () => [
+ { value: 'ge', label: '>=' },
+ { value: 'le', label: '<=' },
+ { value: 'gt', label: '>' },
+ { value: 'lt', label: '<' },
+ ]
+ );
+
+ const lastingSecondsOptions = computed[]>(() => [
+ { value: 60, label: t('views.notification.alerts.lastingDuration.1m') },
+ { value: 300, label: t('views.notification.alerts.lastingDuration.5m') },
+ { value: 600, label: t('views.notification.alerts.lastingDuration.10m') },
+ { value: 1800, label: t('views.notification.alerts.lastingDuration.30m') },
+ { value: 3600, label: t('views.notification.alerts.lastingDuration.1h') },
+ ]);
+
+ const levelOptions = computed[]>(() => [
+ { value: 'info', label: t('views.notification.alerts.levels.info') },
+ { value: 'warning', label: t('views.notification.alerts.levels.warning') },
+ {
+ value: 'critical',
+ label: t('views.notification.alerts.levels.critical'),
+ },
+ ]);
+
+ return {
+ ...useForm(
+ 'notificationAlert',
+ store,
+ useNotificationAlertService(store),
+ formComponentData
+ ),
+ id,
+ form,
+ metricTargetOptions,
+ metricNameOptions,
+ operatorOptions,
+ lastingSecondsOptions,
+ levelOptions,
+ };
+};
+
+export default useNotificationAlert;
diff --git a/frontend/crawlab-ui/src/components/core/notification/channel/CreateEditNotificationChannelDialog.vue b/frontend/crawlab-ui/src/components/core/notification/channel/CreateEditNotificationChannelDialog.vue
new file mode 100644
index 00000000..ec0f39c9
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/notification/channel/CreateEditNotificationChannelDialog.vue
@@ -0,0 +1,33 @@
+
+
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/notification/channel/NotificationChannelForm.vue b/frontend/crawlab-ui/src/components/core/notification/channel/NotificationChannelForm.vue
new file mode 100644
index 00000000..8e62d5fc
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/notification/channel/NotificationChannelForm.vue
@@ -0,0 +1,385 @@
+
+
+
+
+
+
+
+
+
+
+
+
+ {{ op.label }}
+
+
+
+
+
+
+
+
+
+
+ {{ activeProviderOption.label }}
+
+
+
+
+
+
+ {{ op.label }}
+
+
+
+
+
+
+ {{ t('views.notification.channels.providers.custom') }}
+
+
+
+
+
+
+
+
+
+ {{ t('views.notification.channels.providerDocs.label') }}
+
+ -
+
+ {{
+ t(`views.notification.channels.providers.${activeProvider?.name}`)
+ }}
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/notification/channel/useNotificationChannel.ts b/frontend/crawlab-ui/src/components/core/notification/channel/useNotificationChannel.ts
new file mode 100644
index 00000000..e96225a9
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/notification/channel/useNotificationChannel.ts
@@ -0,0 +1,273 @@
+import { computed, watch } from 'vue';
+import { Store } from 'vuex';
+import { useRoute } from 'vue-router';
+import { translate, getDefaultFormComponentData } from '@/utils';
+import useNotificationChannelService from '@/services/notification/useNotificationChannelService';
+import useForm from '@/components/ui/form/useForm';
+import { useI18n } from 'vue-i18n';
+
+const t = translate;
+
+// form component data
+const formComponentData = getDefaultFormComponentData();
+
+const useNotificationChannel = (store: Store) => {
+ const { notificationChannel: state } = store.state as RootStoreState;
+
+ const { locale } = useI18n();
+
+ // route
+ const route = useRoute();
+
+ // notification id
+ const id = computed(() => route.params.id);
+
+ const form = computed(() => state.form);
+
+ const typeOptions = computed(() => [
+ {
+ value: 'mail',
+ label: t('views.notification.channels.types.mail'),
+ icon: ['fa', 'at'],
+ },
+ {
+ value: 'im',
+ label: t('views.notification.channels.types.im'),
+ icon: ['far', 'comment'],
+ },
+ ]);
+
+ const allProviders: NotificationChannelProvider[] = [
+ {
+ type: 'mail',
+ name: 'gmail',
+ icon: ['svg', 'gmail'],
+ smtpServer: 'smtp.gmail.com',
+ smtpPort: 587,
+ docUrl: () =>
+ `https://support.google.com/a/answer/2956491?hl=${locale.value}`,
+ locale: 'en',
+ disabled: true,
+ },
+ {
+ type: 'mail',
+ name: 'outlook',
+ icon: ['svg', 'outlook'],
+ smtpServer: 'smtp-mail.outlook.com',
+ smtpPort: 587,
+ docUrl: () => {
+ if (locale.value === 'zh') {
+ return 'https://support.microsoft.com/zh-cn/office/outlook-com-%E7%9A%84-pop-imap-%E5%92%8C-smtp-%E8%AE%BE%E7%BD%AE-d088b986-291d-42b8-9564-9c414e2aa040';
+ } else {
+ return 'https://support.microsoft.com/en-us/office/pop-imap-and-smtp-settings-for-outlook-com-d088b986-291d-42b8-9564-9c414e2aa040';
+ }
+ },
+ locale: 'en',
+ },
+ {
+ type: 'mail',
+ name: 'qq',
+ icon: ['fab', 'qq'],
+ smtpServer: 'smtp.qq.com',
+ smtpPort: 587,
+ docUrl: 'https://cloud.tencent.com/developer/article/2177098',
+ locale: 'zh',
+ },
+ {
+ type: 'mail',
+ name: '163',
+ icon: ['svg', 'netease'],
+ smtpServer: 'smtp.163.com',
+ smtpPort: 465,
+ docUrl: 'https://www.bilibili.com/read/cv32056718/',
+ locale: 'zh',
+ },
+ {
+ type: 'mail',
+ name: 'icloud',
+ icon: ['fab', 'apple'],
+ smtpServer: 'smtp.mail.me.com',
+ smtpPort: 587,
+ docUrl: () => {
+ if (locale.value === 'zh') {
+ return 'https://support.apple.com/zh-cn/102525';
+ } else {
+ return 'https://support.apple.com/en-us/102525';
+ }
+ },
+ locale: 'en',
+ },
+ {
+ type: 'mail',
+ name: 'yahoo',
+ icon: ['fab', 'yahoo'],
+ smtpServer: 'smtp.mail.yahoo.com',
+ smtpPort: 587,
+ docUrl:
+ 'https://smartreach.io/blog/masterclass/smtp/yahoo-smtp-settings/',
+ locale: 'en',
+ },
+ {
+ type: 'mail',
+ name: 'zoho',
+ icon: ['svg', 'zoho'],
+ smtpServer: 'smtp.zoho.com',
+ smtpPort: 587,
+ docUrl: 'https://www.zoho.com/mail/help/zoho-smtp.html',
+ locale: 'en',
+ },
+ {
+ type: 'mail',
+ name: 'aol',
+ icon: ['svg', 'aol'],
+ smtpServer: 'smtp.aol.com',
+ smtpPort: 587,
+ docUrl: 'https://smartreach.io/blog/masterclass/smtp/aol-mail-settings/',
+ locale: 'en',
+ },
+ {
+ type: 'im',
+ name: 'dingtalk',
+ icon: ['svg', 'dingtalk'],
+ docUrl: 'https://open.dingtalk.com/document/orgapp/custom-robot-access',
+ locale: 'zh',
+ },
+ {
+ type: 'im',
+ name: 'lark',
+ icon: ['svg', 'lark'],
+ docUrl: () => {
+ if (locale.value === 'zh') {
+ return 'https://open.feishu.cn/document/client-docs/bot-v3/add-custom-bot';
+ } else {
+ return 'https://open.larksuite.com/document/client-docs/bot-v3/add-custom-bot';
+ }
+ },
+ locale: 'zh',
+ },
+ {
+ type: 'im',
+ name: 'wechat_work',
+ icon: ['svg', 'wechat_work'],
+ docUrl: 'https://developer.work.weixin.qq.com/document/path/91770',
+ locale: 'zh',
+ },
+ {
+ type: 'im',
+ name: 'slack',
+ icon: ['fab', 'slack'],
+ docUrl: 'https://api.slack.com/messaging/webhooks',
+ locale: 'en',
+ },
+ {
+ type: 'im',
+ name: 'ms_teams',
+ icon: ['svg', 'ms_teams'],
+ docUrl: () =>
+ `https://learn.microsoft.com/${locale.value === 'zh' ? 'zh-cn' : 'en-us'}/microsoftteams/platform/webhooks-and-connectors/how-to/add-incoming-webhook`,
+ locale: 'en',
+ },
+ {
+ type: 'im',
+ name: 'telegram',
+ icon: ['fab', 'telegram'],
+ docUrl: 'https://core.telegram.org/bots/api',
+ locale: 'en',
+ },
+ {
+ type: 'im',
+ name: 'discord',
+ icon: ['fab', 'discord'],
+ docUrl: 'https://discord.com/developers/docs/resources/webhook',
+ locale: 'en',
+ },
+ ];
+
+ const providerOptionGroups = computed(() => {
+ const map: Record = {};
+ allProviders
+ .sort((a, b) => {
+ if (a.disabled !== b.disabled) {
+ return a.disabled ? 1 : -1;
+ }
+ if (a.locale === b.locale) return 0;
+ return a.locale === locale.value ? -1 : 1;
+ })
+ .forEach(p => {
+ const op: SelectOption = {
+ value: p.name,
+ label: t(`views.notification.channels.providers.${p.name}`),
+ icon: p.icon,
+ disabled: p.disabled,
+ };
+ if (!map[p.type]) {
+ map[p.type] = [];
+ }
+ map[p.type].push(op);
+ });
+ return Object.keys(map).map(key => {
+ return {
+ label: t(`views.notification.channels.types.${key}`),
+ children: map[key],
+ };
+ });
+ });
+
+ const activeProvider = computed(() => {
+ const provider = allProviders.find(p => p.name === form.value.provider);
+ if (!provider) {
+ return null;
+ }
+ return provider;
+ });
+
+ const activeProviderOption = computed(() => {
+ if (form.value.provider === 'custom') {
+ return {
+ value: form.value.provider,
+ label: t('views.notification.channels.providers.custom'),
+ icon: ['fa', 'edit'],
+ };
+ }
+ if (!activeProvider.value) {
+ return {
+ value: '',
+ label: '',
+ icon: [],
+ };
+ }
+ const { name, icon, disabled } = activeProvider.value;
+ return {
+ value: name,
+ label: t(`views.notification.channels.providers.${name}`),
+ icon,
+ disabled,
+ };
+ });
+
+ const allProviderNames = computed(() => allProviders.map(p => p.name));
+
+ const getProviderIcon = (provider: string): Icon | undefined => {
+ return allProviders.find(p => p.name === provider)?.icon;
+ };
+
+ return {
+ ...useForm(
+ 'notificationChannel',
+ store,
+ useNotificationChannelService(store),
+ formComponentData
+ ),
+ id,
+ form,
+ typeOptions,
+ providerOptionGroups,
+ activeProvider,
+ activeProviderOption,
+ allProviders,
+ allProviderNames,
+ getProviderIcon,
+ };
+};
+
+export default useNotificationChannel;
diff --git a/frontend/crawlab-ui/src/components/core/notification/request/NotificationRequestStatus.vue b/frontend/crawlab-ui/src/components/core/notification/request/NotificationRequestStatus.vue
new file mode 100644
index 00000000..7e08ad66
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/notification/request/NotificationRequestStatus.vue
@@ -0,0 +1,74 @@
+
+
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/notification/setting/CreateEditNotificationSettingDialog.vue b/frontend/crawlab-ui/src/components/core/notification/setting/CreateEditNotificationSettingDialog.vue
new file mode 100644
index 00000000..efc873f2
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/notification/setting/CreateEditNotificationSettingDialog.vue
@@ -0,0 +1,32 @@
+
+
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/notification/setting/NotificationSettingForm.vue b/frontend/crawlab-ui/src/components/core/notification/setting/NotificationSettingForm.vue
new file mode 100644
index 00000000..b37ec7f8
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/notification/setting/NotificationSettingForm.vue
@@ -0,0 +1,271 @@
+
+
+
+
+
+
+
+
+
+
+
+
+ {{ t('views.notification.settings.form.useCustomSetting.label') }}
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ {{ t('common.actions.cancel') }}
+
+
+ {{ t('common.actions.confirm') }}
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/notification/setting/NotificationSettingTriggerSelect.vue b/frontend/crawlab-ui/src/components/core/notification/setting/NotificationSettingTriggerSelect.vue
new file mode 100644
index 00000000..b709ab49
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/notification/setting/NotificationSettingTriggerSelect.vue
@@ -0,0 +1,88 @@
+
+
+
+ emit('trigger-change', val)"
+ >
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ {{ label }}
+
+
+
+
+ {{ data.label }}
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/notification/setting/useNotificationSetting.ts b/frontend/crawlab-ui/src/components/core/notification/setting/useNotificationSetting.ts
new file mode 100644
index 00000000..a55267f3
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/notification/setting/useNotificationSetting.ts
@@ -0,0 +1,37 @@
+import { useRoute } from 'vue-router';
+import { computed, watch } from 'vue';
+import { Store } from 'vuex';
+import useForm from '@/components/ui/form/useForm';
+import useNotificationSettingService from '@/services/notification/useNotificationSettingService';
+import { getDefaultFormComponentData } from '@/utils/form';
+import { setupGetAllList } from '@/utils';
+
+// form component data
+const formComponentData = getDefaultFormComponentData();
+
+const useNotificationSetting = (store: Store) => {
+ const { notificationSetting: state } = store.state as RootStoreState;
+
+ // route
+ const route = useRoute();
+
+ // notification id
+ const id = computed(() => route.params.id);
+
+ const form = computed(() => state.form);
+
+ setupGetAllList(store, ['notificationAlert']);
+
+ return {
+ ...useForm(
+ 'notificationSetting',
+ store,
+ useNotificationSettingService(store),
+ formComponentData
+ ),
+ id,
+ form,
+ };
+};
+
+export default useNotificationSetting;
diff --git a/frontend/crawlab-ui/src/components/core/project/CreateEditProjectDialog.vue b/frontend/crawlab-ui/src/components/core/project/CreateEditProjectDialog.vue
new file mode 100644
index 00000000..d19064f7
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/project/CreateEditProjectDialog.vue
@@ -0,0 +1,34 @@
+
+
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/project/ProjectForm.vue b/frontend/crawlab-ui/src/components/core/project/ProjectForm.vue
new file mode 100644
index 00000000..3bc2999a
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/project/ProjectForm.vue
@@ -0,0 +1,53 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/project/useProject.ts b/frontend/crawlab-ui/src/components/core/project/useProject.ts
new file mode 100644
index 00000000..9e1784c6
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/project/useProject.ts
@@ -0,0 +1,46 @@
+import { computed, readonly } from 'vue';
+import { Store } from 'vuex';
+import { isDuplicated } from '@/utils/array';
+import useForm from '@/components/ui/form/useForm';
+import useProjectService from '@/services/project/projectService';
+import { getDefaultFormComponentData } from '@/utils/form';
+import {
+ FORM_FIELD_TYPE_INPUT,
+ FORM_FIELD_TYPE_INPUT_TEXTAREA,
+} from '@/constants/form';
+import { translate } from '@/utils/i18n';
+
+// form component data
+const formComponentData = getDefaultFormComponentData();
+
+const useProject = (store: Store) => {
+ // store
+ const ns = 'project';
+ const state = store.state[ns];
+
+ // form rules
+ const formRules: FormRules = {};
+
+ // all project select options
+ const allProjectSelectOptions = computed(() =>
+ state.allList.map(d => {
+ return {
+ label: d.name,
+ value: d._id,
+ };
+ })
+ );
+
+ return {
+ ...useForm(
+ 'project',
+ store,
+ useProjectService(store),
+ formComponentData
+ ),
+ formRules,
+ allProjectSelectOptions,
+ };
+};
+
+export default useProject;
diff --git a/frontend/crawlab-ui/src/components/core/result/ResultCell.vue b/frontend/crawlab-ui/src/components/core/result/ResultCell.vue
new file mode 100644
index 00000000..ec2f8e0a
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/result/ResultCell.vue
@@ -0,0 +1,169 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ {{ value }}
+
+
+
+
+ {{ value }}
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/result/ResultCellDialog.vue b/frontend/crawlab-ui/src/components/core/result/ResultCellDialog.vue
new file mode 100644
index 00000000..b3b063a7
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/result/ResultCellDialog.vue
@@ -0,0 +1,43 @@
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/result/ResultDedupFieldsDialog.vue b/frontend/crawlab-ui/src/components/core/result/ResultDedupFieldsDialog.vue
new file mode 100644
index 00000000..88eacd25
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/result/ResultDedupFieldsDialog.vue
@@ -0,0 +1,62 @@
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/role/CreateEditRoleDialog.vue b/frontend/crawlab-ui/src/components/core/role/CreateEditRoleDialog.vue
new file mode 100644
index 00000000..9a4843c7
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/role/CreateEditRoleDialog.vue
@@ -0,0 +1,31 @@
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/role/RoleForm.vue b/frontend/crawlab-ui/src/components/core/role/RoleForm.vue
new file mode 100644
index 00000000..7f275fd2
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/role/RoleForm.vue
@@ -0,0 +1,57 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/role/useRole.ts b/frontend/crawlab-ui/src/components/core/role/useRole.ts
new file mode 100644
index 00000000..f710b836
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/role/useRole.ts
@@ -0,0 +1,20 @@
+import { Store } from 'vuex';
+import useForm from '@/components/ui/form/useForm';
+import useRoleService from '@/services/role/roleService';
+import { getDefaultFormComponentData } from '@/utils/form';
+import { computed } from 'vue';
+import { getRouteSelectOptions } from '@/utils';
+
+// form component data
+const formComponentData = getDefaultFormComponentData();
+
+const useRole = (store: Store) => {
+ const routesOptions = computed(() => getRouteSelectOptions());
+
+ return {
+ ...useForm('role', store, useRoleService(store), formComponentData),
+ routesOptions,
+ };
+};
+
+export default useRole;
diff --git a/frontend/crawlab-ui/src/components/core/schedule/CreateEditScheduleDialog.vue b/frontend/crawlab-ui/src/components/core/schedule/CreateEditScheduleDialog.vue
new file mode 100644
index 00000000..92fa49f1
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/schedule/CreateEditScheduleDialog.vue
@@ -0,0 +1,34 @@
+
+
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/schedule/RunScheduleDialog.vue b/frontend/crawlab-ui/src/components/core/schedule/RunScheduleDialog.vue
new file mode 100644
index 00000000..f8df4c93
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/schedule/RunScheduleDialog.vue
@@ -0,0 +1,7 @@
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/schedule/ScheduleCron.vue b/frontend/crawlab-ui/src/components/core/schedule/ScheduleCron.vue
new file mode 100644
index 00000000..7b57fac3
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/schedule/ScheduleCron.vue
@@ -0,0 +1,175 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ {{ isValid ? description : t('common.status.invalid') }}
+
+
+
+
+
+
+
+
+
+ {{ isValid ? next : t('common.status.invalid') }}
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/schedule/ScheduleForm.vue b/frontend/crawlab-ui/src/components/core/schedule/ScheduleForm.vue
new file mode 100644
index 00000000..ca23946f
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/schedule/ScheduleForm.vue
@@ -0,0 +1,229 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ {{ n.name }}
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/schedule/useSchedule.ts b/frontend/crawlab-ui/src/components/core/schedule/useSchedule.ts
new file mode 100644
index 00000000..42d84134
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/schedule/useSchedule.ts
@@ -0,0 +1,94 @@
+import { computed, watch } from 'vue';
+import { Store } from 'vuex';
+import useForm from '@/components/ui/form/useForm';
+import useScheduleService from '@/services/schedule/scheduleService';
+import { getDefaultFormComponentData } from '@/utils/form';
+import { parseExpression } from 'cron-parser';
+import { getModeOptions } from '@/utils/task';
+import useSpider from '@/components/core/spider/useSpider';
+import { translate } from '@/utils/i18n';
+import useScheduleDetail from '@/views/schedule/detail/useScheduleDetail';
+
+// i18n
+const t = translate;
+
+// form component data
+const formComponentData = getDefaultFormComponentData();
+
+const useSchedule = (store: Store) => {
+ // store
+ const ns = 'schedule';
+ const state = store.state[ns];
+
+ const { allDict: allSpiderDict } = useSpider(store);
+
+ // form
+ const form = computed(() => state.form);
+
+ // options for default mode
+ const modeOptions = getModeOptions();
+
+ // form rules
+ const formRules: FormRules = {
+ cron: {
+ trigger: 'blur',
+ validator: (_, value: string, callback) => {
+ const invalidMessage = t(
+ 'components.schedule.rules.message.invalidCronExpression'
+ );
+ if (!value) return callback(invalidMessage);
+ if (value.trim().split(' ').length != 5)
+ return callback(invalidMessage);
+ try {
+ parseExpression(value);
+ callback();
+ } catch (e: any) {
+ callback(e.message);
+ }
+ },
+ },
+ };
+
+ // all schedule select options
+ const allScheduleSelectOptions = computed(() =>
+ state.allList.map(d => {
+ return {
+ label: d.name,
+ value: d._id,
+ };
+ })
+ );
+
+ const { activeId } = useScheduleDetail();
+
+ watch(
+ () => form.value?.spider_id,
+ () => {
+ if (activeId.value) return;
+ if (!form.value?.spider_id) return;
+ const spider = allSpiderDict.value.get(form.value?.spider_id);
+ if (!spider) return;
+ const payload = { ...form.value } as Schedule;
+ if (spider.cmd) payload.cmd = spider.cmd;
+ if (spider.param) payload.param = spider.param;
+ if (spider.mode) payload.mode = spider.mode;
+ if (spider.node_ids?.length) payload.node_ids = spider.node_ids;
+ if (spider.node_tags?.length) payload.node_tags = spider.node_tags;
+ store.commit(`${ns}/setForm`, payload);
+ }
+ );
+
+ return {
+ ...useForm(
+ 'schedule',
+ store,
+ useScheduleService(store),
+ formComponentData
+ ),
+ modeOptions,
+ formRules,
+ allScheduleSelectOptions,
+ };
+};
+
+export default useSchedule;
diff --git a/frontend/crawlab-ui/src/components/core/spider/CreateEditSpiderDialog.vue b/frontend/crawlab-ui/src/components/core/spider/CreateEditSpiderDialog.vue
new file mode 100644
index 00000000..768be7e6
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/spider/CreateEditSpiderDialog.vue
@@ -0,0 +1,32 @@
+
+
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/spider/RunSpiderDialog.vue b/frontend/crawlab-ui/src/components/core/spider/RunSpiderDialog.vue
new file mode 100644
index 00000000..746ad244
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/spider/RunSpiderDialog.vue
@@ -0,0 +1,197 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ {{ n.name }}
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/spider/SpiderForm.vue b/frontend/crawlab-ui/src/components/core/spider/SpiderForm.vue
new file mode 100644
index 00000000..608086c2
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/spider/SpiderForm.vue
@@ -0,0 +1,347 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ {{ op.label }}
+
+
+
+
+
+
+ {{ activeTemplateOption?.label }}
+
+
+
+
+
+ {{ activeTemplateOption?.doc_label }}
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ {{ n.name }}
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/spider/SpiderResultDataWithDatabase.vue b/frontend/crawlab-ui/src/components/core/spider/SpiderResultDataWithDatabase.vue
new file mode 100644
index 00000000..fd66d9fd
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/spider/SpiderResultDataWithDatabase.vue
@@ -0,0 +1,120 @@
+
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/spider/SpiderStat.vue b/frontend/crawlab-ui/src/components/core/spider/SpiderStat.vue
new file mode 100644
index 00000000..6f3ab4b3
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/spider/SpiderStat.vue
@@ -0,0 +1,106 @@
+
+
+
+
+
$emit('tasks-click')"
+ />
+ $emit('results-click')"
+ />
+ $emit('duration-click')"
+ >
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/spider/UploadSpiderFilesDialog.vue b/frontend/crawlab-ui/src/components/core/spider/UploadSpiderFilesDialog.vue
new file mode 100644
index 00000000..c705c719
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/spider/UploadSpiderFilesDialog.vue
@@ -0,0 +1,24 @@
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/spider/useSpider.ts b/frontend/crawlab-ui/src/components/core/spider/useSpider.ts
new file mode 100644
index 00000000..73b3cd82
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/spider/useSpider.ts
@@ -0,0 +1,37 @@
+import { useRoute } from 'vue-router';
+import { computed } from 'vue';
+import { Store } from 'vuex';
+import useForm from '@/components/ui/form/useForm';
+import useSpiderService from '@/services/spider/spiderService';
+import { getDefaultFormComponentData } from '@/utils/form';
+import useRequest from '@/services/request';
+import { FILTER_OP_CONTAINS } from '@/constants/filter';
+import { getModeOptions } from '@/utils/task';
+import { translate } from '@/utils/i18n';
+
+// form component data
+const formComponentData = getDefaultFormComponentData();
+
+const useSpider = (store: Store) => {
+ // options for default mode
+ const modeOptions = getModeOptions();
+
+ // route
+ const route = useRoute();
+
+ // spider id
+ const id = computed(() => route.params.id);
+
+ return {
+ ...useForm(
+ 'spider',
+ store,
+ useSpiderService(store),
+ formComponentData
+ ),
+ id,
+ modeOptions,
+ };
+};
+
+export default useSpider;
diff --git a/frontend/crawlab-ui/src/components/core/task/CreateTaskDialog.vue b/frontend/crawlab-ui/src/components/core/task/CreateTaskDialog.vue
new file mode 100644
index 00000000..1d8c114f
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/task/CreateTaskDialog.vue
@@ -0,0 +1,49 @@
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/task/TaskCommand.vue b/frontend/crawlab-ui/src/components/core/task/TaskCommand.vue
new file mode 100644
index 00000000..696b6816
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/task/TaskCommand.vue
@@ -0,0 +1,81 @@
+
+
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/task/TaskForm.vue b/frontend/crawlab-ui/src/components/core/task/TaskForm.vue
new file mode 100644
index 00000000..487c6e59
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/task/TaskForm.vue
@@ -0,0 +1,337 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ {{ n.name }}
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/task/TaskMode.vue b/frontend/crawlab-ui/src/components/core/task/TaskMode.vue
new file mode 100644
index 00000000..3279ca99
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/task/TaskMode.vue
@@ -0,0 +1,95 @@
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/task/TaskPriority.vue b/frontend/crawlab-ui/src/components/core/task/TaskPriority.vue
new file mode 100644
index 00000000..38f2f140
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/task/TaskPriority.vue
@@ -0,0 +1,71 @@
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/task/TaskResultDataWithDatabase.vue b/frontend/crawlab-ui/src/components/core/task/TaskResultDataWithDatabase.vue
new file mode 100644
index 00000000..08ff8251
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/task/TaskResultDataWithDatabase.vue
@@ -0,0 +1,117 @@
+
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/task/TaskResults.vue b/frontend/crawlab-ui/src/components/core/task/TaskResults.vue
new file mode 100644
index 00000000..f9223211
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/task/TaskResults.vue
@@ -0,0 +1,83 @@
+
+
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/task/TaskStatus.vue b/frontend/crawlab-ui/src/components/core/task/TaskStatus.vue
new file mode 100644
index 00000000..59381606
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/task/TaskStatus.vue
@@ -0,0 +1,111 @@
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/task/useTask.ts b/frontend/crawlab-ui/src/components/core/task/useTask.ts
new file mode 100644
index 00000000..31fe9200
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/task/useTask.ts
@@ -0,0 +1,56 @@
+import { useRoute } from 'vue-router';
+import { computed } from 'vue';
+import { Store } from 'vuex';
+import useForm from '@/components/ui/form/useForm';
+import useTaskService from '@/services/task/taskService';
+import { getDefaultFormComponentData } from '@/utils/form';
+import useSpider from '@/components/core/spider/useSpider';
+import {
+ getModeOptions,
+ getModeOptionsDict,
+ getPriorityLabel,
+} from '@/utils/task';
+import { formatTimeAgo } from '@/utils/time';
+
+// form component data
+const formComponentData = getDefaultFormComponentData();
+
+const useTask = (store: Store) => {
+ const ns = 'task' as ListStoreNamespace;
+ const { task: state } = store.state as RootStoreState;
+
+ // options for default mode
+ const modeOptions = getModeOptions();
+ const modeOptionsDict = computed(() => getModeOptionsDict());
+
+ const { allDict: allSpiderDict } = useSpider(store);
+
+ // route
+ const route = useRoute();
+
+ // task id
+ const id = computed(() => route.params.id);
+
+ const allListSelectOptions = computed(() =>
+ state.allList.map(task => {
+ const spider = allSpiderDict.value.get(task.spider_id!);
+ const timeAgo = formatTimeAgo(task.created_ts!);
+ return {
+ label: `${spider?.name} (${timeAgo})`,
+ value: task._id,
+ };
+ })
+ );
+
+ return {
+ ...useForm('task', store, useTaskService(store), formComponentData),
+ allSpiderDict,
+ id,
+ modeOptions,
+ modeOptionsDict,
+ getPriorityLabel,
+ allListSelectOptions,
+ };
+};
+
+export default useTask;
diff --git a/frontend/crawlab-ui/src/components/core/user/CreateEditUserDialog.vue b/frontend/crawlab-ui/src/components/core/user/CreateEditUserDialog.vue
new file mode 100644
index 00000000..12f5dff6
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/user/CreateEditUserDialog.vue
@@ -0,0 +1,33 @@
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/user/UserForm.vue b/frontend/crawlab-ui/src/components/core/user/UserForm.vue
new file mode 100644
index 00000000..47a5ad09
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/user/UserForm.vue
@@ -0,0 +1,201 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/user/UserRole.vue b/frontend/crawlab-ui/src/components/core/user/UserRole.vue
new file mode 100644
index 00000000..e1aa6636
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/user/UserRole.vue
@@ -0,0 +1,43 @@
+
+
+
+
+
+ {{ computedLabel }}
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/core/user/useUser.ts b/frontend/crawlab-ui/src/components/core/user/useUser.ts
new file mode 100644
index 00000000..37c5a0f0
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/core/user/useUser.ts
@@ -0,0 +1,87 @@
+import { computed } from 'vue';
+import { Store } from 'vuex';
+import useForm from '@/components/ui/form/useForm';
+import useUserService from '@/services/user/userService';
+import { getDefaultFormComponentData } from '@/utils/form';
+import { getModeOptions } from '@/utils/task';
+import { ROLE_ADMIN, ROLE_NORMAL } from '@/constants/user';
+import { ElMessage, ElMessageBox } from 'element-plus';
+import { translate } from '@/utils/i18n';
+
+// i18n
+const t = translate;
+
+// form component data
+const formComponentData = getDefaultFormComponentData();
+
+const useUser = (store: Store) => {
+ // store
+ const ns = 'user';
+ const state = store.state[ns];
+
+ // options for default mode
+ const modeOptions = getModeOptions();
+
+ // form rules
+ const formRules: FormRules = {
+ password: {
+ trigger: 'blur',
+ validator: (_, value: string, callback) => {
+ const invalidMessage =
+ 'Invalid password. Length must be no less than 5.';
+ if (0 < value.length && value.length < 5)
+ return callback(invalidMessage);
+ return callback();
+ },
+ },
+ };
+
+ // all user select options
+ const allUserSelectOptions = computed(() =>
+ state.allList.map(d => {
+ return {
+ label: d.username,
+ value: d._id,
+ };
+ })
+ );
+
+ // on change password
+ const onChangePasswordFunc = async (id?: string) => {
+ if (!id) return;
+
+ const { value } = await ElMessageBox.prompt(
+ t('components.user.messageBox.prompt.changePassword'),
+ t('components.user.form.changePassword'),
+ {
+ inputType: 'password',
+ inputPlaceholder: t('components.user.form.newPassword'),
+ inputValidator: (value: string) => {
+ return value?.length < 5
+ ? t('components.user.rules.invalidPassword')
+ : true;
+ },
+ confirmButtonClass: 'edit-user-password-confirm-btn',
+ cancelButtonClass: 'edit-user-password-cancel-btn',
+ }
+ );
+ await store.dispatch(`${ns}/changePassword`, { id, password: value });
+ ElMessage.success(t('common.message.success.save'));
+ };
+
+ const rolesOptions: SelectOption[] = [
+ { label: t('components.user.role.admin'), value: ROLE_ADMIN },
+ { label: t('components.user.role.normal'), value: ROLE_NORMAL },
+ ];
+
+ return {
+ ...useForm('user', store, useUserService(store), formComponentData),
+ modeOptions,
+ formRules,
+ allUserSelectOptions,
+ onChangePasswordFunc,
+ rolesOptions,
+ };
+};
+
+export default useUser;
diff --git a/frontend/crawlab-ui/src/components/index.ts b/frontend/crawlab-ui/src/components/index.ts
new file mode 100644
index 00000000..e85ecaf3
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/index.ts
@@ -0,0 +1,487 @@
+import * as action from './ui/table/action';
+import * as autoLink from './ui/lexical/utils/autoLink';
+import * as column from './ui/table/column';
+import * as data from './ui/table/data';
+import * as date from './ui/date/date';
+import * as fileEditorDropZone from './ui/file/fileEditorDropZone';
+import * as filter from './ui/filter/filter';
+import * as formTable from './ui/form/formTable';
+import * as getSelectedNode from './ui/lexical/utils/getSelectedNode';
+import * as header from './ui/table/header';
+import * as icon from './ui/icon/icon';
+import * as ImageNode from './ui/lexical/nodes/ImageNode';
+import * as LexicalDecoratedTeleports from './ui/lexical/plugins/LexicalDecoratedTeleports';
+import * as markdownTransformers from './ui/lexical/utils/markdownTransformers';
+import * as node from './ui/lexical/utils/node';
+import * as pagination from './ui/table/pagination';
+import * as store from './ui/table/store';
+import * as theme from './ui/lexical/utils/theme';
+import * as VariableNode from './ui/lexical/nodes/VariableNode';
+import AssistantConsole from './core/ai/AssistantConsole.vue';
+import AtomMaterialIcon from './ui/icon/AtomMaterialIcon.vue';
+import BlockOptionsDropdownList from './ui/lexical/components/BlockOptionsDropdownList.vue';
+import Box from './ui/box/Box.vue';
+import Button from './ui/button/Button.vue';
+import ButtonGroup from './ui/button/ButtonGroup.vue';
+import Chart from './ui/chart/Chart.vue';
+import ChatConfigDialog from './ui/chat/ChatConfigDialog.vue';
+import ChatHistory from './ui/chat/ChatHistory.vue';
+import ChatInput from './ui/chat/ChatInput.vue';
+import ChatMessage from './ui/chat/ChatMessage.vue';
+import ChatMessageAction from './ui/chat/ChatMessageAction.vue';
+import ChatMessageList from './ui/chat/ChatMessageList.vue';
+import ChatSidebar from './ui/chat/ChatSidebar.vue';
+import CheckboxTree from './ui/checkbox/CheckboxTree.vue';
+import CheckboxTreeGroup from './ui/checkbox/CheckboxTreeGroup.vue';
+import CheckTag from './ui/tag/CheckTag.vue';
+import CheckTagGroup from './ui/tag/CheckTagGroup.vue';
+import ConfirmDialog from './ui/dialog/ConfirmDialog.vue';
+import ContextMenu from './ui/context-menu/ContextMenu.vue';
+import ContextMenuList from './ui/context-menu/ContextMenuList.vue';
+import CreateEditDatabaseDialog from './core/database/CreateEditDatabaseDialog.vue';
+import CreateEditDatabaseTableDialog from './core/database/CreateEditDatabaseTableDialog.vue';
+import CreateEditDialog from './ui/dialog/CreateEditDialog.vue';
+import CreateEditEnvironmentDialog from './core/environment/CreateEditEnvironmentDialog.vue';
+import CreateEditGitDialog from './core/git/CreateEditGitDialog.vue';
+import CreateEditNodeDialog from './core/node/CreateEditNodeDialog.vue';
+import CreateEditNotificationAlertDialog from './core/notification/alert/CreateEditNotificationAlertDialog.vue';
+import CreateEditNotificationChannelDialog from './core/notification/channel/CreateEditNotificationChannelDialog.vue';
+import CreateEditNotificationSettingDialog from './core/notification/setting/CreateEditNotificationSettingDialog.vue';
+import CreateEditProjectDialog from './core/project/CreateEditProjectDialog.vue';
+import CreateEditRoleDialog from './core/role/CreateEditRoleDialog.vue';
+import CreateEditScheduleDialog from './core/schedule/CreateEditScheduleDialog.vue';
+import CreateEditSpiderDialog from './core/spider/CreateEditSpiderDialog.vue';
+import CreateEditUserDialog from './core/user/CreateEditUserDialog.vue';
+import CreateGitBranchDialog from './core/git/CreateGitBranchDialog.vue';
+import CreateGitSpiderDialog from './core/git/CreateGitSpiderDialog.vue';
+import CreateTaskDialog from './core/task/CreateTaskDialog.vue';
+import CurrentMetrics from './core/metric/CurrentMetrics.vue';
+import DatabaseDatabaseDetail from './core/database/DatabaseDatabaseDetail.vue';
+import DatabaseDataSource from './core/database/DatabaseDataSource.vue';
+import DatabaseForm from './core/database/DatabaseForm.vue';
+import DatabaseNavTabs from './core/database/nav/DatabaseNavTabs.vue';
+import DatabaseSidebar from './core/database/DatabaseSidebar.vue';
+import DatabaseStatus from './core/database/DatabaseStatus.vue';
+import DatabaseTableDetail from './core/database/DatabaseTableDetail.vue';
+import DatabaseTableDetailColumns from './core/database/tables/DatabaseTableDetailColumns.vue';
+import DatabaseTableDetailData from './core/database/tables/DatabaseTableDetailData.vue';
+import DatabaseTableDetailIndexes from './core/database/tables/DatabaseTableDetailIndexes.vue';
+import DateRangePicker from './ui/date/DateRangePicker.vue';
+import DateTimeRangePicker from './ui/date/DateTimeRangePicker.vue';
+import DependencyConfigDialog from './core/dependency/DependencyConfigDialog.vue';
+import DependencyInstallDialog from './core/dependency/DependencyInstallDialog.vue';
+import DependencyLogsDialog from './core/dependency/DependencyLogsDialog.vue';
+import DependencySetupDialog from './core/dependency/DependencySetupDialog.vue';
+import DependencyStatusTag from './core/dependency/DependencyStatusTag.vue';
+import DependencyUninstallDialog from './core/dependency/DependencyUninstallDialog.vue';
+import DependencyVersions from './core/dependency/DependencyVersions.vue';
+import DetailTabList from './ui/list/DetailTabList.vue';
+import Dialog from './ui/dialog/Dialog.vue';
+import DraggableItem from './ui/drag/DraggableItem.vue';
+import DraggableList from './ui/drag/DraggableList.vue';
+import DropdownButton from './ui/lexical/components/DropdownButton.vue';
+import Duration from './ui/time/Duration.vue';
+import EditInput from './ui/input/EditInput.vue';
+import EditorOptionsDropdownList from './ui/editor/EditorOptionsDropdownList.vue';
+import EditTable from './ui/table/EditTable.vue';
+import EditTableActionCell from './ui/table/EditTableActionCell.vue';
+import Empty from './ui/empty/Empty.vue';
+import EnvironmentForm from './core/environment/EnvironmentForm.vue';
+import ExportButton from './ui/button/ExportButton.vue';
+import ExportForm from './ui/export/ExportForm.vue';
+import FaIconButton from './ui/button/FaIconButton.vue';
+import FileActions from './ui/file/FileActions.vue';
+import FileDiff from './ui/file/FileDiff.vue';
+import FileEditor from './ui/file/FileEditor.vue';
+import FileEditorCreateWithAiDialog from './ui/file/FileEditorCreateWithAiDialog.vue';
+import FileEditorNavMenu from './ui/file/FileEditorNavMenu.vue';
+import FileEditorNavMenuContextMenu from './ui/file/FileEditorNavMenuContextMenu.vue';
+import FileEditorNavTabs from './ui/file/FileEditorNavTabs.vue';
+import FileEditorNavTabsContextMenu from './ui/file/FileEditorNavTabsContextMenu.vue';
+import FileEditorNavTabsShowMoreContextMenu from './ui/file/FileEditorNavTabsShowMoreContextMenu.vue';
+import FileEditorSettingsDialog from './ui/file/FileEditorSettingsDialog.vue';
+import FileTab from './ui/file/FileTab.vue';
+import FileUpload from './ui/file/FileUpload.vue';
+import FilterCondition from './ui/filter/FilterCondition.vue';
+import FilterConditionList from './ui/filter/FilterConditionList.vue';
+import FilterInput from './ui/filter/FilterInput.vue';
+import FilterSelect from './ui/filter/FilterSelect.vue';
+import FloatLinkEditor from './ui/lexical/components/FloatLinkEditor.vue';
+import Form from './ui/form/Form.vue';
+import FormItem from './ui/form/FormItem.vue';
+import FormTableField from './ui/form/FormTableField.vue';
+import GitBranchSelect from './core/git/GitBranchSelect.vue';
+import GitCloneLogsDialog from './core/git/GitCloneLogsDialog.vue';
+import GitFileDiffDialog from './core/git/GitFileDiffDialog.vue';
+import GitFileStatus from './core/git/GitFileStatus.vue';
+import GitForm from './core/git/GitForm.vue';
+import GitHubStarBadge from './ui/badge/GitHubStarBadge.vue';
+import GitLogsBox from './core/git/GitLogsBox.vue';
+import GitPath from './core/git/GitPath.vue';
+import GitRepo from './core/git/GitRepo.vue';
+import GitStatus from './core/git/GitStatus.vue';
+import Icon from './ui/icon/Icon.vue';
+import IconButton from './ui/button/IconButton.vue';
+import ImageComponent from './ui/lexical/components/ImageComponent.vue';
+import ImgEmpty from './ui/empty/ImgEmpty.vue';
+import InputList from './ui/input/InputList.vue';
+import InputSelect from './ui/select/InputSelect.vue';
+import InsertImageDialog from './ui/lexical/components/InsertImageDialog.vue';
+import InsertOptionsDropdownList from './ui/lexical/components/InsertOptionsDropdownList.vue';
+import InsertTableDialog from './ui/lexical/components/InsertTableDialog.vue';
+import InsertVariableDialog from './ui/lexical/components/InsertVariableDialog.vue';
+import LabelButton from './ui/button/LabelButton.vue';
+import LexicalAutoFocusPlugin from './ui/lexical/plugins/LexicalAutoFocusPlugin.vue';
+import LexicalAutoLinkPlugin from './ui/lexical/plugins/LexicalAutoLinkPlugin.vue';
+import LexicalClickableLinkPlugin from './ui/lexical/plugins/LexicalClickableLinkPlugin.vue';
+import LexicalContentEditable from './ui/lexical/plugins/LexicalContentEditable.vue';
+import LexicalEditor from './ui/lexical/LexicalEditor.vue';
+import LexicalImagePlugin from './ui/lexical/plugins/LexicalImagePlugin.vue';
+import LexicalLinkPlugin from './ui/lexical/plugins/LexicalLinkPlugin.vue';
+import LexicalListPlugin from './ui/lexical/plugins/LexicalListPlugin.vue';
+import LexicalRichTextPlugin from './ui/lexical/plugins/LexicalRichTextPlugin.vue';
+import LexicalTablePlugin from './ui/lexical/plugins/LexicalTablePlugin.vue';
+import LexicalToolbarPlugin from './ui/lexical/plugins/LexicalToolbarPlugin.vue';
+import LexicalVariablePlugin from './ui/lexical/plugins/LexicalVariablePlugin.vue';
+import LinkTag from './ui/tag/LinkTag.vue';
+import LoadingText from './ui/loading/LoadingText.vue';
+import LogsView from './ui/logs/LogsView.vue';
+import MarkdownEditor from './ui/markdown/MarkdownEditor.vue';
+import MarkdownEditorToolbar from './ui/markdown/MarkdownEditorToolbar.vue';
+import MenuItemIcon from './ui/icon/MenuItemIcon.vue';
+import Metric from './ui/chart/Metric.vue';
+import MetricMonitoringDetail from './core/metric/MetricMonitoringDetail.vue';
+import NavActionBack from './ui/nav/NavActionBack.vue';
+import NavActionButton from './ui/nav/NavActionButton.vue';
+import NavActionFaIcon from './ui/nav/NavActionFaIcon.vue';
+import NavActionGroup from './ui/nav/NavActionGroup.vue';
+import NavActionGroupDetailCommon from './ui/nav/NavActionGroupDetailCommon.vue';
+import NavActionItem from './ui/nav/NavActionItem.vue';
+import NavActions from './ui/nav/NavActions.vue';
+import NavLink from './ui/nav/NavLink.vue';
+import NavSidebar from './ui/nav/NavSidebar.vue';
+import NavSidebarList from './ui/nav/NavSidebarList.vue';
+import NavSidebarTree from './ui/nav/NavSidebarTree.vue';
+import NavTabs from './ui/nav/NavTabs.vue';
+import NodeActive from './core/node/NodeActive.vue';
+import NodeForm from './core/node/NodeForm.vue';
+import NodeRunners from './core/node/NodeRunners.vue';
+import NodeStatus from './core/node/NodeStatus.vue';
+import NodeTag from './core/node/NodeTag.vue';
+import NodeType from './core/node/NodeType.vue';
+import NotificationAlertForm from './core/notification/alert/NotificationAlertForm.vue';
+import NotificationChannelForm from './core/notification/channel/NotificationChannelForm.vue';
+import NotificationRequestStatus from './core/notification/request/NotificationRequestStatus.vue';
+import NotificationSettingForm from './core/notification/setting/NotificationSettingForm.vue';
+import NotificationSettingTriggerSelect from './core/notification/setting/NotificationSettingTriggerSelect.vue';
+import Option from './ui/select/Option.vue';
+import ProjectForm from './core/project/ProjectForm.vue';
+import RangePicker from './ui/date/RangePicker.vue';
+import ResizeHandle from './ui/resize/ResizeHandle.vue';
+import ResultCell from './core/result/ResultCell.vue';
+import ResultCellDialog from './core/result/ResultCellDialog.vue';
+import ResultDedupFieldsDialog from './core/result/ResultDedupFieldsDialog.vue';
+import RoleForm from './core/role/RoleForm.vue';
+import RunScheduleDialog from './core/schedule/RunScheduleDialog.vue';
+import RunSpiderDialog from './core/spider/RunSpiderDialog.vue';
+import ScheduleCron from './core/schedule/ScheduleCron.vue';
+import ScheduleForm from './core/schedule/ScheduleForm.vue';
+import Select from './ui/select/Select.vue';
+import SpiderForm from './core/spider/SpiderForm.vue';
+import SpiderResultDataWithDatabase from './core/spider/SpiderResultDataWithDatabase.vue';
+import SpiderStat from './core/spider/SpiderStat.vue';
+import Switch from './ui/switch/Switch.vue';
+import Table from './ui/table/Table.vue';
+import TableActions from './ui/table/TableActions.vue';
+import TableCell from './ui/table/TableCell.vue';
+import TableColumnsTransfer from './ui/table/TableColumnsTransfer.vue';
+import TableEditCell from './ui/table/TableEditCell.vue';
+import TableHeader from './ui/table/TableHeader.vue';
+import TableHeaderAction from './ui/table/TableHeaderAction.vue';
+import TableHeaderDialog from './ui/table/TableHeaderDialog.vue';
+import TableHeaderDialogFilter from './ui/table/TableHeaderDialogFilter.vue';
+import TableHeaderDialogSort from './ui/table/TableHeaderDialogSort.vue';
+import Tag from './ui/tag/Tag.vue';
+import TaskCommand from './core/task/TaskCommand.vue';
+import TaskForm from './core/task/TaskForm.vue';
+import TaskMode from './core/task/TaskMode.vue';
+import TaskPriority from './core/task/TaskPriority.vue';
+import TaskResultDataWithDatabase from './core/task/TaskResultDataWithDatabase.vue';
+import TaskResults from './core/task/TaskResults.vue';
+import TaskStatus from './core/task/TaskStatus.vue';
+import Time from './ui/time/Time.vue';
+import Tip from './ui/tip/Tip.vue';
+import Transfer from './ui/transfer/Transfer.vue';
+import TransferPanel from './ui/transfer/TransferPanel.vue';
+import UploadFilesDialog from './ui/file/UploadFilesDialog.vue';
+import UploadGitFilesDialog from './core/git/UploadGitFilesDialog.vue';
+import UploadSpiderFilesDialog from './core/spider/UploadSpiderFilesDialog.vue';
+import useCanShowPlaceholder from './ui/lexical/composables/useCanShowPlaceholder';
+import useDatabase from './core/database/useDatabase';
+import useDecorators from './ui/lexical/composables/useDecorators';
+import useEnvironment from './core/environment/useEnvironment';
+import useForm from './ui/form/useForm';
+import useGit from './core/git/useGit';
+import useLexicalEffect from './ui/lexical/composables/useLexicalEffect';
+import useLexicalList from './ui/lexical/composables/useLexicalList';
+import useLexicalMounted from './ui/lexical/composables/useLexicalMounted';
+import useNode from './core/node/useNode';
+import useNotificationAlert from './core/notification/alert/useNotificationAlert';
+import useNotificationChannel from './core/notification/channel/useNotificationChannel';
+import useNotificationSetting from './core/notification/setting/useNotificationSetting';
+import useProject from './core/project/useProject';
+import UserAvatar from './ui/avatar/UserAvatar.vue';
+import UserForm from './core/user/UserForm.vue';
+import useRichTextSetup from './ui/lexical/composables/useRichTextSetup';
+import useRole from './core/role/useRole';
+import UserRole from './core/user/UserRole.vue';
+import useSchedule from './core/schedule/useSchedule';
+import useSpider from './core/spider/useSpider';
+import useTask from './core/task/useTask';
+import useUser from './core/user/useUser';
+import useVariableSetup from './ui/lexical/composables/useVariableSetup';
+
+export {
+ action as action,
+ autoLink as autoLink,
+ column as column,
+ data as data,
+ date as date,
+ fileEditorDropZone as fileEditorDropZone,
+ filter as filter,
+ formTable as formTable,
+ getSelectedNode as getSelectedNode,
+ header as header,
+ icon as icon,
+ ImageNode as ImageNode,
+ LexicalDecoratedTeleports as LexicalDecoratedTeleports,
+ markdownTransformers as markdownTransformers,
+ node as node,
+ pagination as pagination,
+ store as store,
+ theme as theme,
+ VariableNode as VariableNode,
+ AssistantConsole as ClAssistantConsole,
+ AtomMaterialIcon as ClAtomMaterialIcon,
+ BlockOptionsDropdownList as ClBlockOptionsDropdownList,
+ Box as ClBox,
+ Button as ClButton,
+ ButtonGroup as ClButtonGroup,
+ Chart as ClChart,
+ ChatConfigDialog as ClChatConfigDialog,
+ ChatHistory as ClChatHistory,
+ ChatInput as ClChatInput,
+ ChatMessage as ClChatMessage,
+ ChatMessageAction as ClChatMessageAction,
+ ChatMessageList as ClChatMessageList,
+ ChatSidebar as ClChatSidebar,
+ CheckboxTree as ClCheckboxTree,
+ CheckboxTreeGroup as ClCheckboxTreeGroup,
+ CheckTag as ClCheckTag,
+ CheckTagGroup as ClCheckTagGroup,
+ ConfirmDialog as ClConfirmDialog,
+ ContextMenu as ClContextMenu,
+ ContextMenuList as ClContextMenuList,
+ CreateEditDatabaseDialog as ClCreateEditDatabaseDialog,
+ CreateEditDatabaseTableDialog as ClCreateEditDatabaseTableDialog,
+ CreateEditDialog as ClCreateEditDialog,
+ CreateEditEnvironmentDialog as ClCreateEditEnvironmentDialog,
+ CreateEditGitDialog as ClCreateEditGitDialog,
+ CreateEditNodeDialog as ClCreateEditNodeDialog,
+ CreateEditNotificationAlertDialog as ClCreateEditNotificationAlertDialog,
+ CreateEditNotificationChannelDialog as ClCreateEditNotificationChannelDialog,
+ CreateEditNotificationSettingDialog as ClCreateEditNotificationSettingDialog,
+ CreateEditProjectDialog as ClCreateEditProjectDialog,
+ CreateEditRoleDialog as ClCreateEditRoleDialog,
+ CreateEditScheduleDialog as ClCreateEditScheduleDialog,
+ CreateEditSpiderDialog as ClCreateEditSpiderDialog,
+ CreateEditUserDialog as ClCreateEditUserDialog,
+ CreateGitBranchDialog as ClCreateGitBranchDialog,
+ CreateGitSpiderDialog as ClCreateGitSpiderDialog,
+ CreateTaskDialog as ClCreateTaskDialog,
+ CurrentMetrics as ClCurrentMetrics,
+ DatabaseDatabaseDetail as ClDatabaseDatabaseDetail,
+ DatabaseDataSource as ClDatabaseDataSource,
+ DatabaseForm as ClDatabaseForm,
+ DatabaseNavTabs as ClDatabaseNavTabs,
+ DatabaseSidebar as ClDatabaseSidebar,
+ DatabaseStatus as ClDatabaseStatus,
+ DatabaseTableDetail as ClDatabaseTableDetail,
+ DatabaseTableDetailColumns as ClDatabaseTableDetailColumns,
+ DatabaseTableDetailData as ClDatabaseTableDetailData,
+ DatabaseTableDetailIndexes as ClDatabaseTableDetailIndexes,
+ DateRangePicker as ClDateRangePicker,
+ DateTimeRangePicker as ClDateTimeRangePicker,
+ DependencyConfigDialog as ClDependencyConfigDialog,
+ DependencyInstallDialog as ClDependencyInstallDialog,
+ DependencyLogsDialog as ClDependencyLogsDialog,
+ DependencySetupDialog as ClDependencySetupDialog,
+ DependencyStatusTag as ClDependencyStatusTag,
+ DependencyUninstallDialog as ClDependencyUninstallDialog,
+ DependencyVersions as ClDependencyVersions,
+ DetailTabList as ClDetailTabList,
+ Dialog as ClDialog,
+ DraggableItem as ClDraggableItem,
+ DraggableList as ClDraggableList,
+ DropdownButton as ClDropdownButton,
+ Duration as ClDuration,
+ EditInput as ClEditInput,
+ EditorOptionsDropdownList as ClEditorOptionsDropdownList,
+ EditTable as ClEditTable,
+ EditTableActionCell as ClEditTableActionCell,
+ Empty as ClEmpty,
+ EnvironmentForm as ClEnvironmentForm,
+ ExportButton as ClExportButton,
+ ExportForm as ClExportForm,
+ FaIconButton as ClFaIconButton,
+ FileActions as ClFileActions,
+ FileDiff as ClFileDiff,
+ FileEditor as ClFileEditor,
+ FileEditorCreateWithAiDialog as ClFileEditorCreateWithAiDialog,
+ FileEditorNavMenu as ClFileEditorNavMenu,
+ FileEditorNavMenuContextMenu as ClFileEditorNavMenuContextMenu,
+ FileEditorNavTabs as ClFileEditorNavTabs,
+ FileEditorNavTabsContextMenu as ClFileEditorNavTabsContextMenu,
+ FileEditorNavTabsShowMoreContextMenu as ClFileEditorNavTabsShowMoreContextMenu,
+ FileEditorSettingsDialog as ClFileEditorSettingsDialog,
+ FileTab as ClFileTab,
+ FileUpload as ClFileUpload,
+ FilterCondition as ClFilterCondition,
+ FilterConditionList as ClFilterConditionList,
+ FilterInput as ClFilterInput,
+ FilterSelect as ClFilterSelect,
+ FloatLinkEditor as ClFloatLinkEditor,
+ Form as ClForm,
+ FormItem as ClFormItem,
+ FormTableField as ClFormTableField,
+ GitBranchSelect as ClGitBranchSelect,
+ GitCloneLogsDialog as ClGitCloneLogsDialog,
+ GitFileDiffDialog as ClGitFileDiffDialog,
+ GitFileStatus as ClGitFileStatus,
+ GitForm as ClGitForm,
+ GitHubStarBadge as ClGitHubStarBadge,
+ GitLogsBox as ClGitLogsBox,
+ GitPath as ClGitPath,
+ GitRepo as ClGitRepo,
+ GitStatus as ClGitStatus,
+ Icon as ClIcon,
+ IconButton as ClIconButton,
+ ImageComponent as ClImageComponent,
+ ImgEmpty as ClImgEmpty,
+ InputList as ClInputList,
+ InputSelect as ClInputSelect,
+ InsertImageDialog as ClInsertImageDialog,
+ InsertOptionsDropdownList as ClInsertOptionsDropdownList,
+ InsertTableDialog as ClInsertTableDialog,
+ InsertVariableDialog as ClInsertVariableDialog,
+ LabelButton as ClLabelButton,
+ LexicalAutoFocusPlugin as ClLexicalAutoFocusPlugin,
+ LexicalAutoLinkPlugin as ClLexicalAutoLinkPlugin,
+ LexicalClickableLinkPlugin as ClLexicalClickableLinkPlugin,
+ LexicalContentEditable as ClLexicalContentEditable,
+ LexicalEditor as ClLexicalEditor,
+ LexicalImagePlugin as ClLexicalImagePlugin,
+ LexicalLinkPlugin as ClLexicalLinkPlugin,
+ LexicalListPlugin as ClLexicalListPlugin,
+ LexicalRichTextPlugin as ClLexicalRichTextPlugin,
+ LexicalTablePlugin as ClLexicalTablePlugin,
+ LexicalToolbarPlugin as ClLexicalToolbarPlugin,
+ LexicalVariablePlugin as ClLexicalVariablePlugin,
+ LinkTag as ClLinkTag,
+ LoadingText as ClLoadingText,
+ LogsView as ClLogsView,
+ MarkdownEditor as ClMarkdownEditor,
+ MarkdownEditorToolbar as ClMarkdownEditorToolbar,
+ MenuItemIcon as ClMenuItemIcon,
+ Metric as ClMetric,
+ MetricMonitoringDetail as ClMetricMonitoringDetail,
+ NavActionBack as ClNavActionBack,
+ NavActionButton as ClNavActionButton,
+ NavActionFaIcon as ClNavActionFaIcon,
+ NavActionGroup as ClNavActionGroup,
+ NavActionGroupDetailCommon as ClNavActionGroupDetailCommon,
+ NavActionItem as ClNavActionItem,
+ NavActions as ClNavActions,
+ NavLink as ClNavLink,
+ NavSidebar as ClNavSidebar,
+ NavSidebarList as ClNavSidebarList,
+ NavSidebarTree as ClNavSidebarTree,
+ NavTabs as ClNavTabs,
+ NodeActive as ClNodeActive,
+ NodeForm as ClNodeForm,
+ NodeRunners as ClNodeRunners,
+ NodeStatus as ClNodeStatus,
+ NodeTag as ClNodeTag,
+ NodeType as ClNodeType,
+ NotificationAlertForm as ClNotificationAlertForm,
+ NotificationChannelForm as ClNotificationChannelForm,
+ NotificationRequestStatus as ClNotificationRequestStatus,
+ NotificationSettingForm as ClNotificationSettingForm,
+ NotificationSettingTriggerSelect as ClNotificationSettingTriggerSelect,
+ Option as ClOption,
+ ProjectForm as ClProjectForm,
+ RangePicker as ClRangePicker,
+ ResizeHandle as ClResizeHandle,
+ ResultCell as ClResultCell,
+ ResultCellDialog as ClResultCellDialog,
+ ResultDedupFieldsDialog as ClResultDedupFieldsDialog,
+ RoleForm as ClRoleForm,
+ RunScheduleDialog as ClRunScheduleDialog,
+ RunSpiderDialog as ClRunSpiderDialog,
+ ScheduleCron as ClScheduleCron,
+ ScheduleForm as ClScheduleForm,
+ Select as ClSelect,
+ SpiderForm as ClSpiderForm,
+ SpiderResultDataWithDatabase as ClSpiderResultDataWithDatabase,
+ SpiderStat as ClSpiderStat,
+ Switch as ClSwitch,
+ Table as ClTable,
+ TableActions as ClTableActions,
+ TableCell as ClTableCell,
+ TableColumnsTransfer as ClTableColumnsTransfer,
+ TableEditCell as ClTableEditCell,
+ TableHeader as ClTableHeader,
+ TableHeaderAction as ClTableHeaderAction,
+ TableHeaderDialog as ClTableHeaderDialog,
+ TableHeaderDialogFilter as ClTableHeaderDialogFilter,
+ TableHeaderDialogSort as ClTableHeaderDialogSort,
+ Tag as ClTag,
+ TaskCommand as ClTaskCommand,
+ TaskForm as ClTaskForm,
+ TaskMode as ClTaskMode,
+ TaskPriority as ClTaskPriority,
+ TaskResultDataWithDatabase as ClTaskResultDataWithDatabase,
+ TaskResults as ClTaskResults,
+ TaskStatus as ClTaskStatus,
+ Time as ClTime,
+ Tip as ClTip,
+ Transfer as ClTransfer,
+ TransferPanel as ClTransferPanel,
+ UploadFilesDialog as ClUploadFilesDialog,
+ UploadGitFilesDialog as ClUploadGitFilesDialog,
+ UploadSpiderFilesDialog as ClUploadSpiderFilesDialog,
+ useCanShowPlaceholder as useCanShowPlaceholder,
+ useDatabase as useDatabase,
+ useDecorators as useDecorators,
+ useEnvironment as useEnvironment,
+ useForm as useForm,
+ useGit as useGit,
+ useLexicalEffect as useLexicalEffect,
+ useLexicalList as useLexicalList,
+ useLexicalMounted as useLexicalMounted,
+ useNode as useNode,
+ useNotificationAlert as useNotificationAlert,
+ useNotificationChannel as useNotificationChannel,
+ useNotificationSetting as useNotificationSetting,
+ useProject as useProject,
+ UserAvatar as ClUserAvatar,
+ UserForm as ClUserForm,
+ useRichTextSetup as useRichTextSetup,
+ useRole as useRole,
+ UserRole as ClUserRole,
+ useSchedule as useSchedule,
+ useSpider as useSpider,
+ useTask as useTask,
+ useUser as useUser,
+ useVariableSetup as useVariableSetup,
+};
diff --git a/frontend/crawlab-ui/src/components/ui/avatar/UserAvatar.vue b/frontend/crawlab-ui/src/components/ui/avatar/UserAvatar.vue
new file mode 100644
index 00000000..01067191
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/ui/avatar/UserAvatar.vue
@@ -0,0 +1,114 @@
+
+
+
+
+
+ emit('click', e)"
+ >
+
+
+
+ {{ userLabel }}
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/ui/badge/GitHubStarBadge.vue b/frontend/crawlab-ui/src/components/ui/badge/GitHubStarBadge.vue
new file mode 100644
index 00000000..0da131d3
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/ui/badge/GitHubStarBadge.vue
@@ -0,0 +1,41 @@
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/ui/box/Box.vue b/frontend/crawlab-ui/src/components/ui/box/Box.vue
new file mode 100644
index 00000000..6e8a3182
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/ui/box/Box.vue
@@ -0,0 +1,128 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ {{ title }}
+
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/ui/button/Button.vue b/frontend/crawlab-ui/src/components/ui/button/Button.vue
new file mode 100644
index 00000000..329b9201
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/ui/button/Button.vue
@@ -0,0 +1,57 @@
+
+
+
+
+ emit('click', event)"
+ @mouseenter="(event: Event) => emit('mouseenter', event)"
+ @mouseleave="(event: Event) => emit('mouseleave', event)"
+ >
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/ui/button/ButtonGroup.vue b/frontend/crawlab-ui/src/components/ui/button/ButtonGroup.vue
new file mode 100644
index 00000000..02c9232f
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/ui/button/ButtonGroup.vue
@@ -0,0 +1,88 @@
+
+
+
+
+
+
+
+
+
+
+
+ (visible = false)"
+ />
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/ui/button/ExportButton.vue b/frontend/crawlab-ui/src/components/ui/button/ExportButton.vue
new file mode 100644
index 00000000..eafa4c13
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/ui/button/ExportButton.vue
@@ -0,0 +1,165 @@
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/ui/button/FaIconButton.vue b/frontend/crawlab-ui/src/components/ui/button/FaIconButton.vue
new file mode 100644
index 00000000..e44e9f2f
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/ui/button/FaIconButton.vue
@@ -0,0 +1,50 @@
+
+
+
+ emit('click', event)"
+ @mouseenter="(event: Event) => emit('mouseenter', event)"
+ @mouseleave="(event: Event) => emit('mouseleave', event)"
+ >
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/ui/button/IconButton.vue b/frontend/crawlab-ui/src/components/ui/button/IconButton.vue
new file mode 100644
index 00000000..0f589ec4
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/ui/button/IconButton.vue
@@ -0,0 +1,39 @@
+
+
+
+
+
+ emit('click', event)"
+ @mouseenter="(event: Event) => emit('mouseenter', event)"
+ @mouseleave="(event: Event) => emit('mouseleave', event)"
+ />
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/ui/button/LabelButton.vue b/frontend/crawlab-ui/src/components/ui/button/LabelButton.vue
new file mode 100644
index 00000000..c5e0029c
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/ui/button/LabelButton.vue
@@ -0,0 +1,39 @@
+
+
+
+ emit('click', event)"
+ @mouseenter="(event: Event) => emit('mouseenter', event)"
+ @mouseleave="(event: Event) => emit('mouseleave', event)"
+ >
+
+
+
+ {{ label }}
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/ui/button/types.d.ts b/frontend/crawlab-ui/src/components/ui/button/types.d.ts
new file mode 100644
index 00000000..2a47f9c8
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/ui/button/types.d.ts
@@ -0,0 +1,59 @@
+import type { TooltipTriggerType } from 'element-plus/es/components/tooltip/src/trigger';
+import type { ContextMenuItem } from '@/components/ui/context-menu/types';
+
+export interface ButtonProps {
+ tooltip?: string;
+ type?: BasicType;
+ size?: BasicSize;
+ round?: boolean;
+ circle?: boolean;
+ plain?: boolean;
+ disabled?: boolean;
+ isIcon?: boolean;
+ loading?: boolean;
+ onClick?: () => void;
+ className?: string;
+ id?: string;
+ noMargin?: boolean;
+}
+
+export interface IconButtonProps extends ButtonProps {
+ icon: Icon;
+}
+
+export interface FaIconButtonProps extends IconButtonProps {
+ badgeIcon?: Icon;
+ spin?: boolean;
+}
+
+export interface LabelButtonProps extends ButtonProps {
+ label?: string;
+ icon?: Icon;
+}
+
+type GenericButtonProps = (
+ | ButtonProps
+ | IconButtonProps
+ | FaIconButtonProps
+ | LabelButtonProps
+) & {
+ buttonType?: ButtonType;
+};
+
+export interface ButtonGroupProps {
+ buttons?: GenericButtonProps[];
+ dropdownItems?: ContextMenuItem[];
+ dropdownTrigger?: TooltipTriggerType;
+ type?: BasicType;
+ size?: BasicSize;
+}
+
+export interface ButtonEmits {
+ (e: 'click', event: Event): void;
+
+ (e: 'mouseenter', event: Event): void;
+
+ (e: 'mouseleave', event: Event): void;
+}
+
+export type ButtonType = 'button' | 'fa-icon' | 'icon' | 'label';
diff --git a/frontend/crawlab-ui/src/components/ui/chart/Chart.vue b/frontend/crawlab-ui/src/components/ui/chart/Chart.vue
new file mode 100644
index 00000000..b69c8c11
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/ui/chart/Chart.vue
@@ -0,0 +1,66 @@
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/ui/chart/Metric.vue b/frontend/crawlab-ui/src/components/ui/chart/Metric.vue
new file mode 100644
index 00000000..8ba4a164
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/ui/chart/Metric.vue
@@ -0,0 +1,114 @@
+
+
+
+
+
+
+
+
+
+
+ {{ t(title || '') }}
+
+
+ {{ value }}
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/ui/chat/ChatConfigDialog.vue b/frontend/crawlab-ui/src/components/ui/chat/ChatConfigDialog.vue
new file mode 100644
index 00000000..66e300d1
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/ui/chat/ChatConfigDialog.vue
@@ -0,0 +1,99 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/ui/chat/ChatHistory.vue b/frontend/crawlab-ui/src/components/ui/chat/ChatHistory.vue
new file mode 100644
index 00000000..94108282
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/ui/chat/ChatHistory.vue
@@ -0,0 +1,177 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ {{ getConversationTitle(conversation) }}
+
+
+
+
+
+
+ {{ conversation.model }}
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/frontend/crawlab-ui/src/components/ui/chat/ChatInput.vue b/frontend/crawlab-ui/src/components/ui/chat/ChatInput.vue
new file mode 100644
index 00000000..fddea95c
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/ui/chat/ChatInput.vue
@@ -0,0 +1,447 @@
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/ui/chat/ChatMessage.vue b/frontend/crawlab-ui/src/components/ui/chat/ChatMessage.vue
new file mode 100644
index 00000000..c8bb34b7
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/ui/chat/ChatMessage.vue
@@ -0,0 +1,446 @@
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/ui/chat/ChatMessageAction.vue b/frontend/crawlab-ui/src/components/ui/chat/ChatMessageAction.vue
new file mode 100644
index 00000000..27b0bb7b
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/ui/chat/ChatMessageAction.vue
@@ -0,0 +1,274 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+ {{ content }}
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/ui/chat/ChatMessageList.vue b/frontend/crawlab-ui/src/components/ui/chat/ChatMessageList.vue
new file mode 100644
index 00000000..66eca4ee
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/ui/chat/ChatMessageList.vue
@@ -0,0 +1,96 @@
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/ui/chat/ChatSidebar.vue b/frontend/crawlab-ui/src/components/ui/chat/ChatSidebar.vue
new file mode 100644
index 00000000..cc687722
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/ui/chat/ChatSidebar.vue
@@ -0,0 +1,165 @@
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/ui/checkbox/CheckboxTree.vue b/frontend/crawlab-ui/src/components/ui/checkbox/CheckboxTree.vue
new file mode 100644
index 00000000..2e32e510
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/ui/checkbox/CheckboxTree.vue
@@ -0,0 +1,250 @@
+
+
+
+
+
+ updateState(op.id, checked, op)"
+ />
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/ui/checkbox/CheckboxTreeGroup.vue b/frontend/crawlab-ui/src/components/ui/checkbox/CheckboxTreeGroup.vue
new file mode 100644
index 00000000..d14a9c02
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/ui/checkbox/CheckboxTreeGroup.vue
@@ -0,0 +1,123 @@
+
+
+
+
+
+
+
+
+
+ {{ option.label }}
+
+
+
+
+ updateState(op.id!, checked, op)"
+ />
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/ui/context-menu/ContextMenu.vue b/frontend/crawlab-ui/src/components/ui/context-menu/ContextMenu.vue
new file mode 100644
index 00000000..146ffcc2
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/ui/context-menu/ContextMenu.vue
@@ -0,0 +1,38 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/ui/context-menu/ContextMenuList.vue b/frontend/crawlab-ui/src/components/ui/context-menu/ContextMenuList.vue
new file mode 100644
index 00000000..9274b97e
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/ui/context-menu/ContextMenuList.vue
@@ -0,0 +1,117 @@
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/ui/context-menu/types.d.ts b/frontend/crawlab-ui/src/components/ui/context-menu/types.d.ts
new file mode 100644
index 00000000..13439203
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/ui/context-menu/types.d.ts
@@ -0,0 +1,29 @@
+import { TooltipTriggerType } from 'element-plus/es/components/tooltip/src/trigger';
+import { Placement } from '@popperjs/core';
+import { StyleValue } from 'vue';
+
+export interface ContextMenuProps {
+ trigger?: TooltipTriggerType;
+ visible?: boolean;
+ activeItem?: FileNavItem;
+ placement?: Placement;
+ clicking?: boolean;
+ style?: StyleValue;
+}
+
+export interface ContextMenuListProps {
+ items: ContextMenuItem[];
+}
+
+export interface ContextMenuItem {
+ title: string;
+ icon?: Icon;
+ action?: () => void;
+ className?: string;
+ disabled?: boolean;
+}
+
+export type FileEditorContextMenuItemVisibleFn = (
+ contextMenuItem: ContextMenuItem,
+ activeFileNavItem?: FileNavItem
+) => boolean;
diff --git a/frontend/crawlab-ui/src/components/ui/date/DateRangePicker.vue b/frontend/crawlab-ui/src/components/ui/date/DateRangePicker.vue
new file mode 100644
index 00000000..73d9eaf8
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/ui/date/DateRangePicker.vue
@@ -0,0 +1,125 @@
+
+
+
+ emit('change', value)"
+ />
+
+
+
diff --git a/frontend/crawlab-ui/src/components/ui/date/DateTimeRangePicker.vue b/frontend/crawlab-ui/src/components/ui/date/DateTimeRangePicker.vue
new file mode 100644
index 00000000..1950b70b
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/ui/date/DateTimeRangePicker.vue
@@ -0,0 +1,164 @@
+
+
+
+ emit('change', value)"
+ />
+
+
+
diff --git a/frontend/crawlab-ui/src/components/ui/date/RangePicker.vue b/frontend/crawlab-ui/src/components/ui/date/RangePicker.vue
new file mode 100644
index 00000000..99a71070
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/ui/date/RangePicker.vue
@@ -0,0 +1,95 @@
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/ui/date/date.ts b/frontend/crawlab-ui/src/components/ui/date/date.ts
new file mode 100644
index 00000000..50550ec2
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/ui/date/date.ts
@@ -0,0 +1,16 @@
+export const getRangeItemOption = (
+ label: string,
+ key: RangeItemKey,
+ value?: RangeItemValue
+): RangeItemOption => {
+ if (typeof value === 'function') {
+ value = value();
+ }
+ return {
+ label,
+ value: {
+ key,
+ value,
+ },
+ };
+};
diff --git a/frontend/crawlab-ui/src/components/ui/dialog/ConfirmDialog.vue b/frontend/crawlab-ui/src/components/ui/dialog/ConfirmDialog.vue
new file mode 100644
index 00000000..b4b1ffbe
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/ui/dialog/ConfirmDialog.vue
@@ -0,0 +1,58 @@
+
+
+
+
+
+ {{ content }}
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/ui/dialog/CreateEditDialog.vue b/frontend/crawlab-ui/src/components/ui/dialog/CreateEditDialog.vue
new file mode 100644
index 00000000..d5cb66ae
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/ui/dialog/CreateEditDialog.vue
@@ -0,0 +1,85 @@
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/ui/dialog/Dialog.vue b/frontend/crawlab-ui/src/components/ui/dialog/Dialog.vue
new file mode 100644
index 00000000..c5a35477
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/ui/dialog/Dialog.vue
@@ -0,0 +1,99 @@
+
+
+
+
+
+
+
+
+
+ {{ title }}
+
+
+
+
+
+ {{ t('common.actions.cancel') }}
+
+
+ {{ confirmText || t('common.actions.confirm') }}
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/ui/drag/DraggableItem.vue b/frontend/crawlab-ui/src/components/ui/drag/DraggableItem.vue
new file mode 100644
index 00000000..0f6a8dac
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/ui/drag/DraggableItem.vue
@@ -0,0 +1,55 @@
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/ui/drag/DraggableList.vue b/frontend/crawlab-ui/src/components/ui/drag/DraggableList.vue
new file mode 100644
index 00000000..300a4f68
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/ui/drag/DraggableList.vue
@@ -0,0 +1,107 @@
+
+
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/ui/editor/EditorOptionsDropdownList.vue b/frontend/crawlab-ui/src/components/ui/editor/EditorOptionsDropdownList.vue
new file mode 100644
index 00000000..34de563e
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/ui/editor/EditorOptionsDropdownList.vue
@@ -0,0 +1,48 @@
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/ui/empty/Empty.vue b/frontend/crawlab-ui/src/components/ui/empty/Empty.vue
new file mode 100644
index 00000000..af4fa4d7
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/ui/empty/Empty.vue
@@ -0,0 +1,38 @@
+
+
+
+
+
+
+ {{ t(description) }}
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/ui/empty/ImgEmpty.vue b/frontend/crawlab-ui/src/components/ui/empty/ImgEmpty.vue
new file mode 100644
index 00000000..39104457
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/ui/empty/ImgEmpty.vue
@@ -0,0 +1,131 @@
+
+
+
+
+ Group 2
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/ui/export/ExportForm.vue b/frontend/crawlab-ui/src/components/ui/export/ExportForm.vue
new file mode 100644
index 00000000..f3a19da5
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/ui/export/ExportForm.vue
@@ -0,0 +1,55 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/ui/file/FileActions.vue b/frontend/crawlab-ui/src/components/ui/file/FileActions.vue
new file mode 100644
index 00000000..9d39f804
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/ui/file/FileActions.vue
@@ -0,0 +1,77 @@
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/ui/file/FileDiff.vue b/frontend/crawlab-ui/src/components/ui/file/FileDiff.vue
new file mode 100644
index 00000000..e07cd389
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/ui/file/FileDiff.vue
@@ -0,0 +1,130 @@
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/ui/file/FileEditor.vue b/frontend/crawlab-ui/src/components/ui/file/FileEditor.vue
new file mode 100644
index 00000000..00e78b0d
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/ui/file/FileEditor.vue
@@ -0,0 +1,515 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ {{ t('components.file.editor.empty.placeholder') }}
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/ui/file/FileEditorCreateWithAiDialog.vue b/frontend/crawlab-ui/src/components/ui/file/FileEditorCreateWithAiDialog.vue
new file mode 100644
index 00000000..0d720236
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/ui/file/FileEditorCreateWithAiDialog.vue
@@ -0,0 +1,206 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ {{ t('common.actions.cancel') }}
+
+ {{ t('common.actions.confirm') }}
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/ui/file/FileEditorNavMenu.vue b/frontend/crawlab-ui/src/components/ui/file/FileEditorNavMenu.vue
new file mode 100644
index 00000000..d75fabf9
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/ui/file/FileEditorNavMenu.vue
@@ -0,0 +1,673 @@
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/ui/file/FileEditorNavMenuContextMenu.vue b/frontend/crawlab-ui/src/components/ui/file/FileEditorNavMenuContextMenu.vue
new file mode 100644
index 00000000..f67f4eb9
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/ui/file/FileEditorNavMenuContextMenu.vue
@@ -0,0 +1,109 @@
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/ui/file/FileEditorNavTabs.vue b/frontend/crawlab-ui/src/components/ui/file/FileEditorNavTabs.vue
new file mode 100644
index 00000000..6fed27e2
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/ui/file/FileEditorNavTabs.vue
@@ -0,0 +1,255 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ {{ getTitle(item) }}
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/ui/file/FileEditorNavTabsContextMenu.vue b/frontend/crawlab-ui/src/components/ui/file/FileEditorNavTabsContextMenu.vue
new file mode 100644
index 00000000..290eabf8
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/ui/file/FileEditorNavTabsContextMenu.vue
@@ -0,0 +1,54 @@
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/ui/file/FileEditorNavTabsShowMoreContextMenu.vue b/frontend/crawlab-ui/src/components/ui/file/FileEditorNavTabsShowMoreContextMenu.vue
new file mode 100644
index 00000000..c96d6673
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/ui/file/FileEditorNavTabsShowMoreContextMenu.vue
@@ -0,0 +1,47 @@
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/ui/file/FileEditorSettingsDialog.vue b/frontend/crawlab-ui/src/components/ui/file/FileEditorSettingsDialog.vue
new file mode 100644
index 00000000..1b3b6ec4
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/ui/file/FileEditorSettingsDialog.vue
@@ -0,0 +1,118 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+ {{ t('common.actions.cancel') }}
+
+ {{ t('common.actions.confirm') }}
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/ui/file/FileTab.vue b/frontend/crawlab-ui/src/components/ui/file/FileTab.vue
new file mode 100644
index 00000000..26e215c2
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/ui/file/FileTab.vue
@@ -0,0 +1,278 @@
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/ui/file/FileUpload.vue b/frontend/crawlab-ui/src/components/ui/file/FileUpload.vue
new file mode 100644
index 00000000..8e6b1279
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/ui/file/FileUpload.vue
@@ -0,0 +1,320 @@
+
+
+
+
+
+
+ emit('mode-change', value)"
+ >
+
+ {{ label }}
+
+
+
+
+
+ emit('directory-change', value)"
+ @clear="internalDirectory = FILE_ROOT"
+ >
+
+
+ {{ label }}
+
+
+ {{ value }}
+
+
+
+
+
+
+
+
+
+
+ {{
+ t(
+ 'components.file.upload.buttons.folder.clickToSelectFolderToUpload'
+ )
+ }}
+
+
+
+
+
+
+
+
+
+
+ {{ t('components.file.upload.buttons.files.dragFilesHereOr') }}
+ {{
+ t('components.file.upload.buttons.files.clickToUpload')
+ }}
+
+
+
+
+
+
+
+ {{ t('components.file.upload.fileList.title') }}
+
+
+
+
+
+
+
+ {{ data.label }}
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/ui/file/UploadFilesDialog.vue b/frontend/crawlab-ui/src/components/ui/file/UploadFilesDialog.vue
new file mode 100644
index 00000000..f6fe571e
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/ui/file/UploadFilesDialog.vue
@@ -0,0 +1,185 @@
+
+
+
+
+ (mode = value)"
+ @directory-change="onTargetDirectoryChange"
+ @files-change="onFilesChange"
+ />
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/ui/file/fileEditorDropZone.ts b/frontend/crawlab-ui/src/components/ui/file/fileEditorDropZone.ts
new file mode 100644
index 00000000..a16e23a6
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/ui/file/fileEditorDropZone.ts
@@ -0,0 +1,13 @@
+import { useDropzone } from 'crawlab-vue3-dropzone';
+
+const useFileEditorDropZone = () => {
+ const { getRootProps, getInputProps, open } = useDropzone({});
+
+ return {
+ getRootProps,
+ getInputProps,
+ open,
+ };
+};
+
+export default useFileEditorDropZone;
diff --git a/frontend/crawlab-ui/src/components/ui/filter/FilterCondition.vue b/frontend/crawlab-ui/src/components/ui/filter/FilterCondition.vue
new file mode 100644
index 00000000..d48f4524
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/ui/filter/FilterCondition.vue
@@ -0,0 +1,95 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/ui/filter/FilterConditionList.vue b/frontend/crawlab-ui/src/components/ui/filter/FilterConditionList.vue
new file mode 100644
index 00000000..8f92751a
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/ui/filter/FilterConditionList.vue
@@ -0,0 +1,55 @@
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/ui/filter/FilterInput.vue b/frontend/crawlab-ui/src/components/ui/filter/FilterInput.vue
new file mode 100644
index 00000000..0a0cb01d
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/ui/filter/FilterInput.vue
@@ -0,0 +1,59 @@
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/ui/filter/FilterSelect.vue b/frontend/crawlab-ui/src/components/ui/filter/FilterSelect.vue
new file mode 100644
index 00000000..3cc4aeaf
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/ui/filter/FilterSelect.vue
@@ -0,0 +1,136 @@
+
+
+
+
+
+
+
+
+
+ {{ activeOption?.label }}
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/ui/filter/filter.ts b/frontend/crawlab-ui/src/components/ui/filter/filter.ts
new file mode 100644
index 00000000..7d844638
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/ui/filter/filter.ts
@@ -0,0 +1,43 @@
+import { plainClone } from '@/utils';
+import {
+ FILTER_OP_CONTAINS,
+ FILTER_OP_EQUAL,
+ FILTER_OP_GREATER_THAN,
+ FILTER_OP_GREATER_THAN_EQUAL,
+ FILTER_OP_LESS_THAN,
+ FILTER_OP_LESS_THAN_EQUAL,
+ FILTER_OP_NOT_CONTAINS,
+ FILTER_OP_NOT_EQUAL,
+ FILTER_OP_NOT_SET,
+ FILTER_OP_REGEX,
+} from '@/constants';
+
+export const defaultFilterCondition: FilterConditionData = {
+ op: FILTER_OP_NOT_SET,
+ value: '',
+};
+
+export const getDefaultFilterCondition = () => {
+ return plainClone(defaultFilterCondition);
+};
+
+export const conditionTypesOptions: SelectOption[] = [
+ { value: FILTER_OP_NOT_SET, label: 'Not Set' },
+ { value: FILTER_OP_CONTAINS, label: 'Contains' },
+ { value: FILTER_OP_NOT_CONTAINS, label: 'Not Contains' },
+ { value: FILTER_OP_REGEX, label: 'Regex' },
+ { value: FILTER_OP_EQUAL, label: 'Equal to' },
+ { value: FILTER_OP_NOT_EQUAL, label: 'Not Equal to' },
+ { value: FILTER_OP_GREATER_THAN, label: 'Greater than' },
+ { value: FILTER_OP_LESS_THAN, label: 'Less than' },
+ { value: FILTER_OP_GREATER_THAN_EQUAL, label: 'Greater than or Equal to' },
+ { value: FILTER_OP_LESS_THAN_EQUAL, label: 'Less than or Equal to' },
+];
+
+export const conditionTypesMap: { [key: string]: string } = (() => {
+ const map: { [key: string]: string } = {};
+ conditionTypesOptions.forEach(d => {
+ map[d.value] = d.label as string;
+ });
+ return map;
+})();
diff --git a/frontend/crawlab-ui/src/components/ui/form/Form.vue b/frontend/crawlab-ui/src/components/ui/form/Form.vue
new file mode 100644
index 00000000..58f12d78
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/ui/form/Form.vue
@@ -0,0 +1,76 @@
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/ui/form/FormItem.vue b/frontend/crawlab-ui/src/components/ui/form/FormItem.vue
new file mode 100644
index 00000000..763ca25b
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/ui/form/FormItem.vue
@@ -0,0 +1,200 @@
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/ui/form/FormTableField.vue b/frontend/crawlab-ui/src/components/ui/form/FormTableField.vue
new file mode 100644
index 00000000..fbd52682
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/ui/form/FormTableField.vue
@@ -0,0 +1,182 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/frontend/crawlab-ui/src/components/ui/form/formTable.ts b/frontend/crawlab-ui/src/components/ui/form/formTable.ts
new file mode 100644
index 00000000..a5c2ad95
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/ui/form/formTable.ts
@@ -0,0 +1,67 @@
+import { Store } from 'vuex';
+import { plainClone } from '@/utils/object';
+import { computed, Ref } from 'vue';
+
+export const useFormTable = (
+ ns: ListStoreNamespace,
+ store: Store,
+ data: FormComponentData
+) => {
+ const { formTableFieldRefsMap } = data;
+
+ // state
+ const state = store.state[ns as keyof RootStoreState] as BaseStoreState;
+
+ // form list
+ const formList = computed(() => state.formList);
+
+ // get new form
+ const getNewForm = state.newFormFn;
+
+ const onAdd = (index: number) => {
+ formList.value.splice(index, 0, getNewForm());
+ };
+
+ const onClone = (index: number) => {
+ const form = plainClone(formList.value[index]);
+ formList.value.splice(index, 0, form);
+ };
+
+ const onDelete = (index: number) => {
+ formList.value.splice(index, 1);
+ for (const key of formTableFieldRefsMap.value.keys()) {
+ const rowIndex = key[0];
+ if (rowIndex === index) {
+ formTableFieldRefsMap.value.delete(key);
+ }
+ }
+ };
+
+ const onFieldChange = (rowIndex: number, prop: string, value: any) => {
+ if (rowIndex !== -1) {
+ // one row change
+ const item = formList.value[rowIndex] as BaseModel;
+ item[prop] = value;
+ } else {
+ // all rows change
+ for (let i = 0; i < formList.value.length; i++) {
+ onFieldChange(i, prop, value);
+ }
+ }
+ };
+
+ const onFieldRegister = (rowIndex: number, prop: string, formRef: Ref) => {
+ const key = [rowIndex, prop] as FormTableFieldRefsMapKey;
+ formTableFieldRefsMap.value.set(key, formRef);
+ };
+
+ return {
+ onAdd,
+ onClone,
+ onDelete,
+ onFieldChange,
+ onFieldRegister,
+ };
+};
+
+export default useFormTable;
diff --git a/frontend/crawlab-ui/src/components/ui/form/index.ts b/frontend/crawlab-ui/src/components/ui/form/index.ts
new file mode 100644
index 00000000..45d84816
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/ui/form/index.ts
@@ -0,0 +1,2 @@
+export * from './useForm';
+export * from './formTable';
diff --git a/frontend/crawlab-ui/src/components/ui/form/useForm.ts b/frontend/crawlab-ui/src/components/ui/form/useForm.ts
new file mode 100644
index 00000000..abc4d1be
--- /dev/null
+++ b/frontend/crawlab-ui/src/components/ui/form/useForm.ts
@@ -0,0 +1,241 @@
+import { computed, provide, watch } from 'vue';
+import { Store } from 'vuex';
+import useFormTable from '@/components/ui/form/formTable';
+import { EMPTY_OBJECT_ID } from '@/utils/mongo';
+import { translate } from '@/utils/i18n';
+
+// i18n
+const t = translate;
+
+export const useForm = (
+ ns: ListStoreNamespace,
+ store: Store,
+ services: Services,
+ data: FormComponentData
+) => {
+ const { formRef, formTableFieldRefsMap } = data;
+
+ // state
+ const state = store.state[ns as keyof RootStoreState] as BaseStoreState;
+
+ // get new form
+ const getNewForm = state.newFormFn;
+
+ // get new form list
+ const getNewFormList = () => {
+ const list = [];
+ for (let i = 0; i < 5; i++) {
+ list.push(getNewForm());
+ }
+ return list;
+ };
+
+ // form
+ const form = computed(() => state.form);
+
+ // form list
+ const formList = computed(() => state.formList);
+
+ // active dialog key
+ const activeDialogKey = computed(
+ () => state.activeDialogKey as DialogKey
+ );
+
+ // is selective form
+ const isSelectiveForm = computed(() => state.isSelectiveForm);
+
+ // selected form fields
+ const selectedFormFields = computed(() => state.selectedFormFields);
+
+ // readonly form fields
+ const readonlyFormFields = computed(() => state.readonlyFormFields);
+
+ const validateForm = async () => {
+ return await formRef.value?.validate();
+ };
+
+ const resetForm = () => {
+ if (activeDialogKey.value) {
+ switch (activeDialogKey.value) {
+ case 'create':
+ store.commit(`${ns}/setForm`, getNewForm());
+ store.commit(`${ns}/setFormList`, getNewFormList());
+ break;
+ case 'edit':
+ formRef.value?.clearValidate();
+ break;
+ }
+ } else {
+ formRef.value?.resetFields();
+ formTableFieldRefsMap.value = new Map();
+ }
+ };
+
+ // whether form item is disabled
+ const isFormItemDisabled = (prop: string) => {
+ if (readonlyFormFields.value.includes(prop)) {
+ return true;
+ }
+ if (!isSelectiveForm.value) return false;
+ if (!prop) return false;
+ return !selectedFormFields.value.includes(prop);
+ };
+
+ // whether the form is empty
+ const isEmptyForm = (d: any): boolean => {
+ return JSON.stringify(d) === JSON.stringify(getNewForm());
+ };
+ provide<(d: any) => boolean>('fn:isEmptyForm', isEmptyForm);
+
+ // all list select options
+ const allListSelectOptions = computed(
+ () => store.getters[`${ns}/allListSelectOptions`]
+ );
+
+ // all list select options with empty
+ const allListSelectOptionsWithEmpty = computed(() =>
+ allListSelectOptions.value.concat({
+ label: t('common.status.unassigned'),
+ value: EMPTY_OBJECT_ID,
+ })
+ );
+
+ // all dict
+ const allDict = computed