From 262bf413796be799af95801b470f8135265cdd25 Mon Sep 17 00:00:00 2001
From: YeMingfei666 <1619116647@qq.com>
Date: Sat, 8 Feb 2025 10:15:06 +0800
Subject: [PATCH] first commiit
---
.editorconfig | 15 +
.env.development | 17 +
.env.production | 6 +
.eslintrc-auto-import.json | 316 +++
.gitignore | 18 +
.husky/commit-msg | 1 +
.husky/pre-commit | 1 +
.prettierignore | 12 +
.prettierrc.yaml | 41 +
.stylelintignore | 11 +
.stylelintrc.cjs | 52 +
.vscode/extensions.json | 11 +
.vscode/settings.json | 81 +
.vscode/vue3.0.code-snippets | 23 +
.vscode/vue3.2.code-snippets | 17 +
.vscode/vue3.3.code-snippets | 21 +
CHANGELOG.md | 386 ++++
LICENSE | 21 +
README.en-US.md | 176 ++
README.md | 205 +-
commitlint.config.cjs | 95 +
eslint.config.js | 115 ++
index.html | 61 +
licenses/vue-element-admin/LICENSE | 21 +
licenses/vue3-element-admin/LICENSE | 21 +
mock/auth.mock.ts | 43 +
mock/base.ts | 7 +
mock/dept.mock.ts | 153 ++
mock/dict-data.mock.ts | 216 ++
mock/dict.mock.ts | 181 ++
mock/log.mock.ts | 207 ++
mock/menu.mock.ts | 1711 ++++++++++++++++
mock/notice.mock.ts | 416 ++++
mock/role.mock.ts | 334 ++++
mock/user.mock.ts | 237 +++
package.json | 125 ++
public/favicon.ico | Bin 0 -> 4286 bytes
src/App.vue | 32 +
src/api/auth/index.ts | 84 +
src/api/codegen/index.ts | 191 ++
src/api/coup/index.js | 205 ++
src/api/file/index.ts | 81 +
src/api/login.js | 58 +
src/api/system/config.ts | 104 +
src/api/system/dept.ts | 130 ++
src/api/system/dict-data.ts | 162 ++
src/api/system/dict.ts | 180 ++
src/api/system/log.ts | 121 ++
src/api/system/menu.ts | 209 ++
src/api/system/notice.ts | 199 ++
src/api/system/role.ts | 138 ++
src/api/system/user.ts | 384 ++++
src/assets/icons/api.svg | 1 +
src/assets/icons/backtop.svg | 1 +
src/assets/icons/bilibili.svg | 1 +
src/assets/icons/browser.svg | 1 +
src/assets/icons/captcha.svg | 1 +
src/assets/icons/cascader.svg | 1 +
src/assets/icons/client.svg | 1 +
src/assets/icons/close.svg | 1 +
src/assets/icons/close_all.svg | 1 +
src/assets/icons/close_left.svg | 1 +
src/assets/icons/close_other.svg | 1 +
src/assets/icons/close_right.svg | 1 +
src/assets/icons/cnblogs.svg | 1 +
src/assets/icons/code.svg | 1 +
src/assets/icons/collapse.svg | 1 +
src/assets/icons/csdn.svg | 6 +
src/assets/icons/data_statistics.svg | 1 +
src/assets/icons/dict.svg | 1 +
src/assets/icons/document.svg | 1 +
src/assets/icons/down.svg | 1 +
src/assets/icons/download.svg | 1 +
src/assets/icons/enter.svg | 1 +
src/assets/icons/esc.svg | 1 +
src/assets/icons/file.svg | 1 +
src/assets/icons/fullscreen-exit.svg | 1 +
src/assets/icons/fullscreen.svg | 1 +
src/assets/icons/gitcode.svg | 1 +
src/assets/icons/gitee.svg | 1 +
src/assets/icons/github.svg | 1 +
src/assets/icons/homepage.svg | 1 +
src/assets/icons/java.svg | 1 +
src/assets/icons/juejin.svg | 1 +
src/assets/icons/language.svg | 1 +
src/assets/icons/menu.svg | 1 +
src/assets/icons/message.svg | 1 +
src/assets/icons/monitor.svg | 1 +
src/assets/icons/project.svg | 1 +
src/assets/icons/qq.svg | 1 +
src/assets/icons/refresh.svg | 1 +
src/assets/icons/role.svg | 1 +
src/assets/icons/search.svg | 1 +
src/assets/icons/setting.svg | 1 +
src/assets/icons/size.svg | 1 +
src/assets/icons/system.svg | 1 +
src/assets/icons/table.svg | 1 +
src/assets/icons/todo.svg | 1 +
src/assets/icons/tree.svg | 1 +
src/assets/icons/typescript.svg | 1 +
src/assets/icons/up.svg | 1 +
src/assets/icons/user.svg | 1 +
src/assets/icons/visitor.svg | 1 +
src/assets/icons/vue.svg | 1 +
src/assets/icons/wechat.svg | 1 +
src/assets/icons/xml.svg | 1 +
src/assets/images/1024.png | Bin 0 -> 1797 bytes
src/assets/images/401.svg | 398 ++++
src/assets/images/404.svg | 340 ++++
src/assets/images/avatar.png | Bin 0 -> 1865 bytes
src/assets/images/background.webp | Bin 0 -> 156066 bytes
src/assets/images/background_img.jpg | Bin 0 -> 373022 bytes
src/assets/images/data_home_bg1.png | Bin 0 -> 30072 bytes
src/assets/images/data_home_item1.png | Bin 0 -> 1977 bytes
src/assets/images/data_home_item1_icon.png | Bin 0 -> 1268 bytes
src/assets/images/data_home_item2.png | Bin 0 -> 2054 bytes
src/assets/images/data_home_item2_icon.png | Bin 0 -> 919 bytes
src/assets/images/data_home_item3.png | Bin 0 -> 2091 bytes
src/assets/images/data_home_item3_icon.png | Bin 0 -> 1154 bytes
src/assets/images/data_home_item4.png | Bin 0 -> 2210 bytes
src/assets/images/data_home_item4_icon.png | Bin 0 -> 1473 bytes
src/assets/images/default_logo.png | Bin 0 -> 5146 bytes
src/assets/images/home/data_forms1.png | Bin 0 -> 38425 bytes
src/assets/images/home/data_forms2.png | Bin 0 -> 38869 bytes
src/assets/images/home/data_forms3.png | Bin 0 -> 39685 bytes
src/assets/images/home/data_forms4.png | Bin 0 -> 29871 bytes
src/assets/images/img_download_error.png | Bin 0 -> 14486 bytes
src/assets/images/login-bg-dark.jpg | Bin 0 -> 20471 bytes
src/assets/images/login-bg.jpg | Bin 0 -> 32872 bytes
src/assets/images/logo.png | Bin 0 -> 33158 bytes
src/assets/images/perpole.png | Bin 0 -> 877 bytes
src/assets/images/scan.png | Bin 0 -> 1988 bytes
src/assets/images/shalou.png | Bin 0 -> 412 bytes
src/assets/images/shop_editor_bg.png | Bin 0 -> 101293 bytes
src/assets/images/shouye.jpeg | Bin 0 -> 154015 bytes
src/assets/images/upload.png | Bin 0 -> 934 bytes
src/assets/logo.png | Bin 0 -> 2530 bytes
src/components/AppLink/index.vue | 38 +
src/components/Breadcrumb/index.vue | 87 +
src/components/CURD/PageContent.vue | 976 +++++++++
src/components/CURD/PageForm.vue | 156 ++
src/components/CURD/PageModal.vue | 377 ++++
src/components/CURD/PageSearch.vue | 239 +++
src/components/CURD/types.ts | 272 +++
src/components/CURD/usePage.ts | 68 +
src/components/CopyButton/index.vue | 62 +
src/components/Dict/DictLabel.vue | 52 +
src/components/Dict/index.vue | 140 ++
src/components/Fullscreen/index.vue | 11 +
src/components/GithubCorner/index.vue | 62 +
src/components/Hamburger/index.vue | 37 +
src/components/IconSelect/index.vue | 207 ++
src/components/LangSelect/index.vue | 47 +
src/components/MenuSearch/index.vue | 223 +++
src/components/Pagination/index.vue | 92 +
src/components/SizeSelect/index.vue | 42 +
src/components/SvgIcon/index.vue | 41 +
src/components/TableSelect/index.vue | 357 ++++
src/components/Upload/FileUpload.vue | 237 +++
src/components/Upload/MultiImageUpload.vue | 216 ++
src/components/Upload/SingleImageUpload.vue | 203 ++
src/components/WangEditor/index.vue | 90 +
src/directive/index.ts | 9 +
src/directive/permission/index.ts | 64 +
src/enums/DeviceEnum.ts | 14 +
src/enums/FormTypeEnum.ts | 15 +
src/enums/LanguageEnum.ts | 14 +
src/enums/LayoutEnum.ts | 18 +
src/enums/MenuTypeEnum.ts | 22 +
src/enums/QueryTypeEnum.ts | 37 +
src/enums/ResultEnum.ts | 23 +
src/enums/SidebarStatusEnum.ts | 14 +
src/enums/SizeEnum.ts | 19 +
src/enums/ThemeEnum.ts | 18 +
src/lang/index.ts | 31 +
src/lang/package/en.ts | 62 +
src/lang/package/zh-cn.ts | 62 +
src/layout/components/AppMain/index.vue | 36 +
src/layout/components/HeaderTop/index.vue | 34 +
.../NavBar/components/NavbarRight.vue | 78 +
.../NavBar/components/Notification.vue | 237 +++
.../NavBar/components/UserProfile.vue | 60 +
src/layout/components/NavBar/index.vue | 39 +
.../Settings/components/LayoutSelect.vue | 108 +
.../Settings/components/ThemeColorPicker.vue | 41 +
src/layout/components/Settings/index.vue | 136 ++
.../Sidebar/components/SidebarLogo.vue | 53 +
.../Sidebar/components/SidebarMenu.vue | 113 ++
.../Sidebar/components/SidebarMenuItem.vue | 204 ++
.../components/SidebarMenuItemTitle.vue | 44 +
.../Sidebar/components/SidebarMixTopMenu.vue | 91 +
src/layout/components/Sidebar/index.vue | 46 +
src/layout/components/TagsView/index.vue | 442 ++++
src/layout/index.vue | 327 +++
src/main.ts | 19 +
src/plugins/icons.ts | 9 +
src/plugins/index.ts | 31 +
src/plugins/permission.ts | 88 +
src/router/index.ts | 84 +
src/settings.ts | 38 +
src/store/index.ts | 17 +
src/store/modules/app.ts | 107 +
src/store/modules/dict.ts | 41 +
src/store/modules/permission.ts | 116 ++
src/store/modules/settings.ts | 75 +
src/store/modules/tags-view.ts | 254 +++
src/store/modules/user.ts | 123 ++
src/styles/dark/css-vars.css | 7 +
src/styles/index.scss | 28 +
src/styles/reset.scss | 76 +
src/styles/variables.module.scss | 12 +
src/styles/variables.scss | 59 +
src/types/auto-imports.d.ts | 1780 +++++++++++++++++
src/types/components.d.ts | 98 +
src/types/env.d.ts | 33 +
src/types/global.d.ts | 106 +
src/types/router.d.ts | 54 +
src/types/shims-vue.d.ts | 5 +
src/types/socket.d.ts | 6 +
src/utils/auth.ts | 27 +
src/utils/globalCancelToken.js | 29 +
src/utils/i18n.ts | 12 +
src/utils/index.ts | 58 +
src/utils/nprogress.ts | 18 +
src/utils/request-php.js | 134 ++
src/utils/request.ts | 112 ++
src/utils/rsaEncrypt.js | 14 +
src/utils/theme.ts | 52 +
src/utils/websocket.ts | 93 +
src/views/data/credit.vue | 3 +
src/views/data/index.vue | 3 +
src/views/data/sales.vue | 3 +
src/views/data/table.vue | 3 +
src/views/data/work.vue | 3 +
src/views/error/401.vue | 85 +
src/views/error/404.vue | 123 ++
src/views/index/index.vue | 5 +
src/views/login/index.vue | 302 +++
src/views/redirect/index.vue | 15 +
tsconfig.json | 33 +
uno.config.ts | 44 +
vite.config.ts | 231 +++
242 files changed, 19959 insertions(+), 1 deletion(-)
create mode 100644 .editorconfig
create mode 100644 .env.development
create mode 100644 .env.production
create mode 100644 .eslintrc-auto-import.json
create mode 100644 .gitignore
create mode 100644 .husky/commit-msg
create mode 100644 .husky/pre-commit
create mode 100644 .prettierignore
create mode 100644 .prettierrc.yaml
create mode 100644 .stylelintignore
create mode 100644 .stylelintrc.cjs
create mode 100644 .vscode/extensions.json
create mode 100644 .vscode/settings.json
create mode 100644 .vscode/vue3.0.code-snippets
create mode 100644 .vscode/vue3.2.code-snippets
create mode 100644 .vscode/vue3.3.code-snippets
create mode 100644 CHANGELOG.md
create mode 100644 LICENSE
create mode 100644 README.en-US.md
create mode 100644 commitlint.config.cjs
create mode 100644 eslint.config.js
create mode 100644 index.html
create mode 100644 licenses/vue-element-admin/LICENSE
create mode 100644 licenses/vue3-element-admin/LICENSE
create mode 100644 mock/auth.mock.ts
create mode 100644 mock/base.ts
create mode 100644 mock/dept.mock.ts
create mode 100644 mock/dict-data.mock.ts
create mode 100644 mock/dict.mock.ts
create mode 100644 mock/log.mock.ts
create mode 100644 mock/menu.mock.ts
create mode 100644 mock/notice.mock.ts
create mode 100644 mock/role.mock.ts
create mode 100644 mock/user.mock.ts
create mode 100644 package.json
create mode 100644 public/favicon.ico
create mode 100644 src/App.vue
create mode 100644 src/api/auth/index.ts
create mode 100644 src/api/codegen/index.ts
create mode 100644 src/api/coup/index.js
create mode 100644 src/api/file/index.ts
create mode 100644 src/api/login.js
create mode 100644 src/api/system/config.ts
create mode 100644 src/api/system/dept.ts
create mode 100644 src/api/system/dict-data.ts
create mode 100644 src/api/system/dict.ts
create mode 100644 src/api/system/log.ts
create mode 100644 src/api/system/menu.ts
create mode 100644 src/api/system/notice.ts
create mode 100644 src/api/system/role.ts
create mode 100644 src/api/system/user.ts
create mode 100644 src/assets/icons/api.svg
create mode 100644 src/assets/icons/backtop.svg
create mode 100644 src/assets/icons/bilibili.svg
create mode 100644 src/assets/icons/browser.svg
create mode 100644 src/assets/icons/captcha.svg
create mode 100644 src/assets/icons/cascader.svg
create mode 100644 src/assets/icons/client.svg
create mode 100644 src/assets/icons/close.svg
create mode 100644 src/assets/icons/close_all.svg
create mode 100644 src/assets/icons/close_left.svg
create mode 100644 src/assets/icons/close_other.svg
create mode 100644 src/assets/icons/close_right.svg
create mode 100644 src/assets/icons/cnblogs.svg
create mode 100644 src/assets/icons/code.svg
create mode 100644 src/assets/icons/collapse.svg
create mode 100644 src/assets/icons/csdn.svg
create mode 100644 src/assets/icons/data_statistics.svg
create mode 100644 src/assets/icons/dict.svg
create mode 100644 src/assets/icons/document.svg
create mode 100644 src/assets/icons/down.svg
create mode 100644 src/assets/icons/download.svg
create mode 100644 src/assets/icons/enter.svg
create mode 100644 src/assets/icons/esc.svg
create mode 100644 src/assets/icons/file.svg
create mode 100644 src/assets/icons/fullscreen-exit.svg
create mode 100644 src/assets/icons/fullscreen.svg
create mode 100644 src/assets/icons/gitcode.svg
create mode 100644 src/assets/icons/gitee.svg
create mode 100644 src/assets/icons/github.svg
create mode 100644 src/assets/icons/homepage.svg
create mode 100644 src/assets/icons/java.svg
create mode 100644 src/assets/icons/juejin.svg
create mode 100644 src/assets/icons/language.svg
create mode 100644 src/assets/icons/menu.svg
create mode 100644 src/assets/icons/message.svg
create mode 100644 src/assets/icons/monitor.svg
create mode 100644 src/assets/icons/project.svg
create mode 100644 src/assets/icons/qq.svg
create mode 100644 src/assets/icons/refresh.svg
create mode 100644 src/assets/icons/role.svg
create mode 100644 src/assets/icons/search.svg
create mode 100644 src/assets/icons/setting.svg
create mode 100644 src/assets/icons/size.svg
create mode 100644 src/assets/icons/system.svg
create mode 100644 src/assets/icons/table.svg
create mode 100644 src/assets/icons/todo.svg
create mode 100644 src/assets/icons/tree.svg
create mode 100644 src/assets/icons/typescript.svg
create mode 100644 src/assets/icons/up.svg
create mode 100644 src/assets/icons/user.svg
create mode 100644 src/assets/icons/visitor.svg
create mode 100644 src/assets/icons/vue.svg
create mode 100644 src/assets/icons/wechat.svg
create mode 100644 src/assets/icons/xml.svg
create mode 100644 src/assets/images/1024.png
create mode 100644 src/assets/images/401.svg
create mode 100644 src/assets/images/404.svg
create mode 100644 src/assets/images/avatar.png
create mode 100644 src/assets/images/background.webp
create mode 100644 src/assets/images/background_img.jpg
create mode 100644 src/assets/images/data_home_bg1.png
create mode 100644 src/assets/images/data_home_item1.png
create mode 100644 src/assets/images/data_home_item1_icon.png
create mode 100644 src/assets/images/data_home_item2.png
create mode 100644 src/assets/images/data_home_item2_icon.png
create mode 100644 src/assets/images/data_home_item3.png
create mode 100644 src/assets/images/data_home_item3_icon.png
create mode 100644 src/assets/images/data_home_item4.png
create mode 100644 src/assets/images/data_home_item4_icon.png
create mode 100644 src/assets/images/default_logo.png
create mode 100644 src/assets/images/home/data_forms1.png
create mode 100644 src/assets/images/home/data_forms2.png
create mode 100644 src/assets/images/home/data_forms3.png
create mode 100644 src/assets/images/home/data_forms4.png
create mode 100644 src/assets/images/img_download_error.png
create mode 100644 src/assets/images/login-bg-dark.jpg
create mode 100644 src/assets/images/login-bg.jpg
create mode 100644 src/assets/images/logo.png
create mode 100644 src/assets/images/perpole.png
create mode 100644 src/assets/images/scan.png
create mode 100644 src/assets/images/shalou.png
create mode 100644 src/assets/images/shop_editor_bg.png
create mode 100644 src/assets/images/shouye.jpeg
create mode 100644 src/assets/images/upload.png
create mode 100644 src/assets/logo.png
create mode 100644 src/components/AppLink/index.vue
create mode 100644 src/components/Breadcrumb/index.vue
create mode 100644 src/components/CURD/PageContent.vue
create mode 100644 src/components/CURD/PageForm.vue
create mode 100644 src/components/CURD/PageModal.vue
create mode 100644 src/components/CURD/PageSearch.vue
create mode 100644 src/components/CURD/types.ts
create mode 100644 src/components/CURD/usePage.ts
create mode 100644 src/components/CopyButton/index.vue
create mode 100644 src/components/Dict/DictLabel.vue
create mode 100644 src/components/Dict/index.vue
create mode 100644 src/components/Fullscreen/index.vue
create mode 100644 src/components/GithubCorner/index.vue
create mode 100644 src/components/Hamburger/index.vue
create mode 100644 src/components/IconSelect/index.vue
create mode 100644 src/components/LangSelect/index.vue
create mode 100644 src/components/MenuSearch/index.vue
create mode 100644 src/components/Pagination/index.vue
create mode 100644 src/components/SizeSelect/index.vue
create mode 100644 src/components/SvgIcon/index.vue
create mode 100644 src/components/TableSelect/index.vue
create mode 100644 src/components/Upload/FileUpload.vue
create mode 100644 src/components/Upload/MultiImageUpload.vue
create mode 100644 src/components/Upload/SingleImageUpload.vue
create mode 100644 src/components/WangEditor/index.vue
create mode 100644 src/directive/index.ts
create mode 100644 src/directive/permission/index.ts
create mode 100644 src/enums/DeviceEnum.ts
create mode 100644 src/enums/FormTypeEnum.ts
create mode 100644 src/enums/LanguageEnum.ts
create mode 100644 src/enums/LayoutEnum.ts
create mode 100644 src/enums/MenuTypeEnum.ts
create mode 100644 src/enums/QueryTypeEnum.ts
create mode 100644 src/enums/ResultEnum.ts
create mode 100644 src/enums/SidebarStatusEnum.ts
create mode 100644 src/enums/SizeEnum.ts
create mode 100644 src/enums/ThemeEnum.ts
create mode 100644 src/lang/index.ts
create mode 100644 src/lang/package/en.ts
create mode 100644 src/lang/package/zh-cn.ts
create mode 100644 src/layout/components/AppMain/index.vue
create mode 100644 src/layout/components/HeaderTop/index.vue
create mode 100644 src/layout/components/NavBar/components/NavbarRight.vue
create mode 100644 src/layout/components/NavBar/components/Notification.vue
create mode 100644 src/layout/components/NavBar/components/UserProfile.vue
create mode 100644 src/layout/components/NavBar/index.vue
create mode 100644 src/layout/components/Settings/components/LayoutSelect.vue
create mode 100644 src/layout/components/Settings/components/ThemeColorPicker.vue
create mode 100644 src/layout/components/Settings/index.vue
create mode 100644 src/layout/components/Sidebar/components/SidebarLogo.vue
create mode 100644 src/layout/components/Sidebar/components/SidebarMenu.vue
create mode 100644 src/layout/components/Sidebar/components/SidebarMenuItem.vue
create mode 100644 src/layout/components/Sidebar/components/SidebarMenuItemTitle.vue
create mode 100644 src/layout/components/Sidebar/components/SidebarMixTopMenu.vue
create mode 100644 src/layout/components/Sidebar/index.vue
create mode 100644 src/layout/components/TagsView/index.vue
create mode 100644 src/layout/index.vue
create mode 100644 src/main.ts
create mode 100644 src/plugins/icons.ts
create mode 100644 src/plugins/index.ts
create mode 100644 src/plugins/permission.ts
create mode 100644 src/router/index.ts
create mode 100644 src/settings.ts
create mode 100644 src/store/index.ts
create mode 100644 src/store/modules/app.ts
create mode 100644 src/store/modules/dict.ts
create mode 100644 src/store/modules/permission.ts
create mode 100644 src/store/modules/settings.ts
create mode 100644 src/store/modules/tags-view.ts
create mode 100644 src/store/modules/user.ts
create mode 100644 src/styles/dark/css-vars.css
create mode 100644 src/styles/index.scss
create mode 100644 src/styles/reset.scss
create mode 100644 src/styles/variables.module.scss
create mode 100644 src/styles/variables.scss
create mode 100644 src/types/auto-imports.d.ts
create mode 100644 src/types/components.d.ts
create mode 100644 src/types/env.d.ts
create mode 100644 src/types/global.d.ts
create mode 100644 src/types/router.d.ts
create mode 100644 src/types/shims-vue.d.ts
create mode 100644 src/types/socket.d.ts
create mode 100644 src/utils/auth.ts
create mode 100644 src/utils/globalCancelToken.js
create mode 100644 src/utils/i18n.ts
create mode 100644 src/utils/index.ts
create mode 100644 src/utils/nprogress.ts
create mode 100644 src/utils/request-php.js
create mode 100644 src/utils/request.ts
create mode 100644 src/utils/rsaEncrypt.js
create mode 100644 src/utils/theme.ts
create mode 100644 src/utils/websocket.ts
create mode 100644 src/views/data/credit.vue
create mode 100644 src/views/data/index.vue
create mode 100644 src/views/data/sales.vue
create mode 100644 src/views/data/table.vue
create mode 100644 src/views/data/work.vue
create mode 100644 src/views/error/401.vue
create mode 100644 src/views/error/404.vue
create mode 100644 src/views/index/index.vue
create mode 100644 src/views/login/index.vue
create mode 100644 src/views/redirect/index.vue
create mode 100644 tsconfig.json
create mode 100644 uno.config.ts
create mode 100644 vite.config.ts
diff --git a/.editorconfig b/.editorconfig
new file mode 100644
index 0000000..00ee2de
--- /dev/null
+++ b/.editorconfig
@@ -0,0 +1,15 @@
+# http://editorconfig.org
+root = true
+
+# 表示所有文件适用
+[*]
+charset = utf-8 # 设置文件字符集为 utf-8
+end_of_line = lf # 控制换行类型(lf | cr | crlf)
+indent_style = space # 缩进风格(tab | space)
+indent_size = 2 # 缩进大小
+insert_final_newline = true # 始终在文件末尾插入一个新行
+
+# 表示仅 md 文件适用以下规则
+[*.md]
+max_line_length = off # 关闭最大行长度限制
+trim_trailing_whitespace = false # 关闭末尾空格修剪
diff --git a/.env.development b/.env.development
new file mode 100644
index 0000000..5cfea8e
--- /dev/null
+++ b/.env.development
@@ -0,0 +1,17 @@
+# 应用端口
+VITE_APP_PORT=3000
+
+# 代理前缀
+VITE_APP_BASE_API=/dev-api
+
+# 接口地址
+
+VITE_APP_API_URL=https://admintestpapi.sxczgkj.cn/ # 线上
+# VITE_APP_API_URL=https://api.youlai.tech # 线上
+# VITE_APP_API_URL=http://localhost:8989 # 本地
+
+# WebSocket 端点(不配置则关闭),线上 ws://api.youlai.tech/ws ,本地 ws://localhost:8989/ws
+VITE_APP_WS_ENDPOINT=
+
+# 启用 Mock 服务
+VITE_MOCK_DEV_SERVER=false
diff --git a/.env.production b/.env.production
new file mode 100644
index 0000000..17729b6
--- /dev/null
+++ b/.env.production
@@ -0,0 +1,6 @@
+# 代理前缀
+VITE_APP_BASE_API = '/prod-api'
+
+
+# WebSocket端点(可选)
+#VITE_APP_WS_ENDPOINT=wss://api.youlai.tech/ws
diff --git a/.eslintrc-auto-import.json b/.eslintrc-auto-import.json
new file mode 100644
index 0000000..0e8546f
--- /dev/null
+++ b/.eslintrc-auto-import.json
@@ -0,0 +1,316 @@
+{
+ "globals": {
+ "Component": true,
+ "ComponentPublicInstance": true,
+ "ComputedRef": true,
+ "EffectScope": true,
+ "ElMessage": true,
+ "ElMessageBox": true,
+ "ElNotification": true,
+ "InjectionKey": true,
+ "PropType": true,
+ "Ref": true,
+ "VNode": true,
+ "asyncComputed": true,
+ "autoResetRef": true,
+ "computed": true,
+ "computedAsync": true,
+ "computedEager": true,
+ "computedInject": true,
+ "computedWithControl": true,
+ "controlledComputed": true,
+ "controlledRef": true,
+ "createApp": true,
+ "createEventHook": true,
+ "createGlobalState": true,
+ "createInjectionState": true,
+ "createReactiveFn": true,
+ "createReusableTemplate": true,
+ "createSharedComposable": true,
+ "createTemplatePromise": true,
+ "createUnrefFn": true,
+ "customRef": true,
+ "debouncedRef": true,
+ "debouncedWatch": true,
+ "defineAsyncComponent": true,
+ "defineComponent": true,
+ "eagerComputed": true,
+ "effectScope": true,
+ "extendRef": true,
+ "getCurrentInstance": true,
+ "getCurrentScope": true,
+ "h": true,
+ "ignorableWatch": true,
+ "inject": true,
+ "isDefined": true,
+ "isProxy": true,
+ "isReactive": true,
+ "isReadonly": true,
+ "isRef": true,
+ "makeDestructurable": true,
+ "markRaw": true,
+ "nextTick": true,
+ "onActivated": true,
+ "onBeforeMount": true,
+ "onBeforeUnmount": true,
+ "onBeforeUpdate": true,
+ "onClickOutside": true,
+ "onDeactivated": true,
+ "onErrorCaptured": true,
+ "onKeyStroke": true,
+ "onLongPress": true,
+ "onMounted": true,
+ "onRenderTracked": true,
+ "onRenderTriggered": true,
+ "onScopeDispose": true,
+ "onServerPrefetch": true,
+ "onStartTyping": true,
+ "onUnmounted": true,
+ "onUpdated": true,
+ "pausableWatch": true,
+ "provide": true,
+ "reactify": true,
+ "reactifyObject": true,
+ "reactive": true,
+ "reactiveComputed": true,
+ "reactiveOmit": true,
+ "reactivePick": true,
+ "readonly": true,
+ "ref": true,
+ "refAutoReset": true,
+ "refDebounced": true,
+ "refDefault": true,
+ "refThrottled": true,
+ "refWithControl": true,
+ "resolveComponent": true,
+ "resolveRef": true,
+ "resolveUnref": true,
+ "shallowReactive": true,
+ "shallowReadonly": true,
+ "shallowRef": true,
+ "syncRef": true,
+ "syncRefs": true,
+ "templateRef": true,
+ "throttledRef": true,
+ "throttledWatch": true,
+ "toRaw": true,
+ "toReactive": true,
+ "toRef": true,
+ "toRefs": true,
+ "toValue": true,
+ "triggerRef": true,
+ "tryOnBeforeMount": true,
+ "tryOnBeforeUnmount": true,
+ "tryOnMounted": true,
+ "tryOnScopeDispose": true,
+ "tryOnUnmounted": true,
+ "unref": true,
+ "unrefElement": true,
+ "until": true,
+ "useActiveElement": true,
+ "useAnimate": true,
+ "useArrayDifference": true,
+ "useArrayEvery": true,
+ "useArrayFilter": true,
+ "useArrayFind": true,
+ "useArrayFindIndex": true,
+ "useArrayFindLast": true,
+ "useArrayIncludes": true,
+ "useArrayJoin": true,
+ "useArrayMap": true,
+ "useArrayReduce": true,
+ "useArraySome": true,
+ "useArrayUnique": true,
+ "useAsyncQueue": true,
+ "useAsyncState": true,
+ "useAttrs": true,
+ "useBase64": true,
+ "useBattery": true,
+ "useBluetooth": true,
+ "useBreakpoints": true,
+ "useBroadcastChannel": true,
+ "useBrowserLocation": true,
+ "useCached": true,
+ "useClipboard": true,
+ "useCloned": true,
+ "useColorMode": true,
+ "useConfirmDialog": true,
+ "useCounter": true,
+ "useCssModule": true,
+ "useCssVar": true,
+ "useCssVars": true,
+ "useCurrentElement": true,
+ "useCycleList": true,
+ "useDark": true,
+ "useDateFormat": true,
+ "useDebounce": true,
+ "useDebounceFn": true,
+ "useDebouncedRefHistory": true,
+ "useDeviceMotion": true,
+ "useDeviceOrientation": true,
+ "useDevicePixelRatio": true,
+ "useDevicesList": true,
+ "useDisplayMedia": true,
+ "useDocumentVisibility": true,
+ "useDraggable": true,
+ "useDropZone": true,
+ "useElementBounding": true,
+ "useElementByPoint": true,
+ "useElementHover": true,
+ "useElementSize": true,
+ "useElementVisibility": true,
+ "useEventBus": true,
+ "useEventListener": true,
+ "useEventSource": true,
+ "useEyeDropper": true,
+ "useFavicon": true,
+ "useFetch": true,
+ "useFileDialog": true,
+ "useFileSystemAccess": true,
+ "useFocus": true,
+ "useFocusWithin": true,
+ "useFps": true,
+ "useFullscreen": true,
+ "useGamepad": true,
+ "useGeolocation": true,
+ "useIdle": true,
+ "useImage": true,
+ "useInfiniteScroll": true,
+ "useIntersectionObserver": true,
+ "useInterval": true,
+ "useIntervalFn": true,
+ "useKeyModifier": true,
+ "useLastChanged": true,
+ "useLocalStorage": true,
+ "useMagicKeys": true,
+ "useManualRefHistory": true,
+ "useMediaControls": true,
+ "useMediaQuery": true,
+ "useMemoize": true,
+ "useMemory": true,
+ "useMounted": true,
+ "useMouse": true,
+ "useMouseInElement": true,
+ "useMousePressed": true,
+ "useMutationObserver": true,
+ "useNavigatorLanguage": true,
+ "useNetwork": true,
+ "useNow": true,
+ "useObjectUrl": true,
+ "useOffsetPagination": true,
+ "useOnline": true,
+ "usePageLeave": true,
+ "useParallax": true,
+ "useParentElement": true,
+ "usePerformanceObserver": true,
+ "usePermission": true,
+ "usePointer": true,
+ "usePointerLock": true,
+ "usePointerSwipe": true,
+ "usePreferredColorScheme": true,
+ "usePreferredContrast": true,
+ "usePreferredDark": true,
+ "usePreferredLanguages": true,
+ "usePreferredReducedMotion": true,
+ "usePrevious": true,
+ "useRafFn": true,
+ "useRefHistory": true,
+ "useResizeObserver": true,
+ "useScreenOrientation": true,
+ "useScreenSafeArea": true,
+ "useScriptTag": true,
+ "useScroll": true,
+ "useScrollLock": true,
+ "useSessionStorage": true,
+ "useShare": true,
+ "useSlots": true,
+ "useSorted": true,
+ "useSpeechRecognition": true,
+ "useSpeechSynthesis": true,
+ "useStepper": true,
+ "useStorage": true,
+ "useStorageAsync": true,
+ "useStyleTag": true,
+ "useSupported": true,
+ "useSwipe": true,
+ "useTemplateRefsList": true,
+ "useTextDirection": true,
+ "useTextSelection": true,
+ "useTextareaAutosize": true,
+ "useThrottle": true,
+ "useThrottleFn": true,
+ "useThrottledRefHistory": true,
+ "useTimeAgo": true,
+ "useTimeout": true,
+ "useTimeoutFn": true,
+ "useTimeoutPoll": true,
+ "useTimestamp": true,
+ "useTitle": true,
+ "useToNumber": true,
+ "useToString": true,
+ "useToggle": true,
+ "useTransition": true,
+ "useUrlSearchParams": true,
+ "useUserMedia": true,
+ "useVModel": true,
+ "useVModels": true,
+ "useVibrate": true,
+ "useVirtualList": true,
+ "useWakeLock": true,
+ "useWebNotification": true,
+ "useWebSocket": true,
+ "useWebWorker": true,
+ "useWebWorkerFn": true,
+ "useWindowFocus": true,
+ "useWindowScroll": true,
+ "useWindowSize": true,
+ "watch": true,
+ "watchArray": true,
+ "watchAtMost": true,
+ "watchDebounced": true,
+ "watchDeep": true,
+ "watchEffect": true,
+ "watchIgnorable": true,
+ "watchImmediate": true,
+ "watchOnce": true,
+ "watchPausable": true,
+ "watchPostEffect": true,
+ "watchSyncEffect": true,
+ "watchThrottled": true,
+ "watchTriggerable": true,
+ "watchWithFilter": true,
+ "useRoute": true,
+ "useRouter": true,
+ "storeToRefs": true,
+ "whenever": true,
+ "DirectiveBinding": true,
+ "ExtractDefaultPropTypes": true,
+ "ExtractPropTypes": true,
+ "ExtractPublicPropTypes": true,
+ "MaybeRef": true,
+ "MaybeRefOrGetter": true,
+ "WritableComputedRef": true,
+ "acceptHMRUpdate": true,
+ "createPinia": true,
+ "defineStore": true,
+ "getActivePinia": true,
+ "injectLocal": true,
+ "mapActions": true,
+ "mapGetters": true,
+ "mapState": true,
+ "mapStores": true,
+ "mapWritableState": true,
+ "onBeforeRouteLeave": true,
+ "onBeforeRouteUpdate": true,
+ "onWatcherCleanup": true,
+ "provideLocal": true,
+ "setActivePinia": true,
+ "setMapStoreSuffix": true,
+ "useClipboardItems": true,
+ "useI18n": true,
+ "useId": true,
+ "useLink": true,
+ "useModel": true,
+ "useTemplateRef": true
+ }
+}
diff --git a/.gitignore b/.gitignore
new file mode 100644
index 0000000..dccd5ce
--- /dev/null
+++ b/.gitignore
@@ -0,0 +1,18 @@
+node_modules
+.DS_Store
+dist
+dist-ssr
+*.local
+.history
+
+# Editor directories and files
+.idea
+*.suo
+*.ntvs*
+*.njsproj
+*.sln
+*.local
+
+stats.html
+pnpm-lock.yaml
+package-lock.json
diff --git a/.husky/commit-msg b/.husky/commit-msg
new file mode 100644
index 0000000..fd2bf70
--- /dev/null
+++ b/.husky/commit-msg
@@ -0,0 +1 @@
+npx --no-install commitlint --edit $1
diff --git a/.husky/pre-commit b/.husky/pre-commit
new file mode 100644
index 0000000..b18d041
--- /dev/null
+++ b/.husky/pre-commit
@@ -0,0 +1 @@
+npm run lint:lint-staged
diff --git a/.prettierignore b/.prettierignore
new file mode 100644
index 0000000..44421a7
--- /dev/null
+++ b/.prettierignore
@@ -0,0 +1,12 @@
+dist
+node_modules
+public
+.husky
+.vscode
+.idea
+*.sh
+*.md
+
+src/assets
+stats.html
+pnpm-lock.yaml
diff --git a/.prettierrc.yaml b/.prettierrc.yaml
new file mode 100644
index 0000000..d9cf0c7
--- /dev/null
+++ b/.prettierrc.yaml
@@ -0,0 +1,41 @@
+# 在单参数箭头函数中始终添加括号
+arrowParens: "always"
+# JSX 多行元素的闭合标签另起一行
+bracketSameLine: false
+# 对象字面量中的括号之间添加空格
+bracketSpacing: true
+# 自动格式化嵌入的代码(如 Markdown 和 HTML 内的代码)
+embeddedLanguageFormatting: "auto"
+# 忽略 HTML 空白敏感度,将空白视为非重要内容
+htmlWhitespaceSensitivity: "ignore"
+# 不插入 @prettier 的 pragma 注释
+insertPragma: false
+# 在 JSX 中使用双引号
+jsxSingleQuote: false
+# 每行代码的最大长度限制为 100 字符
+printWidth: 100
+# 在 Markdown 中保留原有的换行格式
+proseWrap: "preserve"
+# 仅在必要时添加对象属性的引号
+quoteProps: "as-needed"
+# 不要求文件开头插入 @prettier 的 pragma 注释
+requirePragma: false
+# 在语句末尾添加分号
+semi: true
+# 使用双引号而不是单引号
+singleQuote: false
+# 缩进使用 2 个空格
+tabWidth: 2
+# 在多行元素的末尾添加逗号(ES5 支持的对象、数组等)
+trailingComma: "es5"
+# 使用空格而不是制表符缩进
+useTabs: false
+# Vue 文件中的 ",
+ "",
+ "",
+ ""
+ ],
+ "description": "Vue3.0"
+ }
+}
diff --git a/.vscode/vue3.2.code-snippets b/.vscode/vue3.2.code-snippets
new file mode 100644
index 0000000..a083940
--- /dev/null
+++ b/.vscode/vue3.2.code-snippets
@@ -0,0 +1,17 @@
+{
+ "Vue3.2+快速生成模板": {
+ "scope": "vue",
+ "prefix": "Vue3.2+",
+ "body": [
+ "",
+ "",
+ "",
+ " ${1:test}
",
+ " ",
+ "",
+ "",
+ ""
+ ],
+ "description": "Vue3.2+"
+ }
+}
diff --git a/.vscode/vue3.3.code-snippets b/.vscode/vue3.3.code-snippets
new file mode 100644
index 0000000..705e04f
--- /dev/null
+++ b/.vscode/vue3.3.code-snippets
@@ -0,0 +1,21 @@
+{
+ "Vue3.3+defineOptions快速生成模板": {
+ "scope": "vue",
+ "prefix": "Vue3.3+",
+ "body": [
+ "",
+ "",
+ "",
+ " ${1:test}
",
+ " ",
+ "",
+ "",
+ ""
+ ],
+ "description": "Vue3.3+defineOptions快速生成模板"
+ }
+}
diff --git a/CHANGELOG.md b/CHANGELOG.md
new file mode 100644
index 0000000..fd2cef3
--- /dev/null
+++ b/CHANGELOG.md
@@ -0,0 +1,386 @@
+
+# 2.11.5 (2024/6/18)
+
+## ✨ feat
+
+- 支持后端文件导入([#142](https://github.com/youlaitech/vue3-element-admin/pull/142)) [@cshaptx4869](https://github.com/cshaptx4869)
+
+
+## 🐛 fix
+- vue-dev-tools 插件导致菜单路由切换卡死,暂时关闭 ([28349e](https://github.com/youlaitech/vue3-element-admin/commit/28349efe147afab36531ba148eaac3a448fe6c71)) [@haoxianrui](https://github.com/haoxianrui)
+
+
+
+# 2.11.4 (2024/6/16)
+
+## ✨ feat
+
+- 操作栏增加render配置参数([#138](https://github.com/youlaitech/vue3-element-admin/pull/140)) [@cshaptx4869](https://github.com/cshaptx4869)
+- 左侧工具栏增加type配置参数([#141](https://github.com/youlaitech/vue3-element-admin/pull/141)) [@diamont1001](https://github.com/diamont1001)
+
+## ♻️ refactor
+- 更换权限分配弹窗类型为 drawer 并添加父子联动开关([2d9193](https://github.com/youlaitech/vue3-element-admin/commit/2d9193c47fd224f01f82b9c0b2bbeb5e7cb33584)) [@haoxianrui](https://github.com/haoxianrui)
+
+
+
+# 2.11.3 (2024/6/11)
+
+## ✨ feat
+
+- 支持默认工具栏的导入([#138](https://github.com/youlaitech/vue3-element-admin/pull/138)) [@cshaptx4869](https://github.com/cshaptx4869)
+- 添加CURD导入示例([19e7bb](https://github.com/youlaitech/vue3-element-admin/commit/eab91effd6a01d5a3d9257249c8d06aa252b3bf8)) [@cshaptx4869](https://github.com/cshaptx4869)
+
+## ♻️ refactor
+- 修改导出全量数据选项文本([904fec](https://github.com/youlaitech/vue3-element-admin/commit/904fecad65217650482fcdbb10ffb7f3d27eb9ea)) [@cshaptx4869](https://github.com/cshaptx4869)
+
+## 🐛 fix
+- 菜单列表未适配el-icon导致图标不显示问题修复([e72b68](https://github.com/youlaitech/vue3-element-admin/commit/e72b68337562b5a7ea24ad55bbe00023e1266b40)) [@haoxianrui](https://github.com/haoxianrui)
+
+# 2.11.2 (2024/6/8)
+
+## ✨ feat
+
+- 支持表格远程筛选([#131](https://github.com/youlaitech/vue3-element-admin/pull/131)) [@cshaptx4869](https://github.com/cshaptx4869)
+- 支持标签输入框([#132](https://github.com/youlaitech/vue3-element-admin/pull/132)) [@cshaptx4869](https://github.com/cshaptx4869)
+- 表单项支持tips配置([#133](https://github.com/youlaitech/vue3-element-admin/pull/133)) [@cshaptx4869](https://github.com/cshaptx4869)
+- 前端导出支持全量数据([#134](https://github.com/youlaitech/vue3-element-admin/pull/134)) [@cshaptx4869](https://github.com/cshaptx4869)
+- 支持选中数据导出([#135](https://github.com/youlaitech/vue3-element-admin/pull/135)) [@cshaptx4869](https://github.com/cshaptx4869)
+- 表格默认工具栏的导出、搜索按钮增加权限点控制([883128](https://github.com/youlaitech/vue3-element-admin/commit/8831289b655f2cc086ecdababaa89f8d8a087c42)) [@cshaptx4869](https://github.com/cshaptx4869)
+- 页签title支持动态设置([23876a](https://github.com/youlaitech/vue3-element-admin/commit/23876aa396143bf77cb5c86af8d6023d9ff6555a)) [@haoxianrui](https://github.com/haoxianrui)
+
+## ♻️ refactor
+- 默认工具栏支持自定义([#136](https://github.com/youlaitech/vue3-element-admin/pull/136)) [@cshaptx4869](https://github.com/cshaptx4869)
+- 未配置全量导出接口时选项隐藏([eab91ef](https://github.com/youlaitech/vue3-element-admin/commit/eab91effd6a01d5a3d9257249c8d06aa252b3bf8)) [@cshaptx4869](https://github.com/cshaptx4869)
+
+## 🐛 fix
+- 修复注销登出后redirect跳转路由参数丢失([5626017](https://github.com/youlaitech/vue3-element-admin/commit/562601736731afd20bb1a5140d856f6515720159)) [@haoxianrui](https://github.com/haoxianrui)
+
+# 2.11.1 (2024/6/6)
+
+## ✨ feat
+
+- 增加pagination、request、parseData配置参数([#119](https://github.com/youlaitech/vue3-element-admin/pull/119)) [@cshaptx4869](https://github.com/cshaptx4869)
+- 增加返回顶部功能([#120](https://github.com/youlaitech/vue3-element-admin/pull/120)) [@cshaptx4869](https://github.com/cshaptx4869)
+- 支持前端导出([#126](https://github.com/youlaitech/vue3-element-admin/pull/126)) [@cshaptx4869](https://github.com/cshaptx4869)
+
+## ♻️ refactor
+- 重构布局样式(解决页面抖动问题)([#116](https://github.com/youlaitech/vue3-element-admin/pull/116)) [@cshaptx4869](https://github.com/cshaptx4869)
+- 修改CURD示例编辑弹窗尺寸([#121](https://github.com/youlaitech/vue3-element-admin/pull/121)) [@cshaptx4869](https://github.com/cshaptx4869)
+- 统一注册vue插件([#122](https://github.com/youlaitech/vue3-element-admin/pull/122)) [@cshaptx4869](https://github.com/cshaptx4869)
+- 默认主题跟随系统([#128](https://github.com/youlaitech/vue3-element-admin/pull/128)) [@cshaptx4869](https://github.com/cshaptx4869)
+- 增加"scss.lint.unknownAtRules": "ignore"代码,解决style中使用@apply提示unknow at rules@apply提示问题([Gitee#22](https://gitee.com/youlaiorg/vue3-element-admin/pulls/22)) [@zjsy521](https://gitee.com/zjsy521)
+
+## 🐛 fix
+- 修复左侧布局移动端菜单弹出样式 ([#117](https://github.com/youlaitech/vue3-element-admin/pull/117)) [@cshaptx4869](https://github.com/cshaptx4869)
+
+- 修复编辑后未清空id再新增菜单覆盖的问题([0e78eeb](https://github.com/youlaitech/vue3-element-admin/commit/0e78eeb75008fa8e9732b1b4e7d7a1ea345c7a1b)) [@haoxianrui](https://github.com/haoxianrui)
+- 修复水印层级问题([#123](https://github.com/youlaitech/vue3-element-admin/pull/123)) [@cshaptx4869](https://github.com/cshaptx4869)
+- 修复混合布局样式问题([#124](https://github.com/youlaitech/vue3-element-admin/pull/124)) [@cshaptx4869](https://github.com/cshaptx4869)
+- 修复关闭弹窗时没有clearValidate问题([#125](https://github.com/youlaitech/vue3-element-admin/pull/125)) [@andm31](https://github.com/andm31)
+
+
+
+# 2.11.0 (2024/5/27)
+
+## ✨ feat
+- 菜单添加路由参数设置(author by [haoxianrui](https://github.com/haoxianrui))
+- 增加列表选择组件(author by [cshaptx4869](https://github.com/cshaptx4869))
+- 增加列表选择组件使用示例(author by [cshaptx4869](https://github.com/cshaptx4869))
+- 增加defaultToolbar配置参数(author by [cshaptx4869](https://github.com/cshaptx4869))
+- 表单弹窗支持drawer模式(author by [cshaptx4869](https://github.com/cshaptx4869))
+- 表单项增加computed和watchEffect配置(author by [cshaptx4869](https://github.com/cshaptx4869))
+- 支持switch属性修改(author by [cshaptx4869](https://github.com/cshaptx4869))
+- 表单项增加文本类型支持(author by [cshaptx4869](https://github.com/cshaptx4869))
+- 列表列增加show配置项(author by [cshaptx4869](https://github.com/cshaptx4869))
+- 支持搜索表单显隐控制(author by [cshaptx4869](https://github.com/cshaptx4869))
+- 支持input属性修改(author by [cshaptx4869](https://github.com/cshaptx4869))
+- search配置新增函数能力拓展(author by [xiudaozhe](https://github.com/xiudaozhe))
+- 表格新增列设置控制(author by [haoxianrui](https://github.com/haoxianrui))
+- 搜索添加展开和收缩(author by [haoxianrui](https://github.com/haoxianrui))
+- watch函数增加配置项参数返回(author by [cshaptx4869](https://github.com/cshaptx4869))
+
+## ♻️ refactor
+- 重构图标选择组件(author by [cshaptx4869](https://github.com/cshaptx4869))
+- 重构列表选择组件默认样式 (author by [cshaptx4869](https://github.com/cshaptx4869))
+- 加强对话框表单组件和列表选择组件(author by [cshaptx4869](https://github.com/cshaptx4869))
+- routeMeta增加alwaysShow字段声明(author by [cshaptx4869](https://github.com/cshaptx4869))
+- 分页组件增加溢出滚动效果(author by [cshaptx4869](https://github.com/cshaptx4869))
+- 修正登录表单的Ref类型(author by [cshaptx4869](https://github.com/cshaptx4869))
+- 点击表格刷新按钮不重置页码(author by [cshaptx4869](https://github.com/cshaptx4869))
+- 筛选列超出一定高度滚动(author by [cshaptx4869](https://github.com/cshaptx4869))
+- 优化加强initFn函数,表单项增加initFn函数(author by [cshaptx4869](https://github.com/cshaptx4869))
+- 重构watch、computed、watchEffect调用(author by [cshaptx4869](https://github.com/cshaptx4869))
+- 修改操作成功提示(author by [cshaptx4869](https://github.com/cshaptx4869))
+- PageSearch 改用card作为容器,样式改用unocss写法(author by [cshaptx4869](https://github.com/cshaptx4869))
+- 优化首页 loading 动画效果author by [haoxianrui](https://github.com/haoxianrui))
+
+
+## 🐛 fix
+- 路由是否始终显示不限制只有顶级目录才有的配置,开放至菜单 (author by [haoxianrui](https://github.com/haoxianrui))
+- sockjs-client 报错 global is not defined 导致开发环境无法打开 WebSocket 页面问题修复 (author by [haoxianrui](https://github.com/haoxianrui))
+- 发送用户重启密码功能,最少为6位字符(小于6位登陆时不允许的问题) (author by [dreamnyj](https://gitee.com/dreamnyj))
+- 修复系统设置面板滚动条问题(author by [cshaptx4869](https://github.com/cshaptx4869))
+- 修复表单插槽失效问题(author by [cshaptx4869](https://github.com/cshaptx4869))
+- 修改tagsview刷新丢失query问题(author by [xiudaozhe](https://github.com/xiudaozhe))
+
+## 📦️ build
+- 升级 NPM 包版本至最新 (author by [haoxianrui](https://github.com/haoxianrui))
+
+## ⚙️ ci
+- 规整脚本执行命令(author by [cshaptx4869](https://github.com/cshaptx4869))
+
+
+# 2.10.1 (2024/5/4)
+
+## ♻️ refactor
+- 抽离CURD的使用部分代码为Hooks实现(author by [cshaptx4869](https://github.com/cshaptx4869))
+- 修改CURD导入权限点标识名(author by [cshaptx4869](https://github.com/cshaptx4869))
+- cURD表单字段支持watch监听(author by [cshaptx4869](https://github.com/cshaptx4869))
+- cURD表单input支持number修饰(author by [cshaptx4869](https://github.com/cshaptx4869))
+- cURD表单组件支持checkbox多选框(author by [cshaptx4869](https://github.com/cshaptx4869))
+- 优化axios响应数据TS类型提示(author by [cshaptx4869](https://github.com/cshaptx4869))
+- 修改CURD表单组件自定义类型的attrs传值(author by [cshaptx4869](https://github.com/cshaptx4869))
+- 同步重置密码按钮权限标识重命名(author by [haoxianrui](https://github.com/haoxianrui))
+- 重构API为静态方法实现模块化管理,并将types.ts重命名为model.ts用于存放接口模型定义(author by [haoxianrui](https://github.com/haoxianrui))
+
+
+## 🐛 fix
+- sockjs-client 报错 global is not defined 导致开发环境无法打开 WebSocket 页面问题修复 (author by [haoxianrui](https://github.com/haoxianrui))
+- 主题颜色设置覆盖暗黑模式下el-table行激活的背景色问题修复 (author by [haoxianrui](https://github.com/haoxianrui))
+- 修复因API接口调整而影响的调用页面的问题 (author by [haoxianrui](https://github.com/haoxianrui))
+
+## 📦️ build
+- 升级 NPM 包版本至最新 (author by [haoxianrui](https://github.com/haoxianrui))
+
+
+# 2.10.0 (2024/4/26)
+## ✨ feat
+- 封装增删改查组件(author by [cshaptx4869](https://github.com/cshaptx4869))
+- 集成 vite-plugin-vue-devtools 插件(author by [Tricker39](https://github.com/Tricker39))
+- 增加CURD配置化实现(author by [cshaptx4869](https://github.com/cshaptx4869))
+
+
+# 2.9.3 (2024/04/14)
+## ✨ feat
+- 增加vue文件代码片段(author by [cshaptx4869](https://github.com/cshaptx4869))
+- 菜单 hover 背景色添加值全局SCSS变量进行控制(author by [haoxianrui](https://github.com/haoxianrui))
+
+## ♻️ refactor
+- 加强基础国际化(author by [cshaptx4869](https://github.com/cshaptx4869))
+- 增加语言和布局大小枚举类型(author by [cshaptx4869](https://github.com/cshaptx4869))
+- 增加侧边栏状态枚举类型(author by [cshaptx4869](https://github.com/cshaptx4869))
+- 使用布局枚举替换字面量(author by [haoxianrui](https://github.com/haoxianrui))
+- 控制台使用静态数据循环渲染(author by [april](mailto:april@zen-game.cn))
+- 本地缓存的 token 变量重命名(author by [haoxianrui](https://github.com/haoxianrui))
+- 完善 Vite 环境变量类型声明(author by [haoxianrui](https://github.com/haoxianrui))
+
+## 🐛 fix
+- 修复构建时提示iconComponent.name可能为undefined的报错 (author by [wangji1042](https://github.com/wangji1042))
+- 修复浏览器密码自动填充时可能存在的报错 (author by [cshaptx4869](https://github.com/cshaptx4869))
+- 修复eslint报错(author by [cshaptx4869](https://github.com/cshaptx4869))
+- 移动端下点击左侧菜单节点后关闭侧边栏(author by [haoxianrui](https://github.com/haoxianrui))
+- 添加 size 类型断言修复类型报错(author by [haoxianrui](https://github.com/haoxianrui))
+
+## 📦️ build
+- husky9.x版本适配 (author by [cshaptx4869](https://github.com/cshaptx4869))
+- 升级 npm 包版本至最新(author by [haoxianrui](https://github.com/haoxianrui))
+
+# 2.9.2 (2024/03/05)
+## ✨ feat
+- vscode开发扩展推荐(author by [cshaptx4869](https://github.com/cshaptx4869))
+- 完善基础增删改查Mock接口(author by [haoxianrui](https://github.com/haoxianrui))
+
+## ♻️ refactor
+- 修改login密码框功能实现(author by [cshaptx4869](https://github.com/cshaptx4869))
+- 弱化页面进入动画效果(author by [cshaptx4869](https://github.com/cshaptx4869))
+- 取消推荐TypeScript Vue Plugin (author by [cshaptx4869](https://github.com/cshaptx4869))
+- 网站加载动画替换 (author by [haoxianrui](https://github.com/haoxianrui))
+- 优化主题和主题色监听,避免多个页面重复初始化 (author by [haoxianrui](https://github.com/haoxianrui))
+
+## 🐛 fix
+- AppMain 高度在非固定头部不正确导致出现滚动条问题修复 (author by [haoxianrui](https://github.com/haoxianrui))
+- 修复混合模式开启固定Head时的样式问题 (author by [cshaptx4869](https://github.com/cshaptx4869))
+- 设置面板统一字体大小 (author by [cshaptx4869](https://github.com/cshaptx4869))
+
+## 📦️build
+- 通过env配置控制mock服务 (author by [cshaptx4869](https://github.com/cshaptx4869))
+- 升级依赖包至最新版本 (author by [haoxianrui](https://github.com/haoxianrui))
+- 定义vite全局常量替换项目标题和版本 (author by [cshaptx4869](https://github.com/cshaptx4869))
+
+# 2.9.1 (2024/02/28)
+## ♻️ refactor
+- 项目配置按钮移入navbar(author by [cshaptx4869](https://github.com/cshaptx4869))
+- 优化user数据定义(author by [cshaptx4869](https://github.com/cshaptx4869))
+- 统一设置栏的 SVG 图标风格
+
+## 🐛 fix
+- 规整一些开发依赖(author by [cshaptx4869](https://github.com/cshaptx4869))
+- 修复登录页主题切换问题 (author by [cshaptx4869](https://github.com/cshaptx4869))
+
+## 🚀 pref
+
+- 压缩图片资源 (author by [cshaptx4869](https://github.com/cshaptx4869))
+
+
+# 2.9.0 (2024/02/25)
+
+## ✨ feat
+- 引入 animate.css 动画库
+- 新增水印和配置
+- 动态路由菜单支持 element plus 的图标
+
+## ♻️ refactor
+- Layout 布局重构和相关问题修复
+- sass 使用 @use 替代 @import 引入外部文件指令
+
+## 🐛 fix
+- 修复管理页面部分弹窗无法打开问题
+- 主题颜色设置按钮 hover 等未变化问题修复
+
+
+# 2.8.1 (2024/01/10)
+
+## ✨ feat
+- 替换 Mock 解决方案 vite-plugin-mock 为 vite-plugin-mock-dev-server 适配 Vite5
+
+# 2.8.0 (2023/12/27)
+
+## ⬆️ chore
+- 升级 Vite4 至 Vite5
+
+# 2.7.1 (2023/12/12)
+
+## ♻️ refactor
+- 将打包后的文件进行分类 (author by [ityangzhiwen](https://gitee.com/ityangzhiwen))
+
+# 2.7.0 (2023/11/19)
+
+## ♻️ refactor
+- 代码重构优化
+- 修改自动导入组件类型声明文件路径
+- 完善 typescript 类型
+
+## 🐛 fix
+- 修复管理页面部分弹窗无法打开问题
+
+
+# 2.7.0 (2023/11/19)
+
+## ♻️ refactor
+- 代码重构
+- 修改自动导入组件类型声明文件路径
+- 完善 typescript 类型
+
+## 🐛 fix
+- 修复管理页面部分弹窗无法打开问题
+
+
+# 2.6.3 (2023/10/22)
+
+## ✨ feat
+- 菜单管理新增目录只有一级子路由是否始终显示(alwaysShow)和路由页面是否缓存(keepAlive)的配置
+- 接口文档新增 swagger、knife4j
+- 引入和支持 tsx
+
+## ♻️ refactor
+- 代码瘦身,整理并删除未使用的 svg
+- 控制台样式优化
+
+## 🐛 fix
+- 菜单栏折叠和展开的图标暗黑模式显示问题修复
+
+
+# 2.6.2 (2023/10/11)
+
+## 🐛 fix
+- 主题设置未持久化问题
+- UnoCSS 插件无智能提示
+
+## ♻️ refactor
+- WebSocket 演示样式和代码优化
+- 用户管理代码重构
+
+# 2.6.1 (2023/9/4)
+
+## 🐛 fix
+- 导航顶部模式、混合模式样式在固定 Header 出现的样式问题修复
+- 固定 Header 没有持久化问题修复
+- 字典回显兼容 String 和 Number 类型
+
+# 2.6.0 (2023/8/24)💥💥💥
+
+## ✨ feat
+- 导航顶部模式、混合模式支持(author by [april-tong](https://april-tong.com/))
+- 平台文档(内嵌)(author by [april-tong](https://april-tong.com/))
+
+# 2.5.0 (2023/8/8)
+
+## ✨ feat
+- 新增 Mock(author by [ygcaicn](https://github.com/ygcaicn))
+- 图标 DEMO(author by [ygcaicn](https://github.com/ygcaicn))
+
+## 🐛 fix
+- 字典支持 Number 类型
+
+# 2.4.1 (2023/7/20)
+
+## ✨ feat
+- 整合 vite-plugin-compression 插件打包优化(3.66MB → 1.58MB) (author by [april-tong](https://april-tong.com/))
+- 字典组件封装(author by [haoxr](https://juejin.cn/user/4187394044331261/posts))
+
+## 🐛 fix
+- 分页组件hidden无效
+- 签名无法保存至后端
+- Git 提交 stylelint 校验部分机器报错
+
+# 2.4.0 (2023/6/17)
+
+## ✨ feat
+- 新增组件标签输入框(author by [april-tong](https://april-tong.com/))
+- 新增组件签名(author by [april-tong](https://april-tong.com/))
+- 新增组件表格(author by [april-tong](https://april-tong.com/))
+- Echarts 图表添加下载功能 author by [april-tong](https://april-tong.com/))
+
+## ♻️ refactor
+- 限制包管理器为 pnpm 和 node 版本16+
+- 自定义组件自动导入配置
+- 搜索框样式写法优化
+
+## 🐛 fix
+- 用户导入的部门回显成数字问题修复
+
+## ⬆️ chore
+- element-plus 版本升级 2.3.5 → 2.3.6
+
+# 2.3.1 (2023/5/21)
+
+## 🔄 refactor
+- 组件示例文件名称优化
+
+# 2.2.2 (2023/5/11)
+
+## ✨ feat
+- 组件封装示例添加源码地址
+- 角色、菜单、部门、字段按钮添加权限控制
+
+
+# 2.3.0 (2023/5/12)
+
+## ⬆️ chore
+- vue 版本升级 3.2.45 → 3.3.1 ([CHANGELOG](https://github.com/vuejs/core/blob/main/CHANGELOG.md))
+- vite 版本升级 4.3.1 → 4.3.5
+
+## ♻️ refactor
+- 使用 vue 3.3 版本新特性 `defineOptions` 在 `setup` 定义组件名称,移除重复的 `script` 标签
+
+# 2.2.2 (2023/5/11)
+
+## ✨ feat
+- 用户新增提交添加 `vueUse` 的 `useDebounceFn` 函数实现按钮防抖节流
+
+
+# 2.2.1 (2023/4/25)
+
+## 🐛 fix
+- 图标选择器组件使用 `onClickOutside` 未排除下拉弹出框元素导致无法输入搜索。
+
diff --git a/LICENSE b/LICENSE
new file mode 100644
index 0000000..9825cba
--- /dev/null
+++ b/LICENSE
@@ -0,0 +1,21 @@
+MIT License
+
+Copyright (c) 2021-present 有来开源组织
+
+Permission is hereby granted, free of charge, to any person obtaining a copy
+of this software and associated documentation files (the "Software"), to deal
+in the Software without restriction, including without limitation the rights
+to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
+copies of the Software, and to permit persons to whom the Software is
+furnished to do so, subject to the following conditions:
+
+The above copyright notice and this permission notice shall be included in all
+copies or substantial portions of the Software.
+
+THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
+IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
+FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
+AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
+LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
+OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
+SOFTWARE.
diff --git a/README.en-US.md b/README.en-US.md
new file mode 100644
index 0000000..9fb64d6
--- /dev/null
+++ b/README.en-US.md
@@ -0,0 +1,176 @@
+
+
+
vue3-element-admin
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+## Introduction
+
+[vue3-element-admin](https://gitee.com/youlaiorg/vue3-element-admin) is a free and open-source admin template for backend management frontend, built with popular technologies such as Vue3, Vite5, TypeScript, Element-Plus, and Pinia (with accompanying [backend source code](https://gitee.com/youlaiorg/youlai-boot)).
+
+
+
+
+## Project Features
+
+- **Simple and Easy-to-use**: Upgraded version of [vue-element-admin](https://gitee.com/panjiachen/vue-element-admin) for Vue3, with minimal encapsulation and easy to get started.
+
+- **Data Interaction**: Support both local `Mock` data and remote API. Comes with [Java backend source code](https://gitee.com/youlaiorg/youlai-boot) and online API documentation.
+
+- **Permission Management**: Complete permission system for users, roles, menus, dictionaries, and departments.
+
+- **Essential Infrastructure**: Dynamic routing, button permissions, internationalization, code style, Git commit conventions, and common component encapsulation.
+
+- **Continuous Updates**: Since 2021, the project has maintained an open-source status with continuous updates, integrating new tools and dependencies in real time, and has accumulated a broad user base.
+
+## Project Preview
+
+
+
+
+
+
+
+## Project Links
+
+| Project | Gitee | Github | GitCode |
+|----------| ------------------------------------------------------------ | ------------------------------------------------------------ | ------------------------------------------------------------ |
+| Frontend | [vue3-element-admin](https://gitee.com/youlaiorg/vue3-element-admin) | [vue3-element-admin](https://github.com/youlaitech/vue3-element-admin) | [vue3-element-admin](https://gitcode.net/youlai/vue3-element-admin) |
+| Lite | [vue3-element-template](https://gitee.com/youlaiorg/vue3-element-template) | [vue3-element-template](https://github.com/youlaitech/vue3-element-template) |-|
+| Backend | [youlai-boot](https://gitee.com/youlaiorg/youlai-boot) | [youlai-boot](https://github.com/haoxianrui/youlai-boot.git) | [youlai-boot](https://gitcode.net/youlai/youlai-boot) |
+
+## Environment Setup
+
+| Environment | Name and Version | Download Link |
+| -------------------- | :----------------------------------------------------------- | ------------------------------------------------------------ |
+| **Development Tool** | VSCode | [Download](https://code.visualstudio.com/Download) |
+| **Runtime Environment** | Node ≥18 | [Download](http://nodejs.cn/download) |
+
+
+## Project Setup
+
+```bash
+# Clone the repository
+git clone https://gitee.com/youlaiorg/vue3-element-admin.git
+
+# Change directory
+cd vue3-element-admin
+
+# Install pnpm
+npm install pnpm -g
+
+# Install dependencies
+pnpm install
+
+# Start the project
+pnpm run dev
+```
+
+## Project Deployment
+
+```bash
+# Build the project
+pnpm run build
+
+# Upload files to the remote server
+Copy the files generated in the `dist` directory to the `/usr/share/nginx/html` directory.
+
+# nginx.cofig configuration
+server {
+ listen 80;
+ server_name localhost;
+ location / {
+ root /usr/share/nginx/html;
+ index index.html index.htm;
+ }
+ # Reverse proxy configuration
+ location /prod-api/ {
+ proxy_pass http://vapi.youlai.tech/; # Replace vapi.youlai.tech with your backend API address
+ }
+}
+```
+
+## Local Mock
+
+The project supports both online API and local mock API. By default, it uses the online API. If you want to switch to the mock API, modify the value of `VITE_MOCK_DEV_SERVER` in the `.env.development` file to `true`.
+
+## Backend API
+
+> If you have a basic understanding of Java development, follow these steps to convert online API to local backend API and set up a full-stack development environment.
+
+1. Get the backend source code based on `Java` and `SpringBoot` from [youlai-boot](https://gitee.com/youlaiorg/youlai-boot.git).
+2. Follow the instructions in the backend project's README.md to set up the local environment.
+3. Modify the value of `VITE_APP_API_URL` in the `.env.development` file to `http://localhost:8989`, replacing it with the backend API URL.
+
+## Notes
+
+- **Auto import plugin is disabled by default**
+
+ Component type declarations have been automatically generated for the template project. If you add and use new components, follow the instructions in the screenshot to enable automatic generation. After automatic generation is complete, remember to set it back to `false` to avoid conflicts.
+
+ 
+
+- **Blank page when accessing the project**
+
+ Try upgrading your browser, as older browser engines may not support certain new JavaScript syntax, such as optional chaining operator `?.`.
+
+- **Red highlight on project components, functions, and imports**
+
+ Restart VSCode to try again.
+
+- **Other issues**
+
+ If you have any other issues or suggestions, please open an [issue](https://gitee.com/youlaiorg/vue3-element-admin/issues/new).
+
+## Project Documentation
+
+- [Building a Backend Management System from Scratch with Vue3, Vite, TypeScript, and Element-Plus](https://blog.csdn.net/u013737132/article/details/130191394)
+
+- [ESLint+Prettier+Stylelint+EditorConfig for Standardized and Unified Frontend Code Style](https://blog.csdn.net/u013737132/article/details/130190788)
+- [Git Commit Conventions with Husky, Lint-staged, Commitlint, Commitizen, and cz-git](https://blog.csdn.net/u013737132/article/details/130191363)
+
+## Commit Conventions
+
+Execute `pnpm run commit` to invoke interactive git commit and complete the information input and selection according to the prompts.
+
+
+
+## Community 🚀
+
+> **Follow "Youlai Tech" WeChat Official Account to get the QR code for the community.**
+>
+> If the QR code for the community has expired, please add my WeChat (haoxianrui) and indicate whether you are interested in "Frontend", "Backend", or "Full Stack" to get the latest QR code.
+>
+> This measure is taken to ensure the quality of the community and prevent marketing advertising from infiltrating. Thank you for your understanding!
+
+| Official Account | Community |
+|:----:|:----:|
+|  |  |
+
diff --git a/README.md b/README.md
index 81e5aee..a97ac1a 100644
--- a/README.md
+++ b/README.md
@@ -1 +1,204 @@
-#management-web
+
+
+
vue3-element-admin
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+## 项目简介
+
+[vue3-element-admin](https://gitcode.com/youlai/vue3-element-admin) 是基于 Vue3 + Vite5+ TypeScript5 + Element-Plus + Pinia 等主流技术栈构建的免费开源的中后台管理的前端模板(配套[Java 后端源码](https://gitee.com/youlaiorg/youlai-boot))。
+
+
+## 项目特色
+
+- **简洁易用**:基于 [vue-element-admin](https://gitee.com/panjiachen/vue-element-admin) 升级的 Vue3 版本,无过渡封装 ,易上手。
+
+- **数据交互**:同时支持本地 `Mock` 和线上接口,配套 [Java 后端源码](https://gitee.com/youlaiorg/youlai-boot)和[在线接口文档](https://www.apifox.cn/apidoc/shared-195e783f-4d85-4235-a038-eec696de4ea5)。
+
+- **权限管理**:用户、角色、菜单、字典、部门等完善的权限系统功能。
+
+- **基础设施**:动态路由、按钮权限、国际化、代码规范、Git 提交规范、常用组件封装。
+
+- **持续更新**:项目持续开源更新,实时更新工具和依赖。
+
+
+
+## 项目预览
+
+
+
+
+
+
+
+## 项目源码
+
+| 项目 | Gitee | Github | GitCode|
+| ---- | ----| ---- | ---- |
+| 前端 | [vue3-element-admin](https://gitee.com/youlaiorg/vue3-element-admin) | [vue3-element-admin](https://github.com/youlaitech/vue3-element-admin) | [vue3-element-admin](https://gitcode.com/youlai/vue3-element-admin) |
+| 精简版 | [vue3-element-template](https://gitee.com/youlaiorg/vue3-element-template) | [vue3-element-template](https://github.com/youlaitech/vue3-element-template) |-|
+| 后端 | [youlai-boot](https://gitee.com/youlaiorg/youlai-boot) | [youlai-boot](https://github.com/haoxianrui/youlai-boot.git) |[youlai-boot](https://gitcode.com/youlai/youlai-boot.git)|
+
+## 环境准备
+
+| 环境 | 名称版本 | 下载地址 |
+| -------------------- | :----------------------------------------------------------- | ------------------------------------------------------------ |
+| **开发工具** | VSCode | [下载](https://code.visualstudio.com/Download) |
+| **运行环境** | Node ≥18 (其中 20.6.0 版本不可用) | [下载](http://nodejs.cn/download) |
+
+
+## 项目启动
+
+```bash
+# 克隆代码
+git clone https://gitee.com/youlaiorg/vue3-element-admin.git
+
+# 切换目录
+cd vue3-element-admin
+
+# 安装 pnpm
+npm install pnpm -g
+
+# 设置镜像源(可忽略)
+pnpm config set registry https://registry.npmmirror.com
+
+# 安装依赖
+pnpm install
+
+# 启动运行
+pnpm run dev
+```
+
+
+
+## 项目部署
+
+```bash
+# 项目打包
+pnpm run build
+
+# 上传文件至远程服务器
+将本地打包生成的 dist 目录下的所有文件拷贝至服务器的 /usr/share/nginx/html 目录。
+
+# nginx.cofig 配置
+server {
+ listen 80;
+ server_name localhost;
+ location / {
+ root /usr/share/nginx/html;
+ index index.html index.htm;
+ }
+ # 反向代理配置
+ location /prod-api/ {
+ # api.youlai.tech 替换后端API地址,注意保留后面的斜杠 /
+ proxy_pass http://api.youlai.tech/;
+ }
+}
+```
+
+## 本地Mock
+
+项目同时支持在线和本地 Mock 接口,默认使用线上接口,如需替换为 Mock 接口,修改文件 `.env.development` 的 `VITE_MOCK_DEV_SERVER` 为 `true` **即可**。
+
+## 后端接口
+
+> 如果您具备Java开发基础,按照以下步骤将在线接口转为本地后端接口,创建企业级前后端分离开发环境,助您走向全栈之路。
+
+1. 获取基于 `Java` 和 `SpringBoot` 开发的后端 [youlai-boot](https://gitee.com/youlaiorg/youlai-boot.git) 源码。
+2. 根据后端工程的说明文档 [README.md](https://gitee.com/youlaiorg/youlai-boot#%E9%A1%B9%E7%9B%AE%E8%BF%90%E8%A1%8C) 完成本地启动。
+3. 修改 `.env.development` 文件中的 `VITE_APP_API_URL` 的值,将其从 https://api.youlai.tech 更改为 http://localhost:8989 即可。
+
+
+## 注意事项
+
+- **自动导入插件自动生成默认关闭**
+
+ 模板项目的组件类型声明已自动生成。如果添加和使用新的组件,请按照图示方法开启自动生成。在自动生成完成后,记得将其设置为 `false`,避免重复执行引发冲突。
+
+ 
+
+- **项目启动浏览器访问空白**
+
+ 请升级浏览器尝试,低版本浏览器内核可能不支持某些新的 JavaScript 语法,比如可选链操作符 `?.`。
+
+- **项目同步仓库更新升级**
+
+ 项目同步仓库更新升级之后,建议 `pnpm install` 安装更新依赖之后启动 。
+
+- **项目组件、函数和引用爆红**
+
+ 重启 VSCode 尝试
+
+- **其他问题**
+
+ 如果有其他问题或者建议,建议 [ISSUE](https://gitee.com/youlaiorg/vue3-element-admin/issues/new)
+
+
+
+## 项目文档
+
+- [基于 Vue3 + Vite + TypeScript + Element-Plus 从0到1搭建后台管理系统](https://blog.csdn.net/u013737132/article/details/130191394)
+
+- [ESLint+Prettier+Stylelint+EditorConfig 约束和统一前端代码规范](https://blog.csdn.net/u013737132/article/details/130190788)
+- [Husky + Lint-staged + Commitlint + Commitizen + cz-git 配置 Git 提交规范](https://blog.csdn.net/u013737132/article/details/130191363)
+
+
+## 提交规范
+
+执行 `pnpm run commit` 唤起 git commit 交互,根据提示完成信息的输入和选择。
+
+
+
+
+## 项目统计
+
+
+
+
+Thanks to all the contributors!
+
+[](https://github.com/youlaitech/vue3-element-admin/graphs/contributors)
+
+## G-Star
+
+
+
+## 交流群🚀
+
+> **关注「有来技术」公众号,获取交流群二维码。**
+>
+> 如果交流群的二维码过期,请加微信(haoxianrui)并备注「前端」、「后端」或「全栈」以获取最新二维码。
+>
+> 为确保交流群质量,防止营销广告人群混入,我们采取了此措施。望各位理解!
+
+| 公众号 | 交流群 |
+|:----:|:----:|
+|  |  |
+
diff --git a/commitlint.config.cjs b/commitlint.config.cjs
new file mode 100644
index 0000000..4ecb995
--- /dev/null
+++ b/commitlint.config.cjs
@@ -0,0 +1,95 @@
+module.exports = {
+ // 继承的规则
+ extends: ["@commitlint/config-conventional"],
+ // 自定义规则
+ rules: {
+ // @see https://commitlint.js.org/#/reference-rules
+
+ // 提交类型枚举,git提交type必须是以下类型
+ "type-enum": [
+ 2,
+ "always",
+ [
+ "feat", // 新增功能
+ "fix", // 修复缺陷
+ "docs", // 文档变更
+ "style", // 代码格式(不影响功能,例如空格、分号等格式修正)
+ "refactor", // 代码重构(不包括 bug 修复、功能新增)
+ "perf", // 性能优化
+ "test", // 添加疏漏测试或已有测试改动
+ "build", // 构建流程、外部依赖变更(如升级 npm 包、修改 webpack 配置等)
+ "ci", // 修改 CI 配置、脚本
+ "revert", // 回滚 commit
+ "chore", // 对构建过程或辅助工具和库的更改(不影响源文件、测试用例)
+ "wip", // 对构建过程或辅助工具和库的更改(不影响源文件、测试用例)
+ ],
+ ],
+ "subject-case": [0], // subject大小写不做校验
+ },
+
+ prompt: {
+ messages: {
+ type: "选择你要提交的类型 :",
+ scope: "选择一个提交范围(可选):",
+ customScope: "请输入自定义的提交范围 :",
+ subject: "填写简短精炼的变更描述 :\n",
+ body: '填写更加详细的变更描述(可选)。使用 "|" 换行 :\n',
+ breaking: '列举非兼容性重大的变更(可选)。使用 "|" 换行 :\n',
+ footerPrefixesSelect: "选择关联issue前缀(可选):",
+ customFooterPrefix: "输入自定义issue前缀 :",
+ footer: "列举关联issue (可选) 例如: #31, #I3244 :\n",
+ generatingByAI: "正在通过 AI 生成你的提交简短描述...",
+ generatedSelectByAI: "选择一个 AI 生成的简短描述:",
+ confirmCommit: "是否提交或修改commit ?",
+ },
+ // prettier-ignore
+ types: [
+ { value: "feat", name: "特性: ✨ 新增功能", emoji: ":sparkles:" },
+ { value: "fix", name: "修复: 🐛 修复缺陷", emoji: ":bug:" },
+ { value: "docs", name: "文档: 📝 文档变更(更新README文件,或者注释)", emoji: ":memo:" },
+ { value: "style", name: "格式: 🌈 代码格式(空格、格式化、缺失的分号等)", emoji: ":lipstick:" },
+ { value: "refactor", name: "重构: 🔄 代码重构(不修复错误也不添加特性的代码更改)", emoji: ":recycle:" },
+ { value: "perf", name: "性能: 🚀 性能优化", emoji: ":zap:" },
+ { value: "test", name: "测试: 🧪 添加疏漏测试或已有测试改动", emoji: ":white_check_mark:"},
+ { value: "build", name: "构建: 📦️ 构建流程、外部依赖变更(如升级 npm 包、修改 vite 配置等)", emoji: ":package:"},
+ { value: "ci", name: "集成: ⚙️ 修改 CI 配置、脚本", emoji: ":ferris_wheel:"},
+ { value: "revert", name: "回退: ↩️ 回滚 commit",emoji: ":rewind:"},
+ { value: "chore", name: "其他: 🛠️ 对构建过程或辅助工具和库的更改(不影响源文件、测试用例)", emoji: ":hammer:"},
+ { value: "wip", name: "开发中: 🚧 开发阶段临时提交", emoji: ":construction:"},
+ ],
+ useEmoji: true,
+ emojiAlign: "center",
+ useAI: false,
+ aiNumber: 1,
+ themeColorCode: "",
+ scopes: [],
+ allowCustomScopes: true,
+ allowEmptyScopes: true,
+ customScopesAlign: "bottom",
+ customScopesAlias: "custom",
+ emptyScopesAlias: "empty",
+ upperCaseSubject: false,
+ markBreakingChangeMode: false,
+ allowBreakingChanges: ["feat", "fix"],
+ breaklineNumber: 100,
+ breaklineChar: "|",
+ skipQuestions: [],
+ issuePrefixes: [
+ { value: "closed", name: "closed: ISSUES has been processed" },
+ ],
+ customIssuePrefixAlign: "top",
+ emptyIssuePrefixAlias: "skip",
+ customIssuePrefixAlias: "custom",
+ allowCustomIssuePrefix: true,
+ allowEmptyIssuePrefix: true,
+ confirmColorize: true,
+ maxHeaderLength: Infinity,
+ maxSubjectLength: Infinity,
+ minSubjectLength: 0,
+ scopeOverrides: undefined,
+ defaultBody: "",
+ defaultIssues: "",
+ defaultScope: "",
+ defaultSubject: "",
+ },
+};
diff --git a/eslint.config.js b/eslint.config.js
new file mode 100644
index 0000000..f5997a7
--- /dev/null
+++ b/eslint.config.js
@@ -0,0 +1,115 @@
+import globals from "globals";
+import js from "@eslint/js";
+
+// ESLint 核心插件
+import pluginVue from "eslint-plugin-vue";
+import pluginTypeScript from "@typescript-eslint/eslint-plugin";
+
+// Prettier 插件及配置
+import configPrettier from "eslint-config-prettier";
+import pluginPrettier from "eslint-plugin-prettier";
+
+// 解析器
+import * as parserVue from "vue-eslint-parser";
+import * as parserTypeScript from "@typescript-eslint/parser";
+
+// 定义 ESLint 配置
+export default [
+ // 通用 JavaScript 配置
+ {
+ ...js.configs.recommended,
+ ignores: ["**/.*", "dist/*", "*.d.ts", "public/*", "src/assets/**"],
+ languageOptions: {
+ globals: {
+ ...globals.browser, // 浏览器变量 (window, document 等)
+ ...globals.node, // Node.js 变量 (process, require 等)
+ },
+ },
+ plugins: {
+ prettier: pluginPrettier,
+ },
+ rules: {
+ ...configPrettier.rules,
+ ...pluginPrettier.configs.recommended.rules,
+ "no-debug": "off", // 禁止 debugger
+ "prettier/prettier": [
+ "error",
+ {
+ endOfLine: "auto", // 自动识别换行符
+ },
+ ],
+ },
+ },
+
+ // TypeScript 配置
+ {
+ files: ["**/*.?([cm])ts"],
+ languageOptions: {
+ parser: parserTypeScript,
+ parserOptions: {
+ sourceType: "module",
+ },
+ },
+ plugins: {
+ "@typescript-eslint": pluginTypeScript,
+ },
+ rules: {
+ ...pluginTypeScript.configs.strict.rules,
+ "@typescript-eslint/no-explicit-any": "off", // 允许使用 any
+ "@typescript-eslint/no-empty-function": "off", // 允许空函数
+ "@typescript-eslint/no-empty-object-type": "off", // 允许空对象类型
+ "@typescript-eslint/consistent-type-imports": [
+ "error",
+ { disallowTypeAnnotations: false, fixStyle: "inline-type-imports" },
+ ], // 统一类型导入风格
+ },
+ },
+
+ // TypeScript 声明文件的特殊配置
+ {
+ files: ["**/*.d.ts"],
+ rules: {
+ "eslint-comments/no-unlimited-disable": "off",
+ "unused-imports/no-unused-vars": "off",
+ "@typescript-eslint/ban-ts-comment": "off", // 允许使用 @ts-nocheck 注释
+ },
+ },
+
+ // JavaScript (commonjs) 配置
+ {
+ files: ["**/*.?([cm])js"],
+ rules: {
+ "@typescript-eslint/no-var-requires": "off", // 允许 require
+ },
+ },
+
+ // Vue 文件配置
+ {
+ files: ["**/*.vue"],
+ languageOptions: {
+ parser: parserVue,
+ parserOptions: {
+ parser: "@typescript-eslint/parser",
+ sourceType: "module",
+ },
+ },
+ plugins: {
+ vue: pluginVue,
+ },
+ processor: pluginVue.processors[".vue"],
+ rules: {
+ ...pluginVue.configs["vue3-recommended"].rules,
+ "vue/no-v-html": "off", // 允许 v-html
+ "vue/require-default-prop": "off", // 允许没有默认值的 prop
+ "vue/multi-word-component-names": "off", // 关闭组件名称多词要求
+ "vue/html-self-closing": [
+ "error",
+ {
+ html: { void: "always", normal: "always", component: "always" },
+ svg: "always",
+ math: "always",
+ },
+ ], // 自闭合标签
+ },
+ },
+];
diff --git a/index.html b/index.html
new file mode 100644
index 0000000..bd224a6
--- /dev/null
+++ b/index.html
@@ -0,0 +1,61 @@
+
+
+
+
+
+
+
+
+ vue3-element-admin
+
+
+
+
+
+
+
+
+
diff --git a/licenses/vue-element-admin/LICENSE b/licenses/vue-element-admin/LICENSE
new file mode 100644
index 0000000..3fce384
--- /dev/null
+++ b/licenses/vue-element-admin/LICENSE
@@ -0,0 +1,21 @@
+MIT License
+
+Copyright (c) 2017-present PanJiaChen
+
+Permission is hereby granted, free of charge, to any person obtaining a copy
+of this software and associated documentation files (the "Software"), to deal
+in the Software without restriction, including without limitation the rights
+to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
+copies of the Software, and to permit persons to whom the Software is
+furnished to do so, subject to the following conditions:
+
+The above copyright notice and this permission notice shall be included in all
+copies or substantial portions of the Software.
+
+THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
+IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
+FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
+AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
+LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
+OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
+SOFTWARE.
\ No newline at end of file
diff --git a/licenses/vue3-element-admin/LICENSE b/licenses/vue3-element-admin/LICENSE
new file mode 100644
index 0000000..9825cba
--- /dev/null
+++ b/licenses/vue3-element-admin/LICENSE
@@ -0,0 +1,21 @@
+MIT License
+
+Copyright (c) 2021-present 有来开源组织
+
+Permission is hereby granted, free of charge, to any person obtaining a copy
+of this software and associated documentation files (the "Software"), to deal
+in the Software without restriction, including without limitation the rights
+to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
+copies of the Software, and to permit persons to whom the Software is
+furnished to do so, subject to the following conditions:
+
+The above copyright notice and this permission notice shall be included in all
+copies or substantial portions of the Software.
+
+THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
+IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
+FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
+AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
+LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
+OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
+SOFTWARE.
diff --git a/mock/auth.mock.ts b/mock/auth.mock.ts
new file mode 100644
index 0000000..59803e5
--- /dev/null
+++ b/mock/auth.mock.ts
@@ -0,0 +1,43 @@
+import { defineMock } from "./base";
+
+export default defineMock([
+ {
+ url: "auth/captcha",
+ method: ["GET"],
+ body: {
+ code: "00000",
+ data: {
+ captchaKey: "534b8ef2b0a24121bec76391ddd159f9",
+ captchaBase64:
+ "",
+ },
+ msg: "一切ok",
+ },
+ },
+
+ {
+ url: "auth/login",
+ method: ["POST"],
+ body: {
+ code: "00000",
+ data: {
+ accessToken:
+ "eyJ0eXAiOiJKV1QiLCJhbGciOiJIUzI1NiJ9.eyJzdWIiOiJhZG1pbiIsImRlcHRJZCI6MSwiZGF0YVNjb3BlIjoxLCJ1c2VySWQiOjIsImlhdCI6MTcyODE5MzA1MiwiYXV0aG9yaXRpZXMiOlsiUk9MRV9BRE1JTiJdLCJqdGkiOiJhZDg3NzlhZDZlYWY0OWY3OTE4M2ZmYmI5OWM4MjExMSJ9.58YHwL3sNNC22jyAmOZeSm-7MITzfHb_epBIz7LvWeA",
+ tokenType: "Bearer",
+ refreshToken: null,
+ expires: null,
+ },
+ msg: "一切ok",
+ },
+ },
+
+ {
+ url: "auth/logout",
+ method: ["DELETE"],
+ body: {
+ code: "00000",
+ data: {},
+ msg: "string",
+ },
+ },
+]);
diff --git a/mock/base.ts b/mock/base.ts
new file mode 100644
index 0000000..e68e642
--- /dev/null
+++ b/mock/base.ts
@@ -0,0 +1,7 @@
+import path from "path";
+import { createDefineMock } from "vite-plugin-mock-dev-server";
+
+export const defineMock = createDefineMock((mock) => {
+ // 拼接url
+ mock.url = path.join(import.meta.env.VITE_APP_BASE_API + "/api/v1/", mock.url);
+});
diff --git a/mock/dept.mock.ts b/mock/dept.mock.ts
new file mode 100644
index 0000000..e9b6d1b
--- /dev/null
+++ b/mock/dept.mock.ts
@@ -0,0 +1,153 @@
+import { defineMock } from "./base";
+
+export default defineMock([
+ {
+ url: "dept/options",
+ method: ["GET"],
+ body: {
+ code: "00000",
+ data: [
+ {
+ value: 1,
+ label: "有来技术",
+ children: [
+ {
+ value: 2,
+ label: "研发部门",
+ },
+ {
+ value: 3,
+ label: "测试部门",
+ },
+ ],
+ },
+ ],
+ msg: "一切ok",
+ },
+ },
+
+ {
+ url: "dept",
+ method: ["GET"],
+ body: {
+ code: "00000",
+ data: [
+ {
+ id: 1,
+ parentId: 0,
+ name: "有来技术",
+ code: "YOULAI",
+ sort: 1,
+ status: 1,
+ children: [
+ {
+ id: 2,
+ parentId: 1,
+ name: "研发部门",
+ code: "RD001",
+ sort: 1,
+ status: 1,
+ children: [],
+ createTime: null,
+ updateTime: "2022-04-19 12:46",
+ },
+ {
+ id: 3,
+ parentId: 1,
+ name: "测试部门",
+ code: "QA001",
+ sort: 1,
+ status: 1,
+ children: [],
+ createTime: null,
+ updateTime: "2022-04-19 12:46",
+ },
+ ],
+ createTime: null,
+ updateTime: null,
+ },
+ ],
+ msg: "一切ok",
+ },
+ },
+
+ // 新增部门
+ {
+ url: "dept",
+ method: ["POST"],
+ body({ body }) {
+ return {
+ code: "00000",
+ data: null,
+ msg: "新增部门" + body.name + "成功",
+ };
+ },
+ },
+
+ // 获取部门表单数据
+ {
+ url: "dept/:id/form",
+ method: ["GET"],
+ body: ({ params }) => {
+ return {
+ code: "00000",
+ data: deptMap[params.id],
+ msg: "一切ok",
+ };
+ },
+ },
+
+ // 修改部门
+ {
+ url: "dept/:id",
+ method: ["PUT"],
+ body({ body }) {
+ return {
+ code: "00000",
+ data: null,
+ msg: "修改部门" + body.name + "成功",
+ };
+ },
+ },
+
+ // 删除部门
+ {
+ url: "dept/:id",
+ method: ["DELETE"],
+ body({ params }) {
+ return {
+ code: "00000",
+ data: null,
+ msg: "删除部门" + params.id + "成功",
+ };
+ },
+ },
+]);
+
+// 部门映射表数据
+const deptMap: Record = {
+ 1: {
+ id: 1,
+ name: "有来技术",
+ code: "YOULAI",
+ parentId: 0,
+ status: 1,
+ sort: 1,
+ },
+ 2: {
+ id: 2,
+ name: "研发部门",
+ code: "RD001",
+ parentId: 1,
+ status: 1,
+ sort: 1,
+ },
+ 3: {
+ id: 3,
+ name: "测试部门",
+ code: "QA001",
+ parentId: 1,
+ status: 1,
+ sort: 1,
+ },
+};
diff --git a/mock/dict-data.mock.ts b/mock/dict-data.mock.ts
new file mode 100644
index 0000000..9f33f2d
--- /dev/null
+++ b/mock/dict-data.mock.ts
@@ -0,0 +1,216 @@
+import { defineMock } from "./base";
+
+export default defineMock([
+ {
+ url: "dict-data/page",
+ method: ["GET"],
+ body: {
+ code: "00000",
+ data: {
+ list: [
+ {
+ id: 1,
+ dictCode: "gender",
+ label: "男",
+ value: "1",
+ sort: 1,
+ status: 1,
+ },
+ {
+ id: 2,
+ dictCode: "gender",
+ label: "女",
+ value: "2",
+ sort: 2,
+ status: 1,
+ },
+ {
+ id: 3,
+ dictCode: "gender",
+ label: "保密",
+ value: "0",
+ sort: 3,
+ status: 1,
+ },
+ ],
+ total: 3,
+ },
+ msg: "一切ok",
+ },
+ },
+
+ {
+ url: "dict-data/:dictCode/options",
+ method: ["GET"],
+ body: ({ params }) => {
+ const dictCode = params.dictCode;
+
+ let list = null;
+
+ if (dictCode == "gender") {
+ list = [
+ {
+ value: "1",
+ label: "男",
+ },
+ {
+ value: "2",
+ label: "女",
+ },
+ {
+ value: "0",
+ label: "保密",
+ },
+ ];
+ } else if (dictCode == "notice_level") {
+ list = [
+ {
+ value: "L",
+ label: "低",
+ tag: "info",
+ },
+ {
+ value: "M",
+ label: "中",
+ tag: "warning",
+ },
+ {
+ value: "H",
+ label: "高",
+ tag: "danger",
+ },
+ ];
+ } else if (dictCode == "notice_type") {
+ list = [
+ {
+ value: "1",
+ label: "系统升级",
+ tag: "success",
+ },
+ {
+ value: "2",
+ label: "系统维护",
+ tag: "primary",
+ },
+ {
+ value: "3",
+ label: "安全警告",
+ tag: "danger",
+ },
+ {
+ value: "4",
+ label: "假期通知",
+ tag: "success",
+ },
+ {
+ value: "5",
+ label: "公司新闻",
+ tag: "primary",
+ },
+ {
+ value: "99",
+ label: "其他",
+ tag: "info",
+ },
+ ];
+ }
+
+ return {
+ code: "00000",
+ data: list,
+ msg: "一切ok",
+ };
+ },
+ },
+ // 新增字典数据
+ {
+ url: "dict-data",
+ method: ["POST"],
+ body({ body }) {
+ return {
+ code: "00000",
+ data: null,
+ msg: "新增字典" + body.name + "成功",
+ };
+ },
+ },
+
+ // 获取字典数据表单
+ {
+ url: "dict-data/:id/form",
+ method: ["GET"],
+ body: ({ params }) => {
+ return {
+ code: "00000",
+ data: dictMap[params.id],
+ msg: "一切ok",
+ };
+ },
+ },
+
+ // 修改字典数据
+ {
+ url: "dict-data/:id",
+ method: ["PUT"],
+ body({ body }) {
+ return {
+ code: "00000",
+ data: null,
+ msg: "修改字典数据" + body.name + "成功",
+ };
+ },
+ },
+
+ // 删除字典
+ {
+ url: "dict-data/:id",
+ method: ["DELETE"],
+ body({ params }) {
+ return {
+ code: "00000",
+ data: null,
+ msg: "删除字典" + params.id + "成功",
+ };
+ },
+ },
+]);
+
+// 字典数据映射表数据
+const dictMap: Record = {
+ 1: {
+ code: "00000",
+ data: {
+ id: 1,
+ value: "1",
+ label: "男",
+ sort: 1,
+ status: 1,
+ tagType: "primary",
+ },
+ msg: "一切ok",
+ },
+ 2: {
+ code: "00000",
+ data: {
+ id: 2,
+ value: "2",
+ label: "女",
+ sort: 2,
+ status: 1,
+ tagType: "danger",
+ },
+ msg: "一切ok",
+ },
+ 3: {
+ code: "00000",
+ data: {
+ id: 3,
+ value: "0",
+ label: "保密",
+ sort: 3,
+ status: 1,
+ tagType: "info",
+ },
+ msg: "一切ok",
+ },
+};
diff --git a/mock/dict.mock.ts b/mock/dict.mock.ts
new file mode 100644
index 0000000..5f3e5f2
--- /dev/null
+++ b/mock/dict.mock.ts
@@ -0,0 +1,181 @@
+import { defineMock } from "./base";
+
+export default defineMock([
+ {
+ url: "dict/page",
+ method: ["GET"],
+ body: {
+ code: "00000",
+ data: {
+ list: [
+ {
+ id: 1,
+ name: "性别",
+ dictCode: "gender",
+ status: 1,
+ },
+ ],
+ total: 1,
+ },
+ msg: "一切ok",
+ },
+ },
+
+ // 新增字典
+ {
+ url: "dict",
+ method: ["POST"],
+ body({ body }) {
+ return {
+ code: "00000",
+ data: null,
+ msg: "新增字典" + body.name + "成功",
+ };
+ },
+ },
+
+ // 获取字典表单数据
+ {
+ url: "dict/:id/form",
+ method: ["GET"],
+ body: ({ params }) => {
+ return {
+ code: "00000",
+ data: dictMap[params.id],
+ msg: "一切ok",
+ };
+ },
+ },
+
+ // 修改字典
+ {
+ url: "dict/:id",
+ method: ["PUT"],
+ body({ body }) {
+ return {
+ code: "00000",
+ data: null,
+ msg: "修改字典" + body.name + "成功",
+ };
+ },
+ },
+
+ // 删除字典
+ {
+ url: "dict/:id",
+ method: ["DELETE"],
+ body({ params }) {
+ return {
+ code: "00000",
+ data: null,
+ msg: "删除字典" + params.id + "成功",
+ };
+ },
+ },
+
+ // 所有字典列表
+ {
+ url: "dict/list",
+ method: ["GET"],
+ body() {
+ return {
+ code: "00000",
+ data: [
+ {
+ name: "通知级别",
+ dictCode: "notice_level",
+ dictDataList: [
+ {
+ value: "L",
+ label: "低",
+ tagType: "info",
+ },
+ {
+ value: "M",
+ label: "中",
+ tagType: "warning",
+ },
+ {
+ value: "H",
+ label: "高",
+ tagType: "danger",
+ },
+ ],
+ },
+ {
+ name: "通知类型",
+ dictCode: "notice_type",
+ dictDataList: [
+ {
+ value: "1",
+ label: "系统升级",
+ tagType: "success",
+ },
+ {
+ value: "2",
+ label: "系统维护",
+ tagType: "primary",
+ },
+ {
+ value: "3",
+ label: "安全警告",
+ tagType: "danger",
+ },
+ {
+ value: "4",
+ label: "假期通知",
+ tagType: "success",
+ },
+ {
+ value: "5",
+ label: "公司新闻",
+ tagType: "primary",
+ },
+ {
+ value: "99",
+ label: "其他",
+ tagType: "info",
+ },
+ ],
+ },
+ {
+ name: "性别",
+ dictCode: "gender",
+ dictDataList: [
+ {
+ value: "1",
+ label: "男",
+ tagType: "primary",
+ },
+ {
+ value: "2",
+ label: "女",
+ tagType: "danger",
+ },
+ {
+ value: "0",
+ label: "保密",
+ tagType: "info",
+ },
+ ],
+ },
+ ],
+ msg: "一切ok",
+ };
+ },
+ },
+]);
+
+// 字典映射表数据
+const dictMap: Record = {
+ 1: {
+ code: "00000",
+ data: {
+ id: 1,
+ name: "性别",
+ dictCode: "gender",
+ status: 1,
+ },
+ msg: "一切ok",
+ },
+};
diff --git a/mock/log.mock.ts b/mock/log.mock.ts
new file mode 100644
index 0000000..b75963e
--- /dev/null
+++ b/mock/log.mock.ts
@@ -0,0 +1,207 @@
+import { defineMock } from "./base";
+
+export default defineMock([
+ {
+ url: "logs/page",
+ method: ["GET"],
+ body: {
+ code: "00000",
+ data: {
+ list: [
+ {
+ id: 36192,
+ module: "菜单",
+ content: "菜单列表",
+ requestUri: "/api/v1/menus",
+ method: null,
+ ip: "183.156.148.241",
+ region: "浙江省 杭州市",
+ browser: "Chrome 109.0.0.0",
+ os: "OSX",
+ executionTime: 5,
+ createBy: null,
+ createTime: "2024-07-07 20:38:47",
+ operator: "系统管理员",
+ },
+ {
+ id: 36190,
+ module: "字典",
+ content: "字典分页列表",
+ requestUri: "/api/v1/dict/page",
+ method: null,
+ ip: "183.156.148.241",
+ region: "浙江省 杭州市",
+ browser: "Chrome 109.0.0.0",
+ os: "OSX",
+ executionTime: 9,
+ createBy: null,
+ createTime: "2024-07-07 20:38:45",
+ operator: "系统管理员",
+ },
+ {
+ id: 36193,
+ module: "部门",
+ content: "部门列表",
+ requestUri: "/api/v1/dept",
+ method: null,
+ ip: "192.168.31.134",
+ region: "0 内网IP",
+ browser: "Chrome 125.0.0.0",
+ os: "Windows 10 or Windows Server 2016",
+ executionTime: 27,
+ createBy: null,
+ createTime: "2024-07-07 20:38:45",
+ operator: "系统管理员",
+ },
+ {
+ id: 36191,
+ module: "菜单",
+ content: "菜单列表",
+ requestUri: "/api/v1/menus",
+ method: null,
+ ip: "192.168.31.134",
+ region: "0 内网IP",
+ browser: "Chrome 125.0.0.0",
+ os: "Windows 10 or Windows Server 2016",
+ executionTime: 39,
+ createBy: null,
+ createTime: "2024-07-07 20:38:44",
+ operator: "系统管理员",
+ },
+ {
+ id: 36189,
+ module: "角色",
+ content: "角色分页列表",
+ requestUri: "/api/v1/roles/page",
+ method: null,
+ ip: "192.168.31.134",
+ region: "0 内网IP",
+ browser: "Chrome 125.0.0.0",
+ os: "Windows 10 or Windows Server 2016",
+ executionTime: 55,
+ createBy: null,
+ createTime: "2024-07-07 20:38:43",
+ operator: "系统管理员",
+ },
+ {
+ id: 36188,
+ module: "用户",
+ content: "用户分页列表",
+ requestUri: "/api/v1/users/page",
+ method: null,
+ ip: "192.168.31.134",
+ region: "0 内网IP",
+ browser: "Chrome 125.0.0.0",
+ os: "Windows 10 or Windows Server 2016",
+ executionTime: 92,
+ createBy: null,
+ createTime: "2024-07-07 20:38:42",
+ operator: "系统管理员",
+ },
+ {
+ id: 36187,
+ module: "登录",
+ content: "登录",
+ requestUri: "/api/v1/auth/login",
+ method: null,
+ ip: "192.168.31.134",
+ region: "0 内网IP",
+ browser: "Chrome 125.0.0.0",
+ os: "Windows 10 or Windows Server 2016",
+ executionTime: 19340,
+ createBy: null,
+ createTime: "2024-07-07 20:38:09",
+ operator: "系统管理员",
+ },
+ {
+ id: 36186,
+ module: "登录",
+ content: "登录",
+ requestUri: "/api/v1/auth/login",
+ method: null,
+ ip: "192.168.31.134",
+ region: "0 内网IP",
+ browser: "Chrome 125.0.0.0",
+ os: "Windows 10 or Windows Server 2016",
+ executionTime: 19869,
+ createBy: null,
+ createTime: "2024-07-07 20:37:59",
+ operator: "系统管理员",
+ },
+ {
+ id: 36185,
+ module: "登录",
+ content: "登录",
+ requestUri: "/api/v1/auth/login",
+ method: null,
+ ip: "112.103.111.59",
+ region: "黑龙江省 哈尔滨市",
+ browser: "Chrome 97.0.4692.98",
+ os: "Android",
+ executionTime: 96,
+ createBy: null,
+ createTime: "2024-07-07 20:37:21",
+ operator: "系统管理员",
+ },
+ {
+ id: 36184,
+ module: "登录",
+ content: "登录",
+ requestUri: "/api/v1/auth/login",
+ method: null,
+ ip: "114.86.204.190",
+ region: "上海 上海市",
+ browser: "Chrome 125.0.0.0",
+ os: "Windows 10 or Windows Server 2016",
+ executionTime: 89,
+ createBy: null,
+ createTime: "2024-07-07 20:29:37",
+ operator: "系统管理员",
+ },
+ ],
+ total: 36188,
+ },
+ msg: "一切ok",
+ },
+ },
+ {
+ url: "logs/visit-trend",
+ method: ["GET"],
+ body: {
+ code: "00000",
+ data: {
+ dates: [
+ "2024-06-30",
+ "2024-07-01",
+ "2024-07-02",
+ "2024-07-03",
+ "2024-07-04",
+ "2024-07-05",
+ "2024-07-06",
+ "2024-07-07",
+ ],
+ pvList: [1751, 5168, 4882, 5301, 4721, 4885, 1901, 1003],
+ uvList: null,
+ ipList: [207, 566, 565, 631, 579, 496, 222, 152],
+ },
+ msg: "一切ok",
+ },
+ },
+
+ {
+ url: "logs/visit-stats",
+ method: ["GET"],
+ body: {
+ code: "00000",
+ data: {
+ todayPvCount: 1629,
+ totalPvCount: 286086,
+ pvGrowthRate: -0.65,
+ todayIpCount: 169,
+ totalIpCount: 19985,
+ ipGrowthRate: -0.57,
+ },
+ msg: "一切ok",
+ },
+ },
+]);
diff --git a/mock/menu.mock.ts b/mock/menu.mock.ts
new file mode 100644
index 0000000..9d3840d
--- /dev/null
+++ b/mock/menu.mock.ts
@@ -0,0 +1,1711 @@
+import { defineMock } from "./base";
+
+export default defineMock([
+ {
+ url: "menus/routes",
+ method: ["GET"],
+ body: {
+ code: "00000",
+ data: [
+ {
+ path: "/system",
+ component: "Layout",
+ redirect: "/system/user",
+ name: "/system",
+ meta: {
+ title: "系统管理",
+ icon: "system",
+ hidden: false,
+ alwaysShow: false,
+ params: null,
+ },
+ children: [
+ {
+ path: "user",
+ component: "system/user/index",
+ name: "User",
+ meta: {
+ title: "用户管理",
+ icon: "el-icon-User",
+ hidden: false,
+ keepAlive: true,
+ alwaysShow: false,
+ params: null,
+ },
+ },
+ {
+ path: "role",
+ component: "system/role/index",
+ name: "Role",
+ meta: {
+ title: "角色管理",
+ icon: "role",
+ hidden: false,
+ keepAlive: true,
+ alwaysShow: false,
+ params: null,
+ },
+ },
+ {
+ path: "menu",
+ component: "system/menu/index",
+ name: "SysMenu",
+ meta: {
+ title: "菜单管理",
+ icon: "menu",
+ hidden: false,
+ keepAlive: true,
+ alwaysShow: false,
+ params: null,
+ },
+ },
+ {
+ path: "dept",
+ component: "system/dept/index",
+ name: "Dept",
+ meta: {
+ title: "部门管理",
+ icon: "tree",
+ hidden: false,
+ keepAlive: true,
+ alwaysShow: false,
+ params: null,
+ },
+ },
+ {
+ path: "dict",
+ component: "system/dict/index",
+ name: "Dict",
+ meta: {
+ title: "字典管理",
+ icon: "dict",
+ hidden: false,
+ keepAlive: true,
+ alwaysShow: false,
+ params: null,
+ },
+ },
+ {
+ path: "log",
+ component: "system/log/index",
+ name: "Log",
+ meta: {
+ title: "系统日志",
+ icon: "document",
+ hidden: false,
+ keepAlive: true,
+ alwaysShow: false,
+ params: null,
+ },
+ },
+ {
+ path: "dict-data",
+ component: "system/dict/data",
+ name: "DictData",
+ meta: {
+ title: "字典数据",
+ icon: "",
+ hidden: true,
+ keepAlive: true,
+ alwaysShow: false,
+ params: null,
+ },
+ },
+ {
+ path: "config",
+ component: "system/config/index",
+ name: "Config",
+ meta: {
+ title: "系统配置",
+ icon: "setting",
+ hidden: false,
+ keepAlive: true,
+ alwaysShow: false,
+ params: null,
+ },
+ },
+ {
+ path: "notice",
+ component: "system/notice/index",
+ name: "Notice",
+ meta: {
+ title: "通知公告",
+ icon: "",
+ hidden: false,
+ alwaysShow: false,
+ params: null,
+ },
+ },
+ ],
+ },
+ {
+ path: "/codegen",
+ component: "Layout",
+ name: "/codegen",
+ meta: {
+ title: "系统工具",
+ icon: "menu",
+ hidden: false,
+ alwaysShow: false,
+ params: null,
+ },
+ children: [
+ {
+ path: "codegen",
+ component: "codegen/index",
+ name: "Codegen",
+ meta: {
+ title: "代码生成",
+ icon: "code",
+ hidden: false,
+ keepAlive: true,
+ alwaysShow: false,
+ params: null,
+ },
+ },
+ ],
+ },
+ {
+ path: "/api",
+ component: "Layout",
+ name: "/api",
+ meta: {
+ title: "接口文档",
+ icon: "api",
+ hidden: false,
+ alwaysShow: true,
+ params: null,
+ },
+ children: [
+ {
+ path: "apifox",
+ component: "demo/api/apifox",
+ name: "Apifox",
+ meta: {
+ title: "Apifox",
+ icon: "api",
+ hidden: false,
+ keepAlive: true,
+ alwaysShow: false,
+ params: null,
+ },
+ },
+ ],
+ },
+ {
+ path: "/doc",
+ component: "Layout",
+ redirect: "https://juejin.cn/post/7228990409909108793",
+ name: "/doc",
+ meta: {
+ title: "平台文档",
+ icon: "document",
+ hidden: false,
+ alwaysShow: false,
+ params: null,
+ },
+ children: [
+ {
+ path: "internal-doc",
+ component: "demo/internal-doc",
+ name: "InternalDoc",
+ meta: {
+ title: "document",
+ icon: "document",
+ hidden: false,
+ alwaysShow: false,
+ params: null,
+ },
+ },
+ {
+ path: "https://juejin.cn/post/7228990409909108793",
+ name: "Https://juejin.cn/post/7228990409909108793",
+ meta: {
+ title: "平台文档(外链)",
+ icon: "link",
+ hidden: false,
+ alwaysShow: false,
+ params: null,
+ },
+ },
+ ],
+ },
+ {
+ path: "/multi-level",
+ component: "Layout",
+ name: "/multiLevel",
+ meta: {
+ title: "多级菜单",
+ icon: "cascader",
+ hidden: false,
+ alwaysShow: true,
+ params: null,
+ },
+ children: [
+ {
+ path: "multi-level1",
+ component: "demo/multi-level/level1",
+ name: "MultiLevel1",
+ meta: {
+ title: "菜单一级",
+ icon: "",
+ hidden: false,
+ alwaysShow: true,
+ params: null,
+ },
+ children: [
+ {
+ path: "multi-level2",
+ component: "demo/multi-level/children/level2",
+ name: "MultiLevel2",
+ meta: {
+ title: "菜单二级",
+ icon: "",
+ hidden: false,
+ alwaysShow: false,
+ params: null,
+ },
+ children: [
+ {
+ path: "multi-level3-1",
+ component: "demo/multi-level/children/children/level3-1",
+ name: "MultiLevel31",
+ meta: {
+ title: "菜单三级-1",
+ icon: "",
+ hidden: false,
+ keepAlive: true,
+ alwaysShow: false,
+ params: null,
+ },
+ },
+ {
+ path: "multi-level3-2",
+ component: "demo/multi-level/children/children/level3-2",
+ name: "MultiLevel32",
+ meta: {
+ title: "菜单三级-2",
+ icon: "",
+ hidden: false,
+ keepAlive: true,
+ alwaysShow: false,
+ params: null,
+ },
+ },
+ ],
+ },
+ ],
+ },
+ ],
+ },
+ {
+ path: "/component",
+ component: "Layout",
+ name: "/component",
+ meta: {
+ title: "组件封装",
+ icon: "menu",
+ hidden: false,
+ alwaysShow: false,
+ params: null,
+ },
+ children: [
+ {
+ path: "curd",
+ component: "demo/curd/index",
+ name: "Curd",
+ meta: {
+ title: "增删改查",
+ icon: "",
+ hidden: false,
+ keepAlive: true,
+ alwaysShow: false,
+ params: null,
+ },
+ },
+ {
+ path: "table-select",
+ component: "demo/table-select/index",
+ name: "TableSelect",
+ meta: {
+ title: "列表选择器",
+ icon: "",
+ hidden: false,
+ keepAlive: true,
+ alwaysShow: false,
+ params: null,
+ },
+ },
+ {
+ path: "wang-editor",
+ component: "demo/wang-editor",
+ name: "WangEditor",
+ meta: {
+ title: "富文本编辑器",
+ icon: "",
+ hidden: false,
+ keepAlive: true,
+ alwaysShow: false,
+ params: null,
+ },
+ },
+ {
+ path: "upload",
+ component: "demo/upload",
+ name: "Upload",
+ meta: {
+ title: "图片上传",
+ icon: "",
+ hidden: false,
+ keepAlive: true,
+ alwaysShow: false,
+ params: null,
+ },
+ },
+ {
+ path: "dict-demo",
+ component: "demo/dictionary",
+ name: "DictDemo",
+ meta: {
+ title: "字典组件",
+ icon: "",
+ hidden: false,
+ keepAlive: true,
+ alwaysShow: false,
+ params: null,
+ },
+ },
+ {
+ path: "icon-selector",
+ component: "demo/icon-selector",
+ name: "IconSelector",
+ meta: {
+ title: "图标选择器",
+ icon: "",
+ hidden: false,
+ keepAlive: true,
+ alwaysShow: false,
+ params: null,
+ },
+ },
+ ],
+ },
+ {
+ path: "/route-param",
+ component: "Layout",
+ name: "/routeParam",
+ meta: {
+ title: "路由参数",
+ icon: "el-icon-ElementPlus",
+ hidden: false,
+ alwaysShow: true,
+ params: null,
+ },
+ children: [
+ {
+ path: "route-param-type1",
+ component: "demo/route-param",
+ name: "RouteParamType1",
+ meta: {
+ title: "参数(type=1)",
+ icon: "el-icon-Star",
+ hidden: false,
+ keepAlive: true,
+ alwaysShow: false,
+ params: {
+ type: "1",
+ },
+ },
+ },
+ {
+ path: "route-param-type2",
+ component: "demo/route-param",
+ name: "RouteParamType2",
+ meta: {
+ title: "参数(type=2)",
+ icon: "el-icon-StarFilled",
+ hidden: false,
+ keepAlive: true,
+ alwaysShow: false,
+ params: {
+ type: "2",
+ },
+ },
+ },
+ ],
+ },
+ {
+ path: "/function",
+ component: "Layout",
+ name: "/function",
+ meta: {
+ title: "功能演示",
+ icon: "menu",
+ hidden: false,
+ alwaysShow: false,
+ params: null,
+ },
+ children: [
+ {
+ path: "icon-demo",
+ component: "demo/icons",
+ name: "IconDemo",
+ meta: {
+ title: "Icons",
+ icon: "el-icon-Notification",
+ hidden: false,
+ keepAlive: true,
+ alwaysShow: false,
+ params: null,
+ },
+ },
+ {
+ path: "/function/websocket",
+ component: "demo/websocket",
+ name: "/function/websocket",
+ meta: {
+ title: "Websocket",
+ icon: "",
+ hidden: false,
+ keepAlive: true,
+ alwaysShow: false,
+ params: null,
+ },
+ },
+ {
+ path: "other/:id",
+ component: "demo/other",
+ name: "Other/:id",
+ meta: {
+ title: "敬请期待...",
+ icon: "",
+ hidden: false,
+ alwaysShow: false,
+ params: null,
+ },
+ },
+ ],
+ },
+ ],
+ msg: "一切ok",
+ },
+ },
+
+ // 获取菜单树形列表
+ {
+ url: "menus",
+ method: ["GET"],
+ body: {
+ code: "00000",
+ data: [
+ {
+ id: 1,
+ parentId: 0,
+ name: "系统管理",
+ type: "CATALOG",
+ routeName: "",
+ routePath: "/system",
+ component: "Layout",
+ sort: 1,
+ visible: 1,
+ icon: "system",
+ redirect: "/system/user",
+ perm: null,
+ children: [
+ {
+ id: 2,
+ parentId: 1,
+ name: "用户管理",
+ type: "MENU",
+ routeName: "User",
+ routePath: "user",
+ component: "system/user/index",
+ sort: 1,
+ visible: 1,
+ icon: "el-icon-User",
+ redirect: null,
+ perm: null,
+ children: [
+ {
+ id: 105,
+ parentId: 2,
+ name: "用户查询",
+ type: "BUTTON",
+ routeName: null,
+ routePath: "",
+ component: null,
+ sort: 0,
+ visible: 1,
+ icon: "",
+ redirect: null,
+ perm: "sys:user:query",
+ children: [],
+ },
+ {
+ id: 31,
+ parentId: 2,
+ name: "用户新增",
+ type: "BUTTON",
+ routeName: null,
+ routePath: "",
+ component: null,
+ sort: 1,
+ visible: 1,
+ icon: "",
+ redirect: "",
+ perm: "sys:user:add",
+ children: [],
+ },
+ {
+ id: 32,
+ parentId: 2,
+ name: "用户编辑",
+ type: "BUTTON",
+ routeName: null,
+ routePath: "",
+ component: null,
+ sort: 2,
+ visible: 1,
+ icon: "",
+ redirect: "",
+ perm: "sys:user:edit",
+ children: [],
+ },
+ {
+ id: 33,
+ parentId: 2,
+ name: "用户删除",
+ type: "BUTTON",
+ routeName: null,
+ routePath: "",
+ component: null,
+ sort: 3,
+ visible: 1,
+ icon: "",
+ redirect: "",
+ perm: "sys:user:delete",
+ children: [],
+ },
+ {
+ id: 88,
+ parentId: 2,
+ name: "重置密码",
+ type: "BUTTON",
+ routeName: null,
+ routePath: "",
+ component: null,
+ sort: 4,
+ visible: 1,
+ icon: "",
+ redirect: null,
+ perm: "sys:user:password:reset",
+ children: [],
+ },
+ {
+ id: 106,
+ parentId: 2,
+ name: "用户导入",
+ type: "BUTTON",
+ routeName: null,
+ routePath: "",
+ component: null,
+ sort: 5,
+ visible: 1,
+ icon: "",
+ redirect: null,
+ perm: "sys:user:import",
+ children: [],
+ },
+ {
+ id: 107,
+ parentId: 2,
+ name: "用户导出",
+ type: "BUTTON",
+ routeName: null,
+ routePath: "",
+ component: null,
+ sort: 6,
+ visible: 1,
+ icon: "",
+ redirect: null,
+ perm: "sys:user:export",
+ children: [],
+ },
+ ],
+ },
+ {
+ id: 3,
+ parentId: 1,
+ name: "角色管理",
+ type: "MENU",
+ routeName: "Role",
+ routePath: "role",
+ component: "system/role/index",
+ sort: 2,
+ visible: 1,
+ icon: "role",
+ redirect: null,
+ perm: null,
+ children: [
+ {
+ id: 70,
+ parentId: 3,
+ name: "角色新增",
+ type: "BUTTON",
+ routeName: null,
+ routePath: "",
+ component: null,
+ sort: 1,
+ visible: 1,
+ icon: "",
+ redirect: null,
+ perm: "sys:role:add",
+ children: [],
+ },
+ {
+ id: 71,
+ parentId: 3,
+ name: "角色编辑",
+ type: "BUTTON",
+ routeName: null,
+ routePath: "",
+ component: null,
+ sort: 2,
+ visible: 1,
+ icon: "",
+ redirect: null,
+ perm: "sys:role:edit",
+ children: [],
+ },
+ {
+ id: 72,
+ parentId: 3,
+ name: "角色删除",
+ type: "BUTTON",
+ routeName: null,
+ routePath: "",
+ component: null,
+ sort: 3,
+ visible: 1,
+ icon: "",
+ redirect: null,
+ perm: "sys:role:delete",
+ children: [],
+ },
+ ],
+ },
+ {
+ id: 4,
+ parentId: 1,
+ name: "菜单管理",
+ type: "MENU",
+ routeName: "Menu",
+ routePath: "menu",
+ component: "system/menu/index",
+ sort: 3,
+ visible: 1,
+ icon: "menu",
+ redirect: null,
+ perm: null,
+ children: [
+ {
+ id: 73,
+ parentId: 4,
+ name: "菜单新增",
+ type: "BUTTON",
+ routeName: null,
+ routePath: "",
+ component: null,
+ sort: 1,
+ visible: 1,
+ icon: "",
+ redirect: null,
+ perm: "sys:menu:add",
+ children: [],
+ },
+ {
+ id: 74,
+ parentId: 4,
+ name: "菜单编辑",
+ type: "BUTTON",
+ routeName: null,
+ routePath: "",
+ component: null,
+ sort: 3,
+ visible: 1,
+ icon: "",
+ redirect: null,
+ perm: "sys:menu:edit",
+ children: [],
+ },
+ {
+ id: 75,
+ parentId: 4,
+ name: "菜单删除",
+ type: "BUTTON",
+ routeName: null,
+ routePath: "",
+ component: null,
+ sort: 3,
+ visible: 1,
+ icon: "",
+ redirect: null,
+ perm: "sys:menu:delete",
+ children: [],
+ },
+ ],
+ },
+ {
+ id: 5,
+ parentId: 1,
+ name: "部门管理",
+ type: "MENU",
+ routeName: "Dept",
+ routePath: "dept",
+ component: "system/dept/index",
+ sort: 4,
+ visible: 1,
+ icon: "tree",
+ redirect: null,
+ perm: null,
+ children: [
+ {
+ id: 76,
+ parentId: 5,
+ name: "部门新增",
+ type: "BUTTON",
+ routeName: null,
+ routePath: "",
+ component: null,
+ sort: 1,
+ visible: 1,
+ icon: "",
+ redirect: null,
+ perm: "sys:dept:add",
+ children: [],
+ },
+ {
+ id: 77,
+ parentId: 5,
+ name: "部门编辑",
+ type: "BUTTON",
+ routeName: null,
+ routePath: "",
+ component: null,
+ sort: 2,
+ visible: 1,
+ icon: "",
+ redirect: null,
+ perm: "sys:dept:edit",
+ children: [],
+ },
+ {
+ id: 78,
+ parentId: 5,
+ name: "部门删除",
+ type: "BUTTON",
+ routeName: null,
+ routePath: "",
+ component: null,
+ sort: 3,
+ visible: 1,
+ icon: "",
+ redirect: null,
+ perm: "sys:dept:delete",
+ children: [],
+ },
+ ],
+ },
+ {
+ id: 6,
+ parentId: 1,
+ name: "字典管理",
+ type: "MENU",
+ routeName: "Dict",
+ routePath: "dict",
+ component: "system/dict/index",
+ sort: 5,
+ visible: 1,
+ icon: "dict",
+ redirect: null,
+ perm: null,
+ children: [
+ {
+ id: 79,
+ parentId: 6,
+ name: "字典新增",
+ type: "BUTTON",
+ routeName: null,
+ routePath: "",
+ component: null,
+ sort: 1,
+ visible: 1,
+ icon: "",
+ redirect: null,
+ perm: "sys:dict:add",
+ children: [],
+ },
+ {
+ id: 81,
+ parentId: 6,
+ name: "字典编辑",
+ type: "BUTTON",
+ routeName: null,
+ routePath: "",
+ component: null,
+ sort: 2,
+ visible: 1,
+ icon: "",
+ redirect: null,
+ perm: "sys:dict_type:edit",
+ children: [],
+ },
+ {
+ id: 84,
+ parentId: 6,
+ name: "字典删除",
+ type: "BUTTON",
+ routeName: null,
+ routePath: "",
+ component: null,
+ sort: 3,
+ visible: 1,
+ icon: "",
+ redirect: null,
+ perm: "sys:dict_type:delete",
+ children: [],
+ },
+ ],
+ },
+ {
+ id: 135,
+ parentId: 1,
+ name: "字典数据",
+ type: "MENU",
+ routeName: "DictData",
+ routePath: "dict-data",
+ component: "system/dict/data",
+ sort: 6,
+ visible: 0,
+ icon: "",
+ redirect: null,
+ perm: null,
+ children: [
+ {
+ id: 136,
+ parentId: 135,
+ name: "字典数据新增",
+ type: "BUTTON",
+ routeName: null,
+ routePath: "",
+ component: null,
+ sort: 4,
+ visible: 1,
+ icon: "",
+ redirect: null,
+ perm: "sys:dict-data:add",
+ children: [],
+ },
+ {
+ id: 137,
+ parentId: 135,
+ name: "字典数据编辑",
+ type: "BUTTON",
+ routeName: null,
+ routePath: "",
+ component: null,
+ sort: 5,
+ visible: 1,
+ icon: "",
+ redirect: null,
+ perm: "sys:dict-data:edit",
+ children: [],
+ },
+ {
+ id: 138,
+ parentId: 135,
+ name: "字典数据删除",
+ type: "BUTTON",
+ routeName: null,
+ routePath: "",
+ component: null,
+ sort: 6,
+ visible: 1,
+ icon: "",
+ redirect: null,
+ perm: "sys:dict-data:delete",
+ children: [],
+ },
+ ],
+ },
+ {
+ id: 117,
+ parentId: 1,
+ name: "系统日志",
+ type: "MENU",
+ routeName: "Log",
+ routePath: "log",
+ component: "system/log/index",
+ sort: 6,
+ visible: 1,
+ icon: "document",
+ redirect: null,
+ perm: null,
+ children: [],
+ },
+ {
+ id: 120,
+ parentId: 1,
+ name: "系统配置",
+ type: "MENU",
+ routeName: "Config",
+ routePath: "config",
+ component: "system/config/index",
+ sort: 7,
+ visible: 1,
+ icon: "setting",
+ redirect: null,
+ perm: null,
+ children: [
+ {
+ id: 121,
+ parentId: 120,
+ name: "查询系统配置",
+ type: "BUTTON",
+ routeName: null,
+ routePath: "",
+ component: null,
+ sort: 1,
+ visible: 1,
+ icon: "",
+ redirect: null,
+ perm: "sys:config:query",
+ children: [],
+ },
+ {
+ id: 122,
+ parentId: 120,
+ name: "新增系统配置",
+ type: "BUTTON",
+ routeName: null,
+ routePath: "",
+ component: null,
+ sort: 2,
+ visible: 1,
+ icon: "",
+ redirect: null,
+ perm: "sys:config:add",
+ children: [],
+ },
+ {
+ id: 123,
+ parentId: 120,
+ name: "修改系统配置",
+ type: "BUTTON",
+ routeName: null,
+ routePath: "",
+ component: null,
+ sort: 3,
+ visible: 1,
+ icon: "",
+ redirect: null,
+ perm: "sys:config:update",
+ children: [],
+ },
+ {
+ id: 124,
+ parentId: 120,
+ name: "删除系统配置",
+ type: "BUTTON",
+ routeName: null,
+ routePath: "",
+ component: null,
+ sort: 4,
+ visible: 1,
+ icon: "",
+ redirect: null,
+ perm: "sys:config:delete",
+ children: [],
+ },
+ {
+ id: 125,
+ parentId: 120,
+ name: "刷新系统配置",
+ type: "BUTTON",
+ routeName: null,
+ routePath: "",
+ component: null,
+ sort: 5,
+ visible: 1,
+ icon: "",
+ redirect: null,
+ perm: "sys:config:refresh",
+ children: [],
+ },
+ ],
+ },
+ {
+ id: 126,
+ parentId: 1,
+ name: "通知公告",
+ type: "MENU",
+ routeName: "Notice",
+ routePath: "notice",
+ component: "system/notice/index",
+ sort: 9,
+ visible: 1,
+ icon: "",
+ redirect: null,
+ perm: null,
+ children: [
+ {
+ id: 127,
+ parentId: 126,
+ name: "查询",
+ type: "BUTTON",
+ routeName: null,
+ routePath: "",
+ component: null,
+ sort: 1,
+ visible: 1,
+ icon: "",
+ redirect: null,
+ perm: "sys:notice:query",
+ children: [],
+ },
+ {
+ id: 128,
+ parentId: 126,
+ name: "新增",
+ type: "BUTTON",
+ routeName: null,
+ routePath: "",
+ component: null,
+ sort: 2,
+ visible: 1,
+ icon: "",
+ redirect: null,
+ perm: "sys:notice:add",
+ children: [],
+ },
+ {
+ id: 129,
+ parentId: 126,
+ name: "编辑",
+ type: "BUTTON",
+ routeName: null,
+ routePath: "",
+ component: null,
+ sort: 3,
+ visible: 1,
+ icon: "",
+ redirect: null,
+ perm: "sys:notice:edit",
+ children: [],
+ },
+ {
+ id: 130,
+ parentId: 126,
+ name: "删除",
+ type: "BUTTON",
+ routeName: null,
+ routePath: "",
+ component: null,
+ sort: 4,
+ visible: 1,
+ icon: "",
+ redirect: null,
+ perm: "sys:notice:delete",
+ children: [],
+ },
+ {
+ id: 133,
+ parentId: 126,
+ name: "发布",
+ type: "BUTTON",
+ routeName: null,
+ routePath: "",
+ component: null,
+ sort: 5,
+ visible: 1,
+ icon: "",
+ redirect: null,
+ perm: "sys:notice:publish",
+ children: [],
+ },
+ {
+ id: 134,
+ parentId: 126,
+ name: "撤回",
+ type: "BUTTON",
+ routeName: null,
+ routePath: "",
+ component: null,
+ sort: 6,
+ visible: 1,
+ icon: "",
+ redirect: null,
+ perm: "sys:notice:revoke",
+ children: [],
+ },
+ ],
+ },
+ ],
+ },
+ {
+ id: 118,
+ parentId: 0,
+ name: "系统工具",
+ type: "CATALOG",
+ routeName: null,
+ routePath: "/codegen",
+ component: "Layout",
+ sort: 2,
+ visible: 1,
+ icon: "menu",
+ redirect: null,
+ perm: null,
+ children: [
+ {
+ id: 119,
+ parentId: 118,
+ name: "代码生成",
+ type: "MENU",
+ routeName: "Codegen",
+ routePath: "codegen",
+ component: "codegen/index",
+ sort: 1,
+ visible: 1,
+ icon: "code",
+ redirect: null,
+ perm: null,
+ children: [],
+ },
+ ],
+ },
+ {
+ id: 40,
+ parentId: 0,
+ name: "接口文档",
+ type: "CATALOG",
+ routeName: null,
+ routePath: "/api",
+ component: "Layout",
+ sort: 7,
+ visible: 1,
+ icon: "api",
+ redirect: "",
+ perm: null,
+ children: [
+ {
+ id: 41,
+ parentId: 40,
+ name: "Apifox",
+ type: "MENU",
+ routeName: null,
+ routePath: "apifox",
+ component: "demo/api/apifox",
+ sort: 1,
+ visible: 1,
+ icon: "api",
+ redirect: "",
+ perm: null,
+ children: [],
+ },
+ ],
+ },
+ {
+ id: 26,
+ parentId: 0,
+ name: "平台文档",
+ type: "CATALOG",
+ routeName: null,
+ routePath: "/doc",
+ component: "Layout",
+ sort: 8,
+ visible: 1,
+ icon: "document",
+ redirect: "https://juejin.cn/post/7228990409909108793",
+ perm: null,
+ children: [
+ {
+ id: 102,
+ parentId: 26,
+ name: "平台文档(内嵌)",
+ type: "EXTLINK",
+ routeName: null,
+ routePath: "internal-doc",
+ component: "demo/internal-doc",
+ sort: 1,
+ visible: 1,
+ icon: "document",
+ redirect: "",
+ perm: null,
+ children: [],
+ },
+ {
+ id: 30,
+ parentId: 26,
+ name: "平台文档(外链)",
+ type: "EXTLINK",
+ routeName: null,
+ routePath: "https://juejin.cn/post/7228990409909108793",
+ component: "",
+ sort: 2,
+ visible: 1,
+ icon: "link",
+ redirect: "",
+ perm: null,
+ children: [],
+ },
+ ],
+ },
+ {
+ id: 20,
+ parentId: 0,
+ name: "多级菜单",
+ type: "CATALOG",
+ routeName: null,
+ routePath: "/multi-level",
+ component: "Layout",
+ sort: 9,
+ visible: 1,
+ icon: "cascader",
+ redirect: "",
+ perm: null,
+ children: [
+ {
+ id: 21,
+ parentId: 20,
+ name: "菜单一级",
+ type: "MENU",
+ routeName: null,
+ routePath: "multi-level1",
+ component: "demo/multi-level/level1",
+ sort: 1,
+ visible: 1,
+ icon: "",
+ redirect: "",
+ perm: null,
+ children: [
+ {
+ id: 22,
+ parentId: 21,
+ name: "菜单二级",
+ type: "MENU",
+ routeName: null,
+ routePath: "multi-level2",
+ component: "demo/multi-level/children/level2",
+ sort: 1,
+ visible: 1,
+ icon: "",
+ redirect: null,
+ perm: null,
+ children: [
+ {
+ id: 23,
+ parentId: 22,
+ name: "菜单三级-1",
+ type: "MENU",
+ routeName: null,
+ routePath: "multi-level3-1",
+ component: "demo/multi-level/children/children/level3-1",
+ sort: 1,
+ visible: 1,
+ icon: "",
+ redirect: "",
+ perm: null,
+ children: [],
+ },
+ {
+ id: 24,
+ parentId: 22,
+ name: "菜单三级-2",
+ type: "MENU",
+ routeName: null,
+ routePath: "multi-level3-2",
+ component: "demo/multi-level/children/children/level3-2",
+ sort: 2,
+ visible: 1,
+ icon: "",
+ redirect: "",
+ perm: null,
+ children: [],
+ },
+ ],
+ },
+ ],
+ },
+ ],
+ },
+ {
+ id: 36,
+ parentId: 0,
+ name: "组件封装",
+ type: "CATALOG",
+ routeName: null,
+ routePath: "/component",
+ component: "Layout",
+ sort: 10,
+ visible: 1,
+ icon: "menu",
+ redirect: "",
+ perm: null,
+ children: [
+ {
+ id: 108,
+ parentId: 36,
+ name: "增删改查",
+ type: "MENU",
+ routeName: null,
+ routePath: "curd",
+ component: "demo/curd/index",
+ sort: 0,
+ visible: 1,
+ icon: "",
+ redirect: "",
+ perm: null,
+ children: [],
+ },
+ {
+ id: 109,
+ parentId: 36,
+ name: "列表选择器",
+ type: "MENU",
+ routeName: null,
+ routePath: "table-select",
+ component: "demo/table-select/index",
+ sort: 1,
+ visible: 1,
+ icon: "",
+ redirect: "",
+ perm: null,
+ children: [],
+ },
+ {
+ id: 37,
+ parentId: 36,
+ name: "富文本编辑器",
+ type: "MENU",
+ routeName: null,
+ routePath: "wang-editor",
+ component: "demo/wang-editor",
+ sort: 2,
+ visible: 1,
+ icon: "",
+ redirect: "",
+ perm: null,
+ children: [],
+ },
+ {
+ id: 38,
+ parentId: 36,
+ name: "图片上传",
+ type: "MENU",
+ routeName: null,
+ routePath: "upload",
+ component: "demo/upload",
+ sort: 3,
+ visible: 1,
+ icon: "",
+ redirect: "",
+ perm: null,
+ children: [],
+ },
+ {
+ id: 95,
+ parentId: 36,
+ name: "字典组件",
+ type: "MENU",
+ routeName: null,
+ routePath: "dict-demo",
+ component: "demo/dict",
+ sort: 4,
+ visible: 1,
+ icon: "",
+ redirect: "",
+ perm: null,
+ children: [],
+ },
+ {
+ id: 39,
+ parentId: 36,
+ name: "图标选择器",
+ type: "MENU",
+ routeName: null,
+ routePath: "icon-selector",
+ component: "demo/icon-selector",
+ sort: 4,
+ visible: 1,
+ icon: "",
+ redirect: "",
+ perm: null,
+ children: [],
+ },
+ ],
+ },
+ {
+ id: 110,
+ parentId: 0,
+ name: "路由参数",
+ type: "CATALOG",
+ routeName: null,
+ routePath: "/route-param",
+ component: "Layout",
+ sort: 11,
+ visible: 1,
+ icon: "el-icon-ElementPlus",
+ redirect: null,
+ perm: null,
+ children: [
+ {
+ id: 111,
+ parentId: 110,
+ name: "参数(type=1)",
+ type: "MENU",
+ routeName: null,
+ routePath: "route-param-type1",
+ component: "demo/route-param",
+ sort: 1,
+ visible: 1,
+ icon: "el-icon-Star",
+ redirect: null,
+ perm: null,
+ children: [],
+ },
+ {
+ id: 112,
+ parentId: 110,
+ name: "参数(type=2)",
+ type: "MENU",
+ routeName: null,
+ routePath: "route-param-type2",
+ component: "demo/route-param",
+ sort: 2,
+ visible: 1,
+ icon: "el-icon-StarFilled",
+ redirect: null,
+ perm: null,
+ children: [],
+ },
+ ],
+ },
+ {
+ id: 89,
+ parentId: 0,
+ name: "功能演示",
+ type: "CATALOG",
+ routeName: null,
+ routePath: "/function",
+ component: "Layout",
+ sort: 12,
+ visible: 1,
+ icon: "menu",
+ redirect: "",
+ perm: null,
+ children: [
+ {
+ id: 97,
+ parentId: 89,
+ name: "Icons",
+ type: "MENU",
+ routeName: null,
+ routePath: "icon-demo",
+ component: "demo/icons",
+ sort: 2,
+ visible: 1,
+ icon: "el-icon-Notification",
+ redirect: "",
+ perm: null,
+ children: [],
+ },
+ {
+ id: 90,
+ parentId: 89,
+ name: "Websocket",
+ type: "MENU",
+ routeName: null,
+ routePath: "/function/websocket",
+ component: "demo/websocket",
+ sort: 3,
+ visible: 1,
+ icon: "",
+ redirect: "",
+ perm: null,
+ children: [],
+ },
+ {
+ id: 91,
+ parentId: 89,
+ name: "敬请期待...",
+ type: "CATALOG",
+ routeName: null,
+ routePath: "other/:id",
+ component: "demo/other",
+ sort: 4,
+ visible: 1,
+ icon: "",
+ redirect: "",
+ perm: null,
+ children: [],
+ },
+ ],
+ },
+ ],
+ msg: "一切ok",
+ },
+ },
+
+ // 新增菜单
+ {
+ url: "menus",
+ method: ["POST"],
+ body({ body }) {
+ return {
+ code: "00000",
+ data: null,
+ msg: "新增菜单" + body.name + "成功",
+ };
+ },
+ },
+
+ // 获取菜单表单数据
+ {
+ url: "menus/:id/form",
+ method: ["GET"],
+ body: ({ params }) => {
+ return {
+ code: "00000",
+ data: menuMap[params.id],
+ msg: "一切ok",
+ };
+ },
+ },
+
+ // 修改菜单
+ {
+ url: "menus/:id",
+ method: ["PUT"],
+ body({ body }) {
+ return {
+ code: "00000",
+ data: null,
+ msg: "修改菜单" + body.name + "成功",
+ };
+ },
+ },
+
+ // 删除菜单
+ {
+ url: "menus/:id",
+ method: ["DELETE"],
+ body({ params }) {
+ return {
+ code: "00000",
+ data: null,
+ msg: "删除菜单" + params.id + "成功",
+ };
+ },
+ },
+]);
+
+// 菜单映射表数据
+const menuMap: Record = {
+ 1: {
+ id: 1,
+ parentId: 0,
+ name: "系统管理",
+ type: "CATALOG",
+ routeName: "",
+ routePath: "/system",
+ component: "Layout",
+ perm: null,
+ visible: 1,
+ sort: 1,
+ icon: "system",
+ redirect: "/system/user",
+ keepAlive: null,
+ alwaysShow: null,
+ params: null,
+ },
+ 2: {
+ id: 2,
+ parentId: 1,
+ name: "用户管理",
+ type: "MENU",
+ routeName: "User",
+ routePath: "user",
+ component: "system/user/index",
+ perm: null,
+ visible: 1,
+ sort: 1,
+ icon: "user",
+ redirect: null,
+ keepAlive: 1,
+ alwaysShow: null,
+ },
+ 3: {
+ id: 3,
+ parentId: 1,
+ name: "角色管理",
+ type: "MENU",
+ routeName: "Role",
+ routePath: "role",
+ component: "system/role/index",
+ perm: null,
+ visible: 1,
+ sort: 2,
+ icon: "role",
+ redirect: null,
+ keepAlive: 1,
+ alwaysShow: null,
+ },
+ 4: {
+ id: 4,
+ parentId: 1,
+ name: "菜单管理",
+ type: "MENU",
+ routeName: "Menu",
+ routePath: "menu",
+ component: "system/menu/index",
+ perm: null,
+ visible: 1,
+ sort: 3,
+ icon: "menu",
+ redirect: null,
+ keepAlive: 1,
+ alwaysShow: null,
+ },
+ 5: {
+ id: 5,
+ parentId: 1,
+ name: "部门管理",
+ type: "MENU",
+ routeName: "Dept",
+ routePath: "dept",
+ component: "system/dept/index",
+ perm: null,
+ visible: 1,
+ sort: 4,
+ icon: "tree",
+ redirect: null,
+ keepAlive: 1,
+ alwaysShow: null,
+ },
+ 6: {
+ id: 6,
+ parentId: 1,
+ name: "字典管理",
+ type: "MENU",
+ routeName: "Dict",
+ routePath: "dict",
+ component: "system/dict/index",
+ perm: null,
+ visible: 1,
+ sort: 5,
+ icon: "dict",
+ redirect: null,
+ keepAlive: 1,
+ alwaysShow: null,
+ },
+};
diff --git a/mock/notice.mock.ts b/mock/notice.mock.ts
new file mode 100644
index 0000000..c140c16
--- /dev/null
+++ b/mock/notice.mock.ts
@@ -0,0 +1,416 @@
+import { defineMock } from "./base";
+
+export default defineMock([
+ {
+ url: "notices/page",
+ method: ["GET"],
+ body: {
+ code: "00000",
+ data: {
+ list: [
+ {
+ id: 1,
+ title: "v2.12.0 新增系统日志,访问趋势统计功能。",
+ publishStatus: 1,
+ type: 1,
+ publisherName: "系统管理员",
+ level: "L",
+ publishTime: "2024-09-30 17:21",
+ isRead: null,
+ targetType: 1,
+ createTime: "2024-09-28 11:21",
+ revokeTime: "2024-09-30 17:21",
+ },
+ {
+ id: 2,
+ title: "v2.13.0 新增菜单搜索。",
+ publishStatus: 1,
+ type: 1,
+ publisherName: "系统管理员",
+ level: "L",
+ publishTime: "2024-09-30 17:22",
+ isRead: null,
+ targetType: 1,
+ createTime: "2024-09-28 11:21",
+ revokeTime: "2024-09-30 17:21",
+ },
+ {
+ id: 3,
+ title: "\r\nv2.14.0 新增个人中心。",
+ publishStatus: 1,
+ type: 1,
+ publisherName: "系统管理员",
+ level: "L",
+ publishTime: "2024-09-30 17:23",
+ isRead: null,
+ targetType: 1,
+ createTime: "2024-09-28 11:21",
+ revokeTime: "2024-09-30 17:21",
+ },
+ {
+ id: 4,
+ title: "v2.15.0 登录页面改造。",
+ publishStatus: 1,
+ type: 1,
+ publisherName: "系统管理员",
+ level: "L",
+ publishTime: "2024-09-30 17:24",
+ isRead: null,
+ targetType: 1,
+ createTime: "2024-09-28 11:21",
+ revokeTime: "2024-09-30 17:21",
+ },
+ {
+ id: 5,
+ title: "v2.16.0 通知公告、字典翻译组件。",
+ publishStatus: 1,
+ type: 1,
+ publisherName: "系统管理员",
+ level: "L",
+ publishTime: "2024-09-30 17:25",
+ isRead: null,
+ targetType: 1,
+ createTime: "2024-09-28 11:21",
+ revokeTime: "2024-09-30 17:21",
+ },
+ {
+ id: 6,
+ title: "系统将于本周六凌晨 2 点进行维护,预计维护时间为 2 小时。",
+ publishStatus: 1,
+ type: 2,
+ publisherName: "系统管理员",
+ level: "L",
+ publishTime: "2024-09-30 17:26",
+ isRead: null,
+ targetType: 1,
+ createTime: "2024-09-28 11:21",
+ revokeTime: "2024-09-30 17:21",
+ },
+ {
+ id: 7,
+ title: "最近发现一些钓鱼邮件,请大家提高警惕,不要点击陌生链接。",
+ publishStatus: 1,
+ type: 3,
+ publisherName: "系统管理员",
+ level: "L",
+ publishTime: "2024-09-30 17:27",
+ isRead: null,
+ targetType: 1,
+ createTime: "2024-09-28 11:21",
+ revokeTime: "2024-09-30 17:21",
+ },
+ {
+ id: 8,
+ title: "国庆假期从 10 月 1 日至 10 月 7 日放假,共 7 天。",
+ publishStatus: 1,
+ type: 4,
+ publisherName: "系统管理员",
+ level: "L",
+ publishTime: "2024-09-30 17:28",
+ isRead: null,
+ targetType: 1,
+ createTime: "2024-09-28 11:21",
+ revokeTime: "2024-09-30 17:21",
+ },
+ {
+ id: 9,
+ title: "公司将在 10 月 15 日举办新产品发布会,敬请期待。",
+ publishStatus: 1,
+ type: 5,
+ publisherName: "系统管理员",
+ level: "L",
+ publishTime: "2024-09-30 17:29",
+ isRead: null,
+ targetType: 1,
+ createTime: "2024-09-28 11:21",
+ revokeTime: "2024-09-30 17:21",
+ },
+ {
+ id: 10,
+ title: "v2.16.1 版本修复了 WebSocket 重复连接导致的后台线程阻塞问题,优化了通知公告。",
+ publishStatus: 1,
+ type: 1,
+ publisherName: "系统管理员",
+ level: "L",
+ publishTime: "2024-09-30 17:30",
+ isRead: null,
+ targetType: 1,
+ createTime: "2024-09-28 11:21",
+ revokeTime: "2024-09-30 17:21",
+ },
+ ],
+ total: 10,
+ },
+ msg: "一切ok",
+ },
+ },
+
+ // 新增通知
+ {
+ url: "notices",
+ method: ["POST"],
+ body() {
+ return {
+ code: "00000",
+ data: null,
+ msg: "新增成功",
+ };
+ },
+ },
+
+ // 获取通知表单数据
+ {
+ url: "notices/:id/form",
+ method: ["GET"],
+ body: ({ params }) => {
+ return {
+ code: "00000",
+ data: noticeMap[params.id],
+ msg: "一切ok",
+ };
+ },
+ },
+
+ // 获取通知详情
+ {
+ url: "notices/:id/detail",
+ method: ["GET"],
+ body: ({ params }) => {
+ return {
+ code: "00000",
+ data: noticeMap[params.id],
+ msg: "一切ok",
+ };
+ },
+ },
+ // 修改通知
+ {
+ url: "roles/:id",
+ method: ["PUT"],
+ body({ body }) {
+ return {
+ code: "00000",
+ data: null,
+ msg: "修改通知" + body.name + "成功",
+ };
+ },
+ },
+
+ // 删除通知
+ {
+ url: "roles/:id",
+ method: ["DELETE"],
+ body({ params }) {
+ return {
+ code: "00000",
+ data: null,
+ msg: "删除通知" + params.id + "成功",
+ };
+ },
+ },
+
+ // 我的通知分页列表
+ {
+ url: "notices/my-page",
+ method: ["GET"],
+ body: {
+ code: "00000",
+ data: {
+ list: [
+ {
+ id: 10,
+ title: "v2.16.1 版本修复了 WebSocket 重复连接导致的后台线程阻塞问题,优化了通知公告。",
+ type: 1,
+ level: "L",
+ publisherName: "系统管理员",
+ publishTime: "2024-09-30 17:30",
+ isRead: 0,
+ },
+ {
+ id: 9,
+ title: "公司将在 10 月 15 日举办新产品发布会,敬请期待。",
+ type: 5,
+ level: "L",
+ publisherName: "系统管理员",
+ publishTime: "2024-09-30 17:29",
+ isRead: 0,
+ },
+ {
+ id: 8,
+ title: "国庆假期从 10 月 1 日至 10 月 7 日放假,共 7 天。",
+ type: 4,
+ level: "L",
+ publisherName: "系统管理员",
+ publishTime: "2024-09-30 17:28",
+ isRead: 0,
+ },
+ {
+ id: 7,
+ title: "最近发现一些钓鱼邮件,请大家提高警惕,不要点击陌生链接。",
+ type: 3,
+ level: "L",
+ publisherName: "系统管理员",
+ publishTime: "2024-09-30 17:27",
+ isRead: 0,
+ },
+ {
+ id: 6,
+ title: "系统将于本周六凌晨 2 点进行维护,预计维护时间为 2 小时。",
+ type: 2,
+ level: "L",
+ publisherName: "系统管理员",
+ publishTime: "2024-09-30 17:26",
+ isRead: 0,
+ },
+ ],
+ total: 10,
+ },
+ msg: "一切ok",
+ },
+ },
+]);
+
+// 通知映射表数据
+const noticeMap: Record = {
+ 1: {
+ id: 1,
+ title: "v2.12.0 新增系统日志,访问趋势统计功能。",
+ publishStatus: 1,
+ type: 1,
+ publisherName: "系统管理员",
+ level: "L",
+ levelLabel: null,
+ publishTime: "2024-09-30 17:21",
+ isRead: null,
+ targetType: 1,
+ createTime: "2024-09-28 11:21",
+ revokeTime: "2024-09-30 17:21",
+ },
+ 2: {
+ id: 2,
+ title: "v2.13.0 新增菜单搜索。",
+ publishStatus: 1,
+ type: 1,
+ publisherName: "系统管理员",
+ level: "L",
+ levelLabel: null,
+ publishTime: "2024-09-30 17:22",
+ isRead: null,
+ targetType: 1,
+ createTime: "2024-09-28 11:21",
+ revokeTime: "2024-09-30 17:21",
+ },
+ 3: {
+ id: 3,
+ title: "\r\nv2.14.0 新增个人中心。",
+ publishStatus: 1,
+ type: 1,
+ publisherName: "系统管理员",
+ level: "L",
+ levelLabel: null,
+ publishTime: "2024-09-30 17:23",
+ isRead: null,
+ targetType: 1,
+ createTime: "2024-09-28 11:21",
+ revokeTime: "2024-09-30 17:21",
+ },
+ 4: {
+ id: 4,
+ title: "v2.15.0 登录页面改造。",
+ publishStatus: 1,
+ type: 1,
+ publisherName: "系统管理员",
+ level: "L",
+ levelLabel: null,
+ publishTime: "2024-09-30 17:24",
+ isRead: null,
+ targetType: 1,
+ createTime: "2024-09-28 11:21",
+ revokeTime: "2024-09-30 17:21",
+ },
+
+ 5: {
+ id: 5,
+ title: "v2.16.0 通知公告、字典翻译组件。",
+ publishStatus: 1,
+ type: 1,
+ publisherName: "系统管理员",
+ level: "L",
+ levelLabel: null,
+ publishTime: "2024-09-30 17:25",
+ isRead: null,
+ targetType: 1,
+ createTime: "2024-09-28 11:21",
+ revokeTime: "2024-09-30 17:21",
+ },
+ 6: {
+ id: 6,
+ title: "系统将于本周六凌晨 2 点进行维护,预计维护时间为 2 小时。",
+ publishStatus: 1,
+ type: 2,
+ publisherName: "系统管理员",
+ level: "L",
+ levelLabel: null,
+ publishTime: "2024-09-30 17:26",
+ isRead: null,
+ targetType: 1,
+ createTime: "2024-09-28 11:21",
+ revokeTime: "2024-09-30 17:21",
+ },
+ 7: {
+ id: 7,
+ title: "最近发现一些钓鱼邮件,请大家提高警惕,不要点击陌生链接。",
+ publishStatus: 1,
+ type: 3,
+ publisherName: "系统管理员",
+ level: "L",
+ levelLabel: null,
+ publishTime: "2024-09-30 17:27",
+ isRead: null,
+ targetType: 1,
+ createTime: "2024-09-28 11:21",
+ revokeTime: "2024-09-30 17:21",
+ },
+ 8: {
+ id: 8,
+ title: "国庆假期从 10 月 1 日至 10 月 7 日放假,共 7 天。",
+ publishStatus: 1,
+ type: 4,
+ publisherName: "系统管理员",
+ level: "L",
+ levelLabel: null,
+ publishTime: "2024-09-30 17:28",
+ isRead: null,
+ targetType: 1,
+ createTime: "2024-09-28 11:21",
+ revokeTime: "2024-09-30 17:21",
+ },
+ 9: {
+ id: 9,
+ title: "公司将在 10 月 15 日举办新产品发布会,敬请期待。",
+ publishStatus: 1,
+ type: 5,
+ publisherName: "系统管理员",
+ level: "L",
+ levelLabel: null,
+ publishTime: "2024-09-30 17:29",
+ isRead: null,
+ targetType: 1,
+ createTime: "2024-09-28 11:21",
+ revokeTime: "2024-09-30 17:21",
+ },
+ 10: {
+ id: 10,
+ title: "v2.16.1 版本修复了 WebSocket 重复连接导致的后台线程阻塞问题,优化了通知公告。",
+ publishStatus: 1,
+ type: 1,
+ publisherName: "系统管理员",
+ level: "L",
+ levelLabel: null,
+ publishTime: "2024-09-30 17:30",
+ isRead: null,
+ targetType: 1,
+ createTime: "2024-09-28 11:21",
+ revokeTime: "2024-09-30 17:21",
+ },
+};
diff --git a/mock/role.mock.ts b/mock/role.mock.ts
new file mode 100644
index 0000000..ee85699
--- /dev/null
+++ b/mock/role.mock.ts
@@ -0,0 +1,334 @@
+import { defineMock } from "./base";
+
+export default defineMock([
+ {
+ url: "roles/options",
+ method: ["GET"],
+ body: {
+ code: "00000",
+ data: [
+ {
+ value: 2,
+ label: "系统管理员",
+ },
+ {
+ value: 4,
+ label: "系统管理员1",
+ },
+ {
+ value: 5,
+ label: "系统管理员2",
+ },
+ {
+ value: 6,
+ label: "系统管理员3",
+ },
+ {
+ value: 7,
+ label: "系统管理员4",
+ },
+ {
+ value: 8,
+ label: "系统管理员5",
+ },
+ {
+ value: 9,
+ label: "系统管理员6",
+ },
+ {
+ value: 10,
+ label: "系统管理员7",
+ },
+ {
+ value: 11,
+ label: "系统管理员8",
+ },
+ {
+ value: 12,
+ label: "系统管理员9",
+ },
+ {
+ value: 3,
+ label: "访问游客",
+ },
+ ],
+ msg: "一切ok",
+ },
+ },
+
+ {
+ url: "roles/page",
+ method: ["GET"],
+ body: {
+ code: "00000",
+ data: {
+ list: [
+ {
+ id: 2,
+ name: "系统管理员",
+ code: "ADMIN",
+ status: 1,
+ sort: 2,
+ createTime: "2021-03-25 12:39:54",
+ updateTime: null,
+ },
+ {
+ id: 3,
+ name: "访问游客",
+ code: "GUEST",
+ status: 1,
+ sort: 3,
+ createTime: "2021-05-26 15:49:05",
+ updateTime: "2019-05-05 16:00:00",
+ },
+ {
+ id: 4,
+ name: "系统管理员1",
+ code: "ADMIN1",
+ status: 1,
+ sort: 2,
+ createTime: "2021-03-25 12:39:54",
+ updateTime: null,
+ },
+ {
+ id: 5,
+ name: "系统管理员2",
+ code: "ADMIN2",
+ status: 1,
+ sort: 2,
+ createTime: "2021-03-25 12:39:54",
+ updateTime: null,
+ },
+ {
+ id: 6,
+ name: "系统管理员3",
+ code: "ADMIN3",
+ status: 1,
+ sort: 2,
+ createTime: "2021-03-25 12:39:54",
+ updateTime: null,
+ },
+ {
+ id: 7,
+ name: "系统管理员4",
+ code: "ADMIN4",
+ status: 1,
+ sort: 2,
+ createTime: "2021-03-25 12:39:54",
+ updateTime: null,
+ },
+ {
+ id: 8,
+ name: "系统管理员5",
+ code: "ADMIN5",
+ status: 1,
+ sort: 2,
+ createTime: "2021-03-25 12:39:54",
+ updateTime: null,
+ },
+ {
+ id: 9,
+ name: "系统管理员6",
+ code: "ADMIN6",
+ status: 1,
+ sort: 2,
+ createTime: "2021-03-25 12:39:54",
+ updateTime: "2023-12-04 11:43:15",
+ },
+ {
+ id: 10,
+ name: "系统管理员7",
+ code: "ADMIN7",
+ status: 1,
+ sort: 2,
+ createTime: "2021-03-25 12:39:54",
+ updateTime: null,
+ },
+ {
+ id: 11,
+ name: "系统管理员8",
+ code: "ADMIN8",
+ status: 1,
+ sort: 2,
+ createTime: "2021-03-25 12:39:54",
+ updateTime: null,
+ },
+ ],
+ total: 10,
+ },
+ msg: "一切ok",
+ },
+ },
+
+ // 新增角色
+ {
+ url: "roles",
+ method: ["POST"],
+ body({ body }) {
+ return {
+ code: "00000",
+ data: null,
+ msg: "新增角色" + body.name + "成功",
+ };
+ },
+ },
+
+ // 获取角色表单数据
+ {
+ url: "roles/:id/form",
+ method: ["GET"],
+ body: ({ params }) => {
+ return {
+ code: "00000",
+ data: roleMap[params.id],
+ msg: "一切ok",
+ };
+ },
+ },
+ // 修改角色
+ {
+ url: "roles/:id",
+ method: ["PUT"],
+ body({ body }) {
+ return {
+ code: "00000",
+ data: null,
+ msg: "修改角色" + body.name + "成功",
+ };
+ },
+ },
+
+ // 删除角色
+ {
+ url: "roles/:id",
+ method: ["DELETE"],
+ body({ params }) {
+ return {
+ code: "00000",
+ data: null,
+ msg: "删除角色" + params.id + "成功",
+ };
+ },
+ },
+ // 获取角色拥有的菜单ID
+ {
+ url: "roles/:id/menuIds",
+ method: ["GET"],
+ body: ({}) => {
+ return {
+ code: "00000",
+ data: [
+ 1, 2, 31, 32, 33, 88, 3, 70, 71, 72, 4, 73, 75, 74, 5, 76, 77, 78, 6, 79, 81, 84, 85, 86,
+ 87, 40, 41, 26, 30, 20, 21, 22, 23, 24, 89, 90, 91, 36, 37, 38, 39, 93, 94, 95, 97, 102,
+ 89, 90, 91, 93, 94, 95, 97, 102, 103, 104,
+ ],
+ msg: "一切ok",
+ };
+ },
+ },
+ // 保存角色菜单
+ {
+ url: "roles/:id/menus",
+ method: ["PUT"],
+ body: {
+ code: "00000",
+ data: null,
+ msg: "一切ok",
+ },
+ },
+]);
+
+// 角色映射表数据
+const roleMap: Record = {
+ 2: {
+ id: 2,
+ name: "系统管理员",
+ code: "ADMIN",
+ status: 1,
+ sort: 2,
+ createTime: "2021-03-25 12:39:54",
+ updateTime: null,
+ },
+ 3: {
+ id: 3,
+ name: "访问游客",
+ code: "GUEST",
+ status: 1,
+ sort: 3,
+ createTime: "2021-05-26 15:49:05",
+ updateTime: "2019-05-05 16:00:00",
+ },
+ 4: {
+ id: 4,
+ name: "系统管理员1",
+ code: "ADMIN1",
+ status: 1,
+ sort: 2,
+ createTime: "2021-03-25 12:39:54",
+ updateTime: null,
+ },
+ 5: {
+ id: 5,
+ name: "系统管理员2",
+ code: "ADMIN2",
+ status: 1,
+ sort: 2,
+ createTime: "2021-03-25 12:39:54",
+ updateTime: null,
+ },
+
+ 6: {
+ id: 6,
+ name: "系统管理员3",
+ code: "ADMIN3",
+ status: 1,
+ sort: 2,
+ createTime: "2021-03-25 12:39:54",
+ updateTime: null,
+ },
+ 7: {
+ id: 7,
+ name: "系统管理员4",
+ code: "ADMIN4",
+ status: 1,
+ sort: 2,
+ createTime: "2021-03-25 12:39:54",
+ updateTime: null,
+ },
+ 8: {
+ id: 8,
+ name: "系统管理员5",
+ code: "ADMIN5",
+ status: 1,
+ sort: 2,
+ createTime: "2021-03-25 12:39:54",
+ updateTime: null,
+ },
+ 9: {
+ id: 9,
+ name: "系统管理员6",
+ code: "ADMIN6",
+ status: 1,
+ sort: 2,
+ createTime: "2021-03-25 12:39:54",
+ updateTime: "2023-12-04 11:43:15",
+ },
+ 10: {
+ id: 10,
+ name: "系统管理员7",
+ code: "ADMIN7",
+ status: 1,
+ sort: 2,
+ createTime: "2021-03-25 12:39:54",
+ updateTime: null,
+ },
+ 11: {
+ id: 11,
+ name: "系统管理员8",
+ code: "ADMIN8",
+ status: 1,
+ sort: 2,
+ createTime: "2021-03-25 12:39:54",
+ updateTime: null,
+ },
+};
diff --git a/mock/user.mock.ts b/mock/user.mock.ts
new file mode 100644
index 0000000..57c8482
--- /dev/null
+++ b/mock/user.mock.ts
@@ -0,0 +1,237 @@
+import { defineMock } from "./base";
+
+export default defineMock([
+ {
+ url: "users/me",
+ method: ["GET"],
+ body: {
+ code: "00000",
+ data: {
+ userId: 2,
+ username: "admin",
+ nickname: "系统管理员",
+ avatar: "https://foruda.gitee.com/images/1723603502796844527/03cdca2a_716974.gif",
+ roles: ["ADMIN"],
+ perms: [
+ "sys:notice:edit",
+ "sys:menu:delete",
+ "sys:dict:edit",
+ "sys:notice:query",
+ "sys:dict:delete",
+ "sys:config:add",
+ "sys:config:refresh",
+ "sys:menu:add",
+ "sys:user:add",
+ "sys:user:export",
+ "sys:role:edit",
+ "sys:dept:delete",
+ "sys:config:update",
+ "sys:user:password:reset",
+ "sys:notice:revoke",
+ "sys:user:import",
+ "sys:user:delete",
+ "sys:dict_type:delete",
+ "sys:dict:add",
+ "sys:role:add",
+ "sys:notice:publish",
+ "sys:notice:delete",
+ "sys:dept:edit",
+ "sys:dict_type:edit",
+ "sys:user:query",
+ "sys:user:edit",
+ "sys:config:delete",
+ "sys:dept:add",
+ "sys:notice:add",
+ "sys:role:delete",
+ "sys:menu:edit",
+ "sys:config:query",
+ ],
+ },
+ msg: "一切ok",
+ },
+ },
+
+ {
+ url: "users/page",
+ method: ["GET"],
+ body: {
+ code: "00000",
+ data: {
+ list: [
+ {
+ id: 2,
+ username: "admin",
+ nickname: "系统管理员",
+ mobile: "17621210366",
+ gender: 1,
+ avatar: "https://foruda.gitee.com/images/1723603502796844527/03cdca2a_716974.gif",
+ email: "",
+ status: 1,
+ deptId: 1,
+ roleIds: [2],
+ },
+ {
+ id: 3,
+ username: "test",
+ nickname: "测试小用户",
+ mobile: "17621210366",
+ gender: 1,
+ avatar: "https://foruda.gitee.com/images/1723603502796844527/03cdca2a_716974.gif",
+ email: "youlaitech@163.com",
+ status: 1,
+ deptId: 3,
+ roleIds: [3],
+ },
+ ],
+ total: 2,
+ },
+ msg: "一切ok",
+ },
+ },
+
+ // 新增用户
+ {
+ url: "users",
+ method: ["POST"],
+ body({ body }) {
+ return {
+ code: "00000",
+ data: null,
+ msg: "新增用户" + body.nickname + "成功",
+ };
+ },
+ },
+
+ // 获取用户表单数据
+ {
+ url: "users/:userId/form",
+ method: ["GET"],
+ body: ({ params }) => {
+ return {
+ code: "00000",
+ data: userMap[params.userId],
+ msg: "一切ok",
+ };
+ },
+ },
+ // 修改用户
+ {
+ url: "users/:userId",
+ method: ["PUT"],
+ body({ body }) {
+ return {
+ code: "00000",
+ data: null,
+ msg: "修改用户" + body.nickname + "成功",
+ };
+ },
+ },
+
+ // 删除用户
+ {
+ url: "users/:userId",
+ method: ["DELETE"],
+ body({ params }) {
+ return {
+ code: "00000",
+ data: null,
+ msg: "删除用户" + params.id + "成功",
+ };
+ },
+ },
+
+ // 重置密码
+ {
+ url: "users/:userId/password/reset",
+ method: ["PUT"],
+ body({ query }) {
+ return {
+ code: "00000",
+ data: null,
+ msg: "重置密码成功,新密码为:" + query.password,
+ };
+ },
+ },
+
+ // 导出Excel
+ {
+ url: "users/_export",
+ method: ["GET"],
+ headers: {
+ "Content-Disposition": "attachment; filename=%E7%94%A8%E6%88%B7%E5%88%97%E8%A1%A8.xlsx",
+ "Content-Type": "application/vnd.openxmlformats-officedocument.spreadsheetml.sheet",
+ },
+ },
+
+ {
+ url: "users/profile",
+ method: ["GET"],
+ body: {
+ code: "00000",
+ data: {
+ id: 2,
+ username: "admin",
+ nickname: "系统管理员",
+ avatar: "https://foruda.gitee.com/images/1723603502796844527/03cdca2a_716974.gif",
+ gender: 1,
+ mobile: "17621210366",
+ email: null,
+ deptName: "有来技术",
+ roleNames: "系统管理员",
+ createTime: "2019-10-10",
+ },
+ },
+ },
+
+ {
+ url: "users/profile",
+ method: ["PUT"],
+ body() {
+ return {
+ code: "00000",
+ data: null,
+ msg: "修改个人信息成功",
+ };
+ },
+ },
+
+ {
+ url: "users/password",
+ method: ["PUT"],
+ body() {
+ return {
+ code: "00000",
+ data: null,
+ msg: "修改密码成功",
+ };
+ },
+ },
+]);
+
+// 用户映射表数据
+const userMap: Record = {
+ 2: {
+ id: 2,
+ username: "admin",
+ nickname: "系统管理员",
+ mobile: "17621210366",
+ gender: 1,
+ avatar: "https://foruda.gitee.com/images/1723603502796844527/03cdca2a_716974.gif",
+ email: "",
+ status: 1,
+ deptId: 1,
+ roleIds: [2],
+ },
+ 3: {
+ id: 3,
+ username: "test",
+ nickname: "测试小用户",
+ mobile: "17621210366",
+ gender: 1,
+ avatar: "https://foruda.gitee.com/images/1723603502796844527/03cdca2a_716974.gif",
+ email: "youlaitech@163.com",
+ status: 1,
+ deptId: 3,
+ roleIds: [3],
+ },
+};
diff --git a/package.json b/package.json
new file mode 100644
index 0000000..04be2bf
--- /dev/null
+++ b/package.json
@@ -0,0 +1,125 @@
+{
+ "name": "vue3-element-admin",
+ "version": "2.21.1",
+ "private": true,
+ "type": "module",
+ "scripts": {
+ "dev": "vite",
+ "build": "vue-tsc --noEmit & vite build",
+ "preview": "vite preview",
+ "build-only": "vite build",
+ "type-check": "vue-tsc --noEmit",
+ "lint:eslint": "eslint --fix ./src",
+ "lint:prettier": "prettier --write \"**/*.{js,cjs,ts,json,tsx,css,less,scss,vue,html,md}\"",
+ "lint:stylelint": "stylelint \"**/*.{css,scss,vue}\" --fix",
+ "lint:lint-staged": "lint-staged",
+ "preinstall": "npx only-allow pnpm",
+ "prepare": "husky",
+ "commit": "git-cz"
+ },
+ "config": {
+ "commitizen": {
+ "path": "node_modules/cz-git"
+ }
+ },
+ "lint-staged": {
+ "*.{js,ts}": [
+ "eslint --fix",
+ "prettier --write"
+ ],
+ "*.{cjs,json}": [
+ "prettier --write"
+ ],
+ "*.{vue,html}": [
+ "eslint --fix",
+ "prettier --write",
+ "stylelint --fix"
+ ],
+ "*.{scss,css}": [
+ "stylelint --fix",
+ "prettier --write"
+ ],
+ "*.md": [
+ "prettier --write"
+ ]
+ },
+ "dependencies": {
+ "@element-plus/icons-vue": "^2.3.1",
+ "@stomp/stompjs": "^7.0.0",
+ "@vueuse/core": "^12.5.0",
+ "@wangeditor-next/editor": "^5.6.31",
+ "@wangeditor-next/editor-for-vue": "^5.1.14",
+ "axios": "^1.7.9",
+ "codemirror": "^5.65.18",
+ "codemirror-editor-vue3": "^2.8.0",
+ "default-passive-events": "^2.0.0",
+ "echarts": "^5.6.0",
+ "element-plus": "^2.9.3",
+ "exceljs": "^4.4.0",
+ "js-cookie": "^3.0.5",
+ "jsencrypt": "^3.3.2",
+ "lodash-es": "^4.17.21",
+ "nprogress": "^0.2.0",
+ "path-browserify": "^1.0.1",
+ "path-to-regexp": "^8.2.0",
+ "pinia": "^2.3.1",
+ "qs": "^6.14.0",
+ "sortablejs": "^1.15.6",
+ "vue": "^3.5.13",
+ "vue-i18n": "^11.1.0",
+ "vue-router": "^4.5.0"
+ },
+ "devDependencies": {
+ "@commitlint/cli": "^19.7.1",
+ "@commitlint/config-conventional": "^19.7.1",
+ "@eslint/js": "^9.19.0",
+ "@types/codemirror": "^5.60.15",
+ "@types/lodash": "^4.17.15",
+ "@types/node": "^22.13.1",
+ "@types/nprogress": "^0.2.3",
+ "@types/path-browserify": "^1.0.3",
+ "@types/qs": "^6.9.18",
+ "@types/sortablejs": "^1.15.8",
+ "@typescript-eslint/eslint-plugin": "^8.23.0",
+ "@typescript-eslint/parser": "^8.23.0",
+ "@vitejs/plugin-vue": "^5.2.1",
+ "autoprefixer": "^10.4.20",
+ "commitizen": "^4.3.1",
+ "cz-git": "^1.11.0",
+ "eslint": "^9.19.0",
+ "eslint-config-prettier": "^10.0.1",
+ "eslint-plugin-prettier": "^5.2.3",
+ "eslint-plugin-vue": "^9.32.0",
+ "globals": "^15.14.0",
+ "husky": "^9.1.7",
+ "lint-staged": "^15.4.3",
+ "postcss": "^8.5.1",
+ "postcss-html": "^1.8.0",
+ "postcss-scss": "^4.0.9",
+ "prettier": "^3.4.2",
+ "sass": "^1.84.0",
+ "stylelint": "^16.14.1",
+ "stylelint-config-html": "^1.1.0",
+ "stylelint-config-recess-order": "^6.0.0",
+ "stylelint-config-recommended-scss": "^14.1.0",
+ "stylelint-config-recommended-vue": "^1.6.0",
+ "stylelint-config-standard": "^37.0.0",
+ "terser": "^5.38.0",
+ "typescript": "^5.7.3",
+ "typescript-eslint": "^8.23.0",
+ "unocss": "65.4.3",
+ "unplugin-auto-import": "^19.0.0",
+ "unplugin-vue-components": "^28.0.0",
+ "vite": "^6.1.0",
+ "vite-plugin-mock-dev-server": "^1.8.3",
+ "vite-plugin-svg-icons": "^2.0.1",
+ "vue-eslint-parser": "^9.4.3",
+ "vue-tsc": "^2.2.0"
+ },
+ "engines": {
+ "node": ">=18.0.0"
+ },
+ "repository": "https://gitee.com/youlaiorg/vue3-element-admin.git",
+ "author": "有来开源组织",
+ "license": "MIT"
+}
diff --git a/public/favicon.ico b/public/favicon.ico
new file mode 100644
index 0000000000000000000000000000000000000000..5e82ccc5d9e7108c5a4f9728da4db50a424db60e
GIT binary patch
literal 4286
zcmdVcTWl3Y9LMn~N2!2VKqMN#R%#RoL5NUJt5TGLklsj56a`d3Ap|voqJor5
z@$#U-3mPF}f`PV&f+dYS*eVyr#HbGj)F=Uzh8pDB@%x+I*>h~5T!aUgeAwxpJ@fzX
z{AXs5F=_nE$};-jtm|Y#wKzvc?z!Y;MuRCJpqqmCZ50)%)pI^^1M_@M9&_KZvinTVGn-AA^e2d
z7>~AaIb4ePHov|#`)&|J&!M&UGW6`ZFsG|>O;cro&mqROWl(%s
zi|Oct;GPSpgPU`j>j_uiUi480&Gk7aEAKGV-rQ%3E37kcsr5N@=ko?A{@-{Olc9Oz
z?m3?^aQjDfH0Lw;?f}$b13GT{s9bS-AiH>_%_yy~F*7U5p^$qHLpgkpsnE4>49;B#
z>tZW@JjU~1$9G@BkLbVs+lKfNl2+f>p2AR$+s$=
z&i6|3oBH_q(+5mpg>6f$0?fb>Sk$9?;YxBy3|0=>o3}$b9LGXD0*$2`ht}#4>^HaT
zz7ALdJx{yEt84Y_xfqMDq1eYm<qQ2dj431e|ybtqk7O$jxnPZjW<
zx#pv;EmIxxlXVzQyn|2JK2Z)u{~q^V^rM=ufnuM=yt22W*czJ;@fXTqn0slQ7vM00^|0k;
zW7p>@#(e1cl|vowMo+l-TAS)4q?&4-YkyP!ioXe+;KzLmZLR<1nse@Y@O|x%b#Qek
z;+jpk_0WY_o1yr!6aE(hk6kDtf)Ro8E!*!wUD{__FhHNOV^b$`z1^8;9p*j>r>
zkVh=-r;7h49zqse9k^HDR~<|7J{0=|X2SQ~k@f~?j7R$Gg~{Aa=^O6e=lD$Pp+xuC
zE!oCay7$^V;`ut%)7Cub4E28{;&>8Aq4;aj0jBsh*Y9lF%4q@fMfG-hX%3f^pYS>A
zd@ZzAG&i$c9fEOIEd74;b*P}Ncemaxnb7;?J1Dl!x5D?U+#sHH+t>Nr=d(O}YCgmZ_LlQHxBj~*wcpFtwmM9{Z8n+6u%~d$q$6e
+
+
+
+
+
+
+
+
+
diff --git a/src/api/auth/index.ts b/src/api/auth/index.ts
new file mode 100644
index 0000000..64fb97b
--- /dev/null
+++ b/src/api/auth/index.ts
@@ -0,0 +1,84 @@
+import request from "@/utils/request";
+
+const AUTH_BASE_URL = "/api/v1/auth";
+
+const AuthAPI = {
+ /** 登录接口*/
+ login(data: LoginFormData) {
+ const formData = new FormData();
+ formData.append("username", data.username);
+ formData.append("password", data.password);
+ formData.append("captchaKey", data.captchaKey);
+ formData.append("captchaCode", data.captchaCode);
+ return request({
+ url: `${AUTH_BASE_URL}/login`,
+ method: "post",
+ data: formData,
+ headers: {
+ "Content-Type": "multipart/form-data",
+ },
+ });
+ },
+
+ /** 刷新 token 接口*/
+ refreshToken(refreshToken: string) {
+ return request({
+ url: `${AUTH_BASE_URL}/refresh-token`,
+ method: "post",
+ params: { refreshToken: refreshToken },
+ headers: {
+ Authorization: "no-auth",
+ },
+ });
+ },
+
+ /** 注销登录接口 */
+ logout() {
+ return request({
+ url: `${AUTH_BASE_URL}/logout`,
+ method: "delete",
+ });
+ },
+
+ /** 获取验证码接口*/
+ getCaptcha() {
+ return request({
+ url: `${AUTH_BASE_URL}/captcha`,
+ method: "get",
+ });
+ },
+};
+
+export default AuthAPI;
+
+/** 登录表单数据 */
+export interface LoginFormData {
+ /** 用户名 */
+ username: string;
+ /** 密码 */
+ password: string;
+ /** 验证码缓存key */
+ captchaKey: string;
+ /** 验证码 */
+ captchaCode: string;
+}
+
+/** 登录响应 */
+export interface LoginResult {
+ /** 访问令牌 */
+ accessToken: string;
+ /** 刷新令牌 */
+ refreshToken: string;
+ /** 令牌类型 */
+ tokenType: string;
+ /** 过期时间(秒) */
+ expiresIn: number;
+}
+
+/** 验证码信息 */
+export interface CaptchaInfo {
+ /** 验证码缓存key */
+ captchaKey: string;
+ /** 验证码图片Base64字符串 */
+ captchaBase64: string;
+}
diff --git a/src/api/codegen/index.ts b/src/api/codegen/index.ts
new file mode 100644
index 0000000..af592ba
--- /dev/null
+++ b/src/api/codegen/index.ts
@@ -0,0 +1,191 @@
+import request from "@/utils/request";
+
+const GENERATOR_BASE_URL = "/api/v1/codegen";
+
+const GeneratorAPI = {
+ /** 获取数据表分页列表 */
+ getTablePage(params: TablePageQuery) {
+ return request>({
+ url: `${GENERATOR_BASE_URL}/table/page`,
+ method: "get",
+ params: params,
+ });
+ },
+
+ /** 获取代码生成配置 */
+ getGenConfig(tableName: string) {
+ return request({
+ url: `${GENERATOR_BASE_URL}/${tableName}/config`,
+ method: "get",
+ });
+ },
+
+ /** 获取代码生成配置 */
+ saveGenConfig(tableName: string, data: GenConfigForm) {
+ return request({
+ url: `${GENERATOR_BASE_URL}/${tableName}/config`,
+ method: "post",
+ data: data,
+ });
+ },
+
+ /** 获取代码生成预览数据 */
+ getPreviewData(tableName: string) {
+ return request({
+ url: `${GENERATOR_BASE_URL}/${tableName}/preview`,
+ method: "get",
+ });
+ },
+
+ /** 重置代码生成配置 */
+ resetGenConfig(tableName: string) {
+ return request({
+ url: `${GENERATOR_BASE_URL}/${tableName}/config`,
+ method: "delete",
+ });
+ },
+
+ /**
+ * 下载 ZIP 文件
+ * @param url
+ * @param fileName
+ */
+ download(tableName: string) {
+ return request({
+ url: `${GENERATOR_BASE_URL}/${tableName}/download`,
+ method: "get",
+ responseType: "blob",
+ }).then((response) => {
+ const fileName = decodeURI(
+ response.headers["content-disposition"].split(";")[1].split("=")[1]
+ );
+
+ const blob = new Blob([response.data], { type: "application/zip" });
+ const a = document.createElement("a");
+ const url = window.URL.createObjectURL(blob);
+ a.href = url;
+ a.download = fileName;
+ a.click();
+ window.URL.revokeObjectURL(url);
+ });
+ },
+};
+
+export default GeneratorAPI;
+
+/** 代码生成预览对象 */
+export interface GeneratorPreviewVO {
+ /** 文件生成路径 */
+ path: string;
+ /** 文件名称 */
+ fileName: string;
+ /** 文件内容 */
+ content: string;
+}
+
+/** 数据表分页查询参数 */
+export interface TablePageQuery extends PageQuery {
+ /** 关键字(表名) */
+ keywords?: string;
+}
+
+/** 数据表分页对象 */
+export interface TablePageVO {
+ /** 表名称 */
+ tableName: string;
+
+ /** 表描述 */
+ tableComment: string;
+
+ /** 存储引擎 */
+ engine: string;
+
+ /** 字符集排序规则 */
+ tableCollation: string;
+
+ /** 创建时间 */
+ createTime: string;
+}
+
+/** 代码生成配置表单 */
+export interface GenConfigForm {
+ /** 主键 */
+ id?: number;
+
+ /** 表名 */
+ tableName?: string;
+
+ /** 业务名 */
+ businessName?: string;
+
+ /** 模块名 */
+ moduleName?: string;
+
+ /** 包名 */
+ packageName?: string;
+
+ /** 实体名 */
+ entityName?: string;
+
+ /** 作者 */
+ author?: string;
+
+ /** 上级菜单 */
+ parentMenuId?: number;
+
+ /** 后端应用名 */
+ backendAppName?: string;
+ /** 前端应用名 */
+ frontendAppName?: string;
+
+ /** 字段配置列表 */
+ fieldConfigs?: FieldConfig[];
+}
+
+/** 字段配置 */
+export interface FieldConfig {
+ /** 主键 */
+ id?: number;
+
+ /** 列名 */
+ columnName?: string;
+
+ /** 列类型 */
+ columnType?: string;
+
+ /** 字段名 */
+ fieldName?: string;
+
+ /** 字段类型 */
+ fieldType?: string;
+
+ /** 字段描述 */
+ fieldComment?: string;
+
+ /** 是否在列表显示 */
+ isShowInList?: number;
+
+ /** 是否在表单显示 */
+ isShowInForm?: number;
+
+ /** 是否在查询条件显示 */
+ isShowInQuery?: number;
+
+ /** 是否必填 */
+ isRequired?: number;
+
+ /** 表单类型 */
+ formType?: number;
+
+ /** 查询类型 */
+ queryType?: number;
+
+ /** 字段长度 */
+ maxLength?: number;
+
+ /** 字段排序 */
+ fieldSort?: number;
+
+ /** 字典类型 */
+ dictType?: string;
+}
diff --git a/src/api/coup/index.js b/src/api/coup/index.js
new file mode 100644
index 0000000..246fff7
--- /dev/null
+++ b/src/api/coup/index.js
@@ -0,0 +1,205 @@
+// 代客下单
+import request from "@/utils/request-php";
+import { getToken } from '@/utils/auth'
+function getLoginName() {
+ const obj = localStorage.getItem("userInfo") || '';
+ const { username } = obj ? JSON.parse(obj) : {};
+ return username
+}
+// 抖音团购核销准备
+export function $douyin_fulfilmentcertificateprepare(data) {
+ return request({
+ url: 'douyin/fulfilmentcertificateprepare',
+ method: "post",
+ data: {
+ ...data
+ }
+ });
+}
+// 抖音团购核销
+export function $douyin_certificateprepare(data) {
+ return request({
+ url: 'douyin/certificateprepare',
+ method: "post",
+ data: {
+
+ ...data
+ }
+ });
+}
+
+// 抖音团购核销撤销
+export function $douyin_fulfilmentcertificatecancel(data) {
+ return request({
+ url: 'douyin/fulfilmentcertificatecancel',
+ method: "post",
+ data: {
+ ...data
+ }
+ });
+}
+// 抖音团购核销记录
+export function $douyin_orderlist(data) {
+ return request({
+ url: 'douyin/orderlist',
+ method: "post",
+ data: {
+
+ ...data
+ }
+ });
+}
+
+
+// 抖音门店列表
+export function $douyin_storelist(data) {
+ return request({
+ url: 'douyin/storelist',
+ method: "post",
+ data: {
+
+ ...data
+ }
+ });
+}
+
+// 抖音绑定门店
+export function $douyin_bindstore(data) {
+ return request({
+ url: 'douyin/bindstore',
+ method: "post",
+ data: {
+
+ ...data
+ }
+ });
+}
+// 抖音订单查询
+
+export function $douyin_orderquery(data) {
+ return request({
+ url: 'douyin/orderquery',
+ method: "post",
+ data: {
+
+ ...data
+ }
+ });
+}
+//会员签入
+export function $douyin_checkIn(data) {
+ return request({
+ url: 'douyin/checkIn',
+ method: "post",
+ data: {
+ clientType: 'ADMIN',
+ token: getToken(),
+ loginName: getLoginName(),
+
+ ...data
+ }
+ });
+}
+
+//美团
+
+// 美团获取uisdk 绑定 链接
+
+export function $meituan_getuisdkurl(data) {
+ return request({
+ url: 'meituan/getuisdkurl',
+ method: "post",
+ data: {
+
+ ...data
+ }
+ });
+}
+// 美团获取uisdk 解绑 链接
+
+export function $meituan_getuisdkuniurl(data) {
+ return request({
+ url: 'meituan/getuisdkuniurl',
+ method: "post",
+ data: {
+
+ ...data
+ }
+ });
+}
+// 美团团购核销准备
+
+export function $meituan_fulfilmentcertificateprepare(data) {
+ return request({
+ url: 'meituan/fulfilmentcertificateprepare',
+ method: "post",
+ data: {
+
+ ...data
+ }
+ });
+}
+// 美团执行核销
+
+export function $meituan_certificateprepare(data) {
+ return request({
+ url: 'meituan/certificateprepare',
+ method: "post",
+ data: {
+
+ ...data
+ }
+ });
+}
+// 美团团购核销记录
+
+
+export function $meituan_orderlist(data) {
+ return request({
+ url: 'meituan/orderlist',
+ method: "post",
+ data: {
+
+ ...data
+ }
+ });
+}
+
+// 美团团购撤销
+
+
+export function $meituan_fulfilmentcertificatecancel(data) {
+ return request({
+ url: 'meituan/fulfilmentcertificatecancel',
+ method: "post",
+ data: {
+
+ ...data
+ }
+ });
+}
+
+// 美团查询绑定状态
+export function $meituan_searchstorestatus(data) {
+ return request({
+ url: 'meituan/searchstorestatus',
+ method: "post",
+ data: {
+
+ ...data
+ }
+ });
+}
+
+// 登出
+export function $logout(data) {
+ return request({
+ url: 'user/logout',
+ method: "post",
+ data: {
+
+ ...data
+ }
+ });
+}
+
diff --git a/src/api/file/index.ts b/src/api/file/index.ts
new file mode 100644
index 0000000..0b10977
--- /dev/null
+++ b/src/api/file/index.ts
@@ -0,0 +1,81 @@
+import request from "@/utils/request";
+
+const FileAPI = {
+ /**
+ * 上传文件
+ *
+ * @param formData
+ */
+ upload(formData: FormData) {
+ return request({
+ url: "/api/v1/files",
+ method: "post",
+ data: formData,
+ headers: {
+ "Content-Type": "multipart/form-data",
+ },
+ });
+ },
+
+ /**
+ * 上传文件
+ */
+ uploadFile(file: File) {
+ const formData = new FormData();
+ formData.append("file", file);
+ return request({
+ url: "/api/v1/files",
+ method: "post",
+ data: formData,
+ headers: {
+ "Content-Type": "multipart/form-data",
+ },
+ });
+ },
+
+ /**
+ * 删除文件
+ *
+ * @param filePath 文件完整路径
+ */
+ delete(filePath?: string) {
+ return request({
+ url: "/api/v1/files",
+ method: "delete",
+ params: { filePath: filePath },
+ });
+ },
+
+ /**
+ * 下载文件
+ * @param url
+ * @param fileName
+ */
+ download(url: string, fileName?: string) {
+ return request({
+ url: url,
+ method: "get",
+ responseType: "blob",
+ }).then((res) => {
+ const blob = new Blob([res.data]);
+ const a = document.createElement("a");
+ const url = window.URL.createObjectURL(blob);
+ a.href = url;
+ a.download = fileName || "下载文件";
+ a.click();
+ window.URL.revokeObjectURL(url);
+ });
+ },
+};
+
+export default FileAPI;
+
+/**
+ * 文件API类型声明
+ */
+export interface FileInfo {
+ /** 文件名 */
+ name: string;
+ /** 文件路径 */
+ url: string;
+}
diff --git a/src/api/login.js b/src/api/login.js
new file mode 100644
index 0000000..b2f6603
--- /dev/null
+++ b/src/api/login.js
@@ -0,0 +1,58 @@
+import request from "@/utils/request";
+
+export function login(data) {
+ return request({
+ url: "auth/login",
+ method: "post",
+ data,
+ });
+}
+
+export function getInfo() {
+ return request({
+ url: "auth/info",
+ method: "get",
+ });
+}
+
+export function changChildShop(data) {
+ return request({
+ url: "/api/tbShopInfo/changChildShop",
+ method: "post",
+ data,
+ });
+}
+
+export function getCodeImg(header) {
+ return request({
+ url: "auth/code",
+ method: "get",
+ });
+}
+export function getqueryChildShop(params) {
+ return request({
+ url: "api/tbShopInfo/queryChildShop",
+ method: "get",
+ params,
+ });
+}
+
+export function logout() {
+ return request({
+ url: "auth/logout",
+ method: "delete",
+ });
+}
+
+/**
+ * 个人中心 修改密码
+ * @param {*} data
+ * @returns
+ */
+export function updatePass(data) {
+ return request({
+ url: "/api/users/updatePass",
+ method: "post",
+ data,
+ });
+}
diff --git a/src/api/system/config.ts b/src/api/system/config.ts
new file mode 100644
index 0000000..95c1bb8
--- /dev/null
+++ b/src/api/system/config.ts
@@ -0,0 +1,104 @@
+import request from "@/utils/request";
+
+const CONFIG_BASE_URL = "/api/v1/config";
+
+const ConfigAPI = {
+ /** 获取系统配置分页数据 */
+ getPage(queryParams?: ConfigPageQuery) {
+ return request>({
+ url: `${CONFIG_BASE_URL}/page`,
+ method: "get",
+ params: queryParams,
+ });
+ },
+ /**
+ * 获取系统配置表单数据
+ *
+ * @param id ConfigID
+ * @returns Config表单数据
+ */
+ getFormData(id: number) {
+ return request({
+ url: `${CONFIG_BASE_URL}/${id}/form`,
+ method: "get",
+ });
+ },
+
+ /** 添加系统配置*/
+ add(data: ConfigForm) {
+ return request({
+ url: `${CONFIG_BASE_URL}`,
+ method: "post",
+ data: data,
+ });
+ },
+
+ /**
+ * 更新系统配置
+ *
+ * @param id ConfigID
+ * @param data Config表单数据
+ */
+ update(id: number, data: ConfigForm) {
+ return request({
+ url: `${CONFIG_BASE_URL}/${id}`,
+ method: "put",
+ data: data,
+ });
+ },
+
+ /**
+ * 删除系统配置
+ *
+ * @param ids 系统配置ID
+ */
+ deleteById(id: number) {
+ return request({
+ url: `${CONFIG_BASE_URL}/${id}`,
+ method: "delete",
+ });
+ },
+
+ refreshCache() {
+ return request({
+ url: `${CONFIG_BASE_URL}/refresh`,
+ method: "PUT",
+ });
+ },
+};
+
+export default ConfigAPI;
+
+/** $系统配置分页查询参数 */
+export interface ConfigPageQuery extends PageQuery {
+ /** 搜索关键字 */
+ keywords?: string;
+}
+
+/** 系统配置表单对象 */
+export interface ConfigForm {
+ /** 主键 */
+ id?: number;
+ /** 配置名称 */
+ configName?: string;
+ /** 配置键 */
+ configKey?: string;
+ /** 配置值 */
+ configValue?: string;
+ /** 描述、备注 */
+ remark?: string;
+}
+
+/** 系统配置分页对象 */
+export interface ConfigPageVO {
+ /** 主键 */
+ id?: number;
+ /** 配置名称 */
+ configName?: string;
+ /** 配置键 */
+ configKey?: string;
+ /** 配置值 */
+ configValue?: string;
+ /** 描述、备注 */
+ remark?: string;
+}
diff --git a/src/api/system/dept.ts b/src/api/system/dept.ts
new file mode 100644
index 0000000..e3e7530
--- /dev/null
+++ b/src/api/system/dept.ts
@@ -0,0 +1,130 @@
+import request from "@/utils/request";
+
+const DEPT_BASE_URL = "/api/v1/dept";
+
+const DeptAPI = {
+ /**
+ * 获取部门列表
+ *
+ * @param queryParams 查询参数(可选)
+ * @returns 部门树形表格数据
+ */
+ getList(queryParams?: DeptQuery) {
+ return request({
+ url: `${DEPT_BASE_URL}`,
+ method: "get",
+ params: queryParams,
+ });
+ },
+
+ /** 获取部门下拉列表 */
+ getOptions() {
+ return request({
+ url: `${DEPT_BASE_URL}/options`,
+ method: "get",
+ });
+ },
+
+ /**
+ * 获取部门表单数据
+ *
+ * @param id 部门ID
+ * @returns 部门表单数据
+ */
+ getFormData(id: number) {
+ return request({
+ url: `${DEPT_BASE_URL}/${id}/form`,
+ method: "get",
+ });
+ },
+
+ /**
+ * 新增部门
+ *
+ * @param data 部门表单数据
+ * @returns 请求结果
+ */
+ add(data: DeptForm) {
+ return request({
+ url: `${DEPT_BASE_URL}`,
+ method: "post",
+ data: data,
+ });
+ },
+
+ /**
+ * 修改部门
+ *
+ * @param id 部门ID
+ * @param data 部门表单数据
+ * @returns 请求结果
+ */
+ update(id: string, data: DeptForm) {
+ return request({
+ url: `${DEPT_BASE_URL}/${id}`,
+ method: "put",
+ data: data,
+ });
+ },
+
+ /**
+ * 删除部门
+ *
+ * @param ids 部门ID,多个以英文逗号(,)分隔
+ * @returns 请求结果
+ */
+ deleteByIds(ids: string) {
+ return request({
+ url: `${DEPT_BASE_URL}/${ids}`,
+ method: "delete",
+ });
+ },
+};
+
+export default DeptAPI;
+
+/** 部门查询参数 */
+export interface DeptQuery {
+ /** 搜索关键字 */
+ keywords?: string;
+ /** 状态 */
+ status?: number;
+}
+
+/** 部门类型 */
+export interface DeptVO {
+ /** 子部门 */
+ children?: DeptVO[];
+ /** 创建时间 */
+ createTime?: Date;
+ /** 部门ID */
+ id?: number;
+ /** 部门名称 */
+ name?: string;
+ /** 部门编号 */
+ code?: string;
+ /** 父部门ID */
+ parentId?: number;
+ /** 排序 */
+ sort?: number;
+ /** 状态(1:启用;0:禁用) */
+ status?: number;
+ /** 修改时间 */
+ updateTime?: Date;
+}
+
+/** 部门表单类型 */
+export interface DeptForm {
+ /** 部门ID(新增不填) */
+ id?: string;
+ /** 部门名称 */
+ name?: string;
+ /** 部门编号 */
+ code?: string;
+ /** 父部门ID */
+ parentId: string;
+ /** 排序 */
+ sort?: number;
+ /** 状态(1:启用;0:禁用) */
+ status?: number;
+}
diff --git a/src/api/system/dict-data.ts b/src/api/system/dict-data.ts
new file mode 100644
index 0000000..412e1fe
--- /dev/null
+++ b/src/api/system/dict-data.ts
@@ -0,0 +1,162 @@
+import request from "@/utils/request";
+
+const DICT_DATA_BASE_URL = "/api/v1/dict-data";
+
+const DictDataAPI = {
+ /**
+ * 获取字典分页列表
+ *
+ * @param queryParams 查询参数
+ * @returns 字典分页结果
+ */
+ getPage(queryParams: DictDataPageQuery) {
+ return request>({
+ url: `${DICT_DATA_BASE_URL}/page`,
+ method: "get",
+ params: queryParams,
+ });
+ },
+
+ /**
+ * 获取字典数据表单
+ *
+ * @param id 字典ID
+ * @returns 字典数据表单
+ */
+ getFormData(id: number) {
+ return request>({
+ url: `${DICT_DATA_BASE_URL}/${id}/form`,
+ method: "get",
+ });
+ },
+
+ /**
+ * 新增字典数据
+ *
+ * @param data 字典数据
+ */
+ add(data: DictDataForm) {
+ return request({
+ url: `${DICT_DATA_BASE_URL}`,
+ method: "post",
+ data: data,
+ });
+ },
+
+ /**
+ * 修改字典数据
+ *
+ * @param id 字典ID
+ * @param data 字典数据
+ */
+ update(id: number, data: DictDataForm) {
+ return request({
+ url: `${DICT_DATA_BASE_URL}/${id}`,
+ method: "put",
+ data: data,
+ });
+ },
+
+ /**
+ * 删除字典
+ *
+ * @param ids 字典ID,多个以英文逗号(,)分隔
+ */
+ deleteByIds(ids: string) {
+ return request({
+ url: `${DICT_DATA_BASE_URL}/${ids}`,
+ method: "delete",
+ });
+ },
+
+ /**
+ * 获取字典的数据项
+ *
+ * @param dictCode 字典编码
+ * @returns 字典数据项
+ */
+ getOptions(dictCode: string) {
+ return request({
+ url: `${DICT_DATA_BASE_URL}/${dictCode}/options`,
+ method: "get",
+ });
+ },
+};
+
+export default DictDataAPI;
+
+/**
+ * 字典查询参数
+ */
+export interface DictDataPageQuery extends PageQuery {
+ /** 关键字(字典数据值/标签) */
+ keywords?: string;
+
+ /** 字典编码 */
+ dictCode?: string;
+}
+
+/**
+ * 字典分页对象
+ */
+export interface DictDataPageVO {
+ /**
+ * 字典ID
+ */
+ id: number;
+ /**
+ * 字典编码
+ */
+ dictCode: string;
+ /**
+ * 字典数据值
+ */
+ value: string;
+ /**
+ * 字典数据标签
+ */
+ label: string;
+ /**
+ * 状态(1:启用,0:禁用)
+ */
+ status: number;
+ /**
+ * 字典排序
+ */
+ sort?: number;
+}
+
+/**
+ * 字典
+ */
+export interface DictDataForm {
+ /**
+ * 字典ID
+ */
+ id?: number;
+ /**
+ * 字典编码
+ */
+ dictCode?: string;
+ /**
+ * 字典数据值
+ */
+ value?: string;
+ /**
+ * 字典数据标签
+ */
+ label?: string;
+ /**
+ * 状态(1:启用,0:禁用)
+ */
+ status?: number;
+ /**
+ * 字典排序
+ */
+ sort?: number;
+
+ /**
+ * 标签类型
+ */
+ tagType?: "success" | "warning" | "info" | "primary" | "danger" | undefined;
+}
diff --git a/src/api/system/dict.ts b/src/api/system/dict.ts
new file mode 100644
index 0000000..ed29b7d
--- /dev/null
+++ b/src/api/system/dict.ts
@@ -0,0 +1,180 @@
+import request from "@/utils/request";
+
+const DICT_BASE_URL = "/api/v1/dict";
+
+const DictAPI = {
+ /**
+ * 获取字典分页列表
+ *
+ * @param queryParams 查询参数
+ * @returns 字典分页结果
+ */
+ getPage(queryParams: DictPageQuery) {
+ return request>({
+ url: `${DICT_BASE_URL}/page`,
+ method: "get",
+ params: queryParams,
+ });
+ },
+
+ /**
+ * 获取字典表单数据
+ *
+ * @param id 字典ID
+ * @returns 字典表单数据
+ */
+ getFormData(id: number) {
+ return request>({
+ url: `${DICT_BASE_URL}/${id}/form`,
+ method: "get",
+ });
+ },
+
+ /**
+ * 新增字典
+ *
+ * @param data 字典表单数据
+ */
+ add(data: DictForm) {
+ return request({
+ url: `${DICT_BASE_URL}`,
+ method: "post",
+ data: data,
+ });
+ },
+
+ /**
+ * 修改字典
+ *
+ * @param id 字典ID
+ * @param data 字典表单数据
+ */
+ update(id: number, data: DictForm) {
+ return request({
+ url: `${DICT_BASE_URL}/${id}`,
+ method: "put",
+ data: data,
+ });
+ },
+
+ /**
+ * 删除字典
+ *
+ * @param ids 字典ID,多个以英文逗号(,)分隔
+ */
+ deleteByIds(ids: string) {
+ return request({
+ url: `${DICT_BASE_URL}/${ids}`,
+ method: "delete",
+ });
+ },
+
+ /**
+ * 获取字典列表
+ *
+ * @returns 字典列表
+ */
+ getList() {
+ return request({
+ url: `${DICT_BASE_URL}/list`,
+ method: "get",
+ });
+ },
+};
+
+export default DictAPI;
+
+/**
+ * 字典查询参数
+ */
+export interface DictPageQuery extends PageQuery {
+ /**
+ * 关键字(字典名称/编码)
+ */
+ keywords?: string;
+
+ /**
+ * 字典状态(1:启用,0:禁用)
+ */
+ status?: number;
+}
+
+/**
+ * 字典分页对象
+ */
+export interface DictPageVO {
+ /**
+ * 字典ID
+ */
+ id: number;
+ /**
+ * 字典名称
+ */
+ name: string;
+ /**
+ * 字典编码
+ */
+ dictCode: string;
+ /**
+ * 字典状态(1:启用,0:禁用)
+ */
+ status: number;
+}
+
+/**
+ * 字典
+ */
+export interface DictForm {
+ /**
+ * 字典ID
+ */
+ id?: number;
+ /**
+ * 字典名称
+ */
+ name?: string;
+ /**
+ * 字典编码
+ */
+ dictCode?: string;
+ /**
+ * 字典状态(1-启用,0-禁用)
+ */
+ status?: number;
+ /**
+ * 备注
+ */
+ remark?: string;
+}
+
+/**
+ * 字典数据项分页VO
+ *
+ * @description 字典数据分页对象
+ */
+export interface DictVO {
+ /** 字典名称 */
+ name: string;
+
+ /** 字典编码 */
+ dictCode: string;
+
+ /** 字典数据集合 */
+ dictDataList: DictData[];
+}
+
+/**
+ * 字典数据
+ *
+ * @description 字典数据
+ */
+export interface DictData {
+ /** 字典数据值 */
+ value: string;
+
+ /** 字典数据标签 */
+ label: string;
+
+ /** 标签类型 */
+ tagType: string;
+}
diff --git a/src/api/system/log.ts b/src/api/system/log.ts
new file mode 100644
index 0000000..32dbc88
--- /dev/null
+++ b/src/api/system/log.ts
@@ -0,0 +1,121 @@
+import request from "@/utils/request";
+
+const LOG_BASE_URL = "/api/v1/logs";
+
+const LogAPI = {
+ /**
+ * 获取日志分页列表
+ *
+ * @param queryParams 查询参数
+ */
+ getPage(queryParams: LogPageQuery) {
+ return request>({
+ url: `${LOG_BASE_URL}/page`,
+ method: "get",
+ params: queryParams,
+ });
+ },
+
+ /**
+ * 获取访问趋势
+ *
+ * @param queryParams
+ * @returns
+ */
+ getVisitTrend(queryParams: VisitTrendQuery) {
+ return request({
+ url: `${LOG_BASE_URL}/visit-trend`,
+ method: "get",
+ params: queryParams,
+ });
+ },
+
+ /**
+ * 获取访问统计
+ *
+ * @param queryParams
+ * @returns
+ */
+ getVisitStats() {
+ return request({
+ url: `${LOG_BASE_URL}/visit-stats`,
+ method: "get",
+ });
+ },
+};
+
+export default LogAPI;
+
+/**
+ * 日志分页查询对象
+ */
+export interface LogPageQuery extends PageQuery {
+ /** 搜索关键字 */
+ keywords?: string;
+ /** 操作时间 */
+ createTime?: [string, string];
+}
+
+/**
+ * 系统日志分页VO
+ */
+export interface LogPageVO {
+ /** 主键 */
+ id: number;
+ /** 日志模块 */
+ module: string;
+ /** 日志内容 */
+ content: string;
+ /** 请求路径 */
+ requestUri: string;
+ /** 请求方法 */
+ method: string;
+ /** IP 地址 */
+ ip: string;
+ /** 地区 */
+ region: string;
+ /** 浏览器 */
+ browser: string;
+ /** 终端系统 */
+ os: string;
+ /** 执行时间(毫秒) */
+ executionTime: number;
+ /** 操作人 */
+ operator: string;
+}
+
+/** 访问趋势视图对象 */
+export interface VisitTrendVO {
+ /** 日期列表 */
+ dates: string[];
+ /** 浏览量(PV) */
+ pvList: number[];
+ /** 访客数(UV) */
+ uvList: number[];
+ /** IP数 */
+ ipList: number[];
+}
+
+/** 访问趋势查询参数 */
+export interface VisitTrendQuery {
+ /** 开始日期 */
+ startDate: string;
+ /** 结束日期 */
+ endDate: string;
+}
+
+/** 访问统计 */
+export interface VisitStatsVO {
+ /** 今日访客数(UV) */
+ todayUvCount: number;
+ /** 总访客数 */
+ totalUvCount: number;
+ /** 访客数同比增长率(相对于昨天同一时间段的增长率) */
+ uvGrowthRate: number;
+ /** 今日浏览量(PV) */
+ todayPvCount: number;
+ /** 总浏览量 */
+ totalPvCount: number;
+ /** 同比增长率(相对于昨天同一时间段的增长率) */
+ pvGrowthRate: number;
+}
diff --git a/src/api/system/menu.ts b/src/api/system/menu.ts
new file mode 100644
index 0000000..a3d9b63
--- /dev/null
+++ b/src/api/system/menu.ts
@@ -0,0 +1,209 @@
+import request from "@/utils/request";
+// 菜单基础URL
+const MENU_BASE_URL = "/api/v1/menus";
+
+const MenuAPI = {
+ /**
+ * 获取当前用户的路由列表
+ *
+ * 无需传入角色,后端解析token获取角色自行判断是否拥有路由的权限
+ *
+ * @returns 路由列表
+ */
+ getRoutes() {
+ return request({
+ url: `${MENU_BASE_URL}/routes`,
+ method: "get",
+ });
+ },
+
+ /**
+ * 获取菜单树形列表
+ *
+ * @param queryParams 查询参数
+ * @returns 菜单树形列表
+ */
+ getList(queryParams: MenuQuery) {
+ return request({
+ url: `${MENU_BASE_URL}`,
+ method: "get",
+ params: queryParams,
+ });
+ },
+
+ /**
+ * 获取菜单下拉数据源
+ *
+ * @returns 菜单下拉数据源
+ */
+ getOptions(onlyParent?: boolean) {
+ return request({
+ url: `${MENU_BASE_URL}/options`,
+ method: "get",
+ params: { onlyParent: onlyParent },
+ });
+ },
+
+ /**
+ * 获取菜单表单数据
+ *
+ * @param id 菜单ID
+ */
+ getFormData(id: string) {
+ return request({
+ url: `${MENU_BASE_URL}/${id}/form`,
+ method: "get",
+ });
+ },
+
+ /**
+ * 添加菜单
+ *
+ * @param data 菜单表单数据
+ * @returns 请求结果
+ */
+ add(data: MenuForm) {
+ return request({
+ url: `${MENU_BASE_URL}`,
+ method: "post",
+ data: data,
+ });
+ },
+
+ /**
+ * 修改菜单
+ *
+ * @param id 菜单ID
+ * @param data 菜单表单数据
+ * @returns 请求结果
+ */
+ update(id: string, data: MenuForm) {
+ return request({
+ url: `${MENU_BASE_URL}/${id}`,
+ method: "put",
+ data: data,
+ });
+ },
+
+ /**
+ * 删除菜单
+ *
+ * @param id 菜单ID
+ * @returns 请求结果
+ */
+ deleteById(id: number) {
+ return request({
+ url: `${MENU_BASE_URL}/${id}`,
+ method: "delete",
+ });
+ },
+};
+
+export default MenuAPI;
+
+import type { MenuTypeEnum } from "@/enums/MenuTypeEnum";
+
+/** 菜单查询参数 */
+export interface MenuQuery {
+ /** 搜索关键字 */
+ keywords?: string;
+}
+
+/** 菜单视图对象 */
+export interface MenuVO {
+ /** 子菜单 */
+ children?: MenuVO[];
+ /** 组件路径 */
+ component?: string;
+ /** ICON */
+ icon?: string;
+ /** 菜单ID */
+ id?: string;
+ /** 菜单名称 */
+ name?: string;
+ /** 父菜单ID */
+ parentId?: string;
+ /** 按钮权限标识 */
+ perm?: string;
+ /** 跳转路径 */
+ redirect?: string;
+ /** 路由名称 */
+ routeName?: string;
+ /** 路由相对路径 */
+ routePath?: string;
+ /** 菜单排序(数字越小排名越靠前) */
+ sort?: number;
+ /** 菜单 */
+ type?: MenuTypeEnum;
+ /** 菜单是否可见(1:显示;0:隐藏) */
+ visible?: number;
+}
+
+/** 菜单表单对象 */
+export interface MenuForm {
+ /** 菜单ID */
+ id?: string;
+ /** 父菜单ID */
+ parentId?: string;
+ /** 菜单名称 */
+ name?: string;
+ /** 菜单是否可见(1-是 0-否) */
+ visible: number;
+ /** ICON */
+ icon?: string;
+ /** 排序 */
+ sort?: number;
+ /** 路由名称 */
+ routeName?: string;
+ /** 路由路径 */
+ routePath?: string;
+ /** 组件路径 */
+ component?: string;
+ /** 跳转路由路径 */
+ redirect?: string;
+ /** 菜单 */
+ type?: MenuTypeEnum;
+ /** 权限标识 */
+ perm?: string;
+ /** 【菜单】是否开启页面缓存 */
+ keepAlive?: number;
+ /** 【目录】只有一个子路由是否始终显示 */
+ alwaysShow?: number;
+ /** 参数 */
+ params?: KeyValue[];
+}
+
+interface KeyValue {
+ key: string;
+ value: string;
+}
+
+/** RouteVO,路由对象 */
+export interface RouteVO {
+ /** 子路由列表 */
+ children: RouteVO[];
+ /** 组件路径 */
+ component?: string;
+ /** 路由属性 */
+ meta?: Meta;
+ /** 路由名称 */
+ name?: string;
+ /** 路由路径 */
+ path?: string;
+ /** 跳转链接 */
+ redirect?: string;
+}
+
+/** Meta,路由属性 */
+export interface Meta {
+ /** 【目录】只有一个子路由是否始终显示 */
+ alwaysShow?: boolean;
+ /** 是否隐藏(true-是 false-否) */
+ hidden?: boolean;
+ /** ICON */
+ icon?: string;
+ /** 【菜单】是否开启页面缓存 */
+ keepAlive?: boolean;
+ /** 路由title */
+ title?: string;
+}
diff --git a/src/api/system/notice.ts b/src/api/system/notice.ts
new file mode 100644
index 0000000..17bb1ef
--- /dev/null
+++ b/src/api/system/notice.ts
@@ -0,0 +1,199 @@
+import request from "@/utils/request";
+
+const NOTICE_BASE_URL = "/api/v1/notices";
+
+const NoticeAPI = {
+ /** 获取通知公告分页数据 */
+ getPage(queryParams?: NoticePageQuery) {
+ return request>({
+ url: `${NOTICE_BASE_URL}/page`,
+ method: "get",
+ params: queryParams,
+ });
+ },
+
+ /**
+ * 获取通知公告表单数据
+ *
+ * @param id NoticeID
+ * @returns Notice表单数据
+ */
+ getFormData(id: number) {
+ return request({
+ url: `${NOTICE_BASE_URL}/${id}/form`,
+ method: "get",
+ });
+ },
+
+ /**
+ * 添加通知公告
+ *
+ * @param data Notice表单数据
+ * @returns
+ */
+ add(data: NoticeForm) {
+ return request({
+ url: `${NOTICE_BASE_URL}`,
+ method: "post",
+ data: data,
+ });
+ },
+
+ /**
+ * 更新通知公告
+ *
+ * @param id NoticeID
+ * @param data Notice表单数据
+ */
+ update(id: number, data: NoticeForm) {
+ return request({
+ url: `${NOTICE_BASE_URL}/${id}`,
+ method: "put",
+ data: data,
+ });
+ },
+
+ /**
+ * 批量删除通知公告,多个以英文逗号(,)分割
+ *
+ * @param ids 通知公告ID字符串,多个以英文逗号(,)分割
+ */
+ deleteByIds(ids: string) {
+ return request({
+ url: `${NOTICE_BASE_URL}/${ids}`,
+ method: "delete",
+ });
+ },
+
+ /**
+ * 发布通知
+ *
+ * @param id 被发布的通知公告id
+ * @returns
+ */
+ publish(id: number) {
+ return request({
+ url: `${NOTICE_BASE_URL}/${id}/publish`,
+ method: "put",
+ });
+ },
+
+ /**
+ * 撤回通知
+ *
+ * @param id 撤回的通知id
+ * @returns
+ */
+ revoke(id: number) {
+ return request({
+ url: `${NOTICE_BASE_URL}/${id}/revoke`,
+ method: "put",
+ });
+ },
+ /**
+ * 查看通知
+ *
+ * @param id
+ */
+ getDetail(id: string) {
+ return request({
+ url: `${NOTICE_BASE_URL}/${id}/detail`,
+ method: "get",
+ });
+ },
+
+ /* 全部已读 */
+ readAll() {
+ return request({
+ url: `${NOTICE_BASE_URL}/read-all`,
+ method: "put",
+ });
+ },
+
+ /** 获取我的通知分页列表 */
+ getMyNoticePage(queryParams?: NoticePageQuery) {
+ return request>({
+ url: `${NOTICE_BASE_URL}/my-page`,
+ method: "get",
+ params: queryParams,
+ });
+ },
+};
+
+export default NoticeAPI;
+
+/** 通知公告分页查询参数 */
+export interface NoticePageQuery extends PageQuery {
+ /** 标题 */
+ title?: string;
+ /** 发布状态(0:未发布,1:已发布,-1:已撤回) */
+ publishStatus?: number;
+
+ isRead?: number;
+}
+
+/** 通知公告表单对象 */
+export interface NoticeForm {
+ id?: number;
+ /** 通知标题 */
+ title?: string;
+ /** 通知内容 */
+ content?: string;
+ /** 通知类型 */
+ type?: number;
+ /** 优先级(L:低,M:中,H:高) */
+ level?: string;
+ /** 目标类型(1-全体 2-指定) */
+ targetType?: number;
+ /** 目标ID合集,以,分割 */
+ targetUserIds?: string;
+}
+
+/** 通知公告分页对象 */
+export interface NoticePageVO {
+ id: string;
+ /** 通知标题 */
+ title?: string;
+ /** 通知内容 */
+ content?: string;
+ /** 通知类型 */
+ type?: number;
+ /** 发布人 */
+ publisherId?: bigint;
+ /** 优先级(0-低 1-中 2-高) */
+ priority?: number;
+ /** 目标类型(0-全体 1-指定) */
+ targetType?: number;
+ /** 发布状态(0-未发布 1已发布 2已撤回) */
+ publishStatus?: number;
+ /** 发布时间 */
+ publishTime?: Date;
+ /** 撤回时间 */
+ revokeTime?: Date;
+}
+
+export interface NoticeDetailVO {
+ /** 通知ID */
+ id?: string;
+
+ /** 通知标题 */
+ title?: string;
+
+ /** 通知内容 */
+ content?: string;
+
+ /** 通知类型 */
+ type?: number;
+
+ /** 发布人 */
+ publisherName?: string;
+
+ /** 优先级(L-低 M-中 H-高) */
+ level?: string;
+
+ /** 发布时间 */
+ publishTime?: Date;
+
+ /** 发布状态 */
+ publishStatus?: number;
+}
diff --git a/src/api/system/role.ts b/src/api/system/role.ts
new file mode 100644
index 0000000..cc21e34
--- /dev/null
+++ b/src/api/system/role.ts
@@ -0,0 +1,138 @@
+import request from "@/utils/request";
+
+const ROLE_BASE_URL = "/api/v1/roles";
+
+const RoleAPI = {
+ /** 获取角色分页数据 */
+ getPage(queryParams?: RolePageQuery) {
+ return request>({
+ url: `${ROLE_BASE_URL}/page`,
+ method: "get",
+ params: queryParams,
+ });
+ },
+
+ /** 获取角色下拉数据源 */
+ getOptions() {
+ return request({
+ url: `${ROLE_BASE_URL}/options`,
+ method: "get",
+ });
+ },
+ /**
+ * 获取角色的菜单ID集合
+ *
+ * @param roleId 角色ID
+ * @returns 角色的菜单ID集合
+ */
+ getRoleMenuIds(roleId: number) {
+ return request({
+ url: `${ROLE_BASE_URL}/${roleId}/menuIds`,
+ method: "get",
+ });
+ },
+
+ /**
+ * 分配菜单权限
+ *
+ * @param roleId 角色ID
+ * @param data 菜单ID集合
+ */
+ updateRoleMenus(roleId: number, data: number[]) {
+ return request({
+ url: `${ROLE_BASE_URL}/${roleId}/menus`,
+ method: "put",
+ data: data,
+ });
+ },
+
+ /**
+ * 获取角色表单数据
+ *
+ * @param id 角色ID
+ * @returns 角色表单数据
+ */
+ getFormData(id: number) {
+ return request({
+ url: `${ROLE_BASE_URL}/${id}/form`,
+ method: "get",
+ });
+ },
+
+ /** 添加角色 */
+ add(data: RoleForm) {
+ return request({
+ url: `${ROLE_BASE_URL}`,
+ method: "post",
+ data: data,
+ });
+ },
+
+ /**
+ * 更新角色
+ *
+ * @param id 角色ID
+ * @param data 角色表单数据
+ */
+ update(id: number, data: RoleForm) {
+ return request({
+ url: `${ROLE_BASE_URL}/${id}`,
+ method: "put",
+ data: data,
+ });
+ },
+
+ /**
+ * 批量删除角色,多个以英文逗号(,)分割
+ *
+ * @param ids 角色ID字符串,多个以英文逗号(,)分割
+ */
+ deleteByIds(ids: string) {
+ return request({
+ url: `${ROLE_BASE_URL}/${ids}`,
+ method: "delete",
+ });
+ },
+};
+
+export default RoleAPI;
+
+/** 角色分页查询参数 */
+export interface RolePageQuery extends PageQuery {
+ /** 搜索关键字 */
+ keywords?: string;
+}
+
+/** 角色分页对象 */
+export interface RolePageVO {
+ /** 角色编码 */
+ code?: string;
+ /** 角色ID */
+ id?: number;
+ /** 角色名称 */
+ name?: string;
+ /** 排序 */
+ sort?: number;
+ /** 角色状态 */
+ status?: number;
+ /** 创建时间 */
+ createTime?: Date;
+ /** 修改时间 */
+ updateTime?: Date;
+}
+
+/** 角色表单对象 */
+export interface RoleForm {
+ /** 角色ID */
+ id?: number;
+ /** 角色编码 */
+ code?: string;
+ /** 数据权限 */
+ dataScope?: number;
+ /** 角色名称 */
+ name?: string;
+ /** 排序 */
+ sort?: number;
+ /** 角色状态(1-正常;0-停用) */
+ status?: number;
+}
diff --git a/src/api/system/user.ts b/src/api/system/user.ts
new file mode 100644
index 0000000..b58e6f5
--- /dev/null
+++ b/src/api/system/user.ts
@@ -0,0 +1,384 @@
+import request from "@/utils/request";
+
+const USER_BASE_URL = "/api/v1/users";
+
+const UserAPI = {
+ /**
+ * 获取当前登录用户信息
+ *
+ * @returns 登录用户昵称、头像信息,包括角色和权限
+ */
+ getInfo() {
+ return request({
+ url: `${USER_BASE_URL}/me`,
+ method: "get",
+ });
+ },
+
+ /**
+ * 获取用户分页列表
+ *
+ * @param queryParams 查询参数
+ */
+ getPage(queryParams: UserPageQuery) {
+ return request>({
+ url: `${USER_BASE_URL}/page`,
+ method: "get",
+ params: queryParams,
+ });
+ },
+
+ /**
+ * 获取用户表单详情
+ *
+ * @param userId 用户ID
+ * @returns 用户表单详情
+ */
+ getFormData(userId: number) {
+ return request({
+ url: `${USER_BASE_URL}/${userId}/form`,
+ method: "get",
+ });
+ },
+
+ /**
+ * 添加用户
+ *
+ * @param data 用户表单数据
+ */
+ add(data: UserForm) {
+ return request({
+ url: `${USER_BASE_URL}`,
+ method: "post",
+ data: data,
+ });
+ },
+
+ /**
+ * 修改用户
+ *
+ * @param id 用户ID
+ * @param data 用户表单数据
+ */
+ update(id: number, data: UserForm) {
+ return request({
+ url: `${USER_BASE_URL}/${id}`,
+ method: "put",
+ data: data,
+ });
+ },
+
+ /**
+ * 修改用户密码
+ *
+ * @param id 用户ID
+ * @param password 新密码
+ */
+ resetPassword(id: number, password: string) {
+ return request({
+ url: `${USER_BASE_URL}/${id}/password/reset`,
+ method: "put",
+ params: { password: password },
+ });
+ },
+
+ /**
+ * 批量删除用户,多个以英文逗号(,)分割
+ *
+ * @param ids 用户ID字符串,多个以英文逗号(,)分割
+ */
+ deleteByIds(ids: string) {
+ return request({
+ url: `${USER_BASE_URL}/${ids}`,
+ method: "delete",
+ });
+ },
+
+ /** 下载用户导入模板 */
+ downloadTemplate() {
+ return request({
+ url: `${USER_BASE_URL}/template`,
+ method: "get",
+ responseType: "blob",
+ });
+ },
+
+ /**
+ * 导出用户
+ *
+ * @param queryParams 查询参数
+ */
+ export(queryParams: UserPageQuery) {
+ return request({
+ url: `${USER_BASE_URL}/export`,
+ method: "get",
+ params: queryParams,
+ responseType: "blob",
+ });
+ },
+
+ /**
+ * 导入用户
+ *
+ * @param deptId 部门ID
+ * @param file 导入文件
+ */
+ import(deptId: number, file: File) {
+ const formData = new FormData();
+ formData.append("file", file);
+ return request({
+ url: `${USER_BASE_URL}/import`,
+ method: "post",
+ params: { deptId: deptId },
+ data: formData,
+ headers: {
+ "Content-Type": "multipart/form-data",
+ },
+ });
+ },
+
+ /** 获取个人中心用户信息 */
+ getProfile() {
+ return request({
+ url: `${USER_BASE_URL}/profile`,
+ method: "get",
+ });
+ },
+
+ /** 修改个人中心用户信息 */
+ updateProfile(data: UserProfileForm) {
+ return request({
+ url: `${USER_BASE_URL}/profile`,
+ method: "put",
+ data: data,
+ });
+ },
+
+ /** 修改个人中心用户密码 */
+ changePassword(data: PasswordChangeForm) {
+ return request({
+ url: `${USER_BASE_URL}/password`,
+ method: "put",
+ data: data,
+ });
+ },
+
+ /** 发送短信验证码(绑定或更换手机号)*/
+ sendMobileCode(mobile: string) {
+ return request({
+ url: `${USER_BASE_URL}/mobile/code`,
+ method: "post",
+ params: { mobile: mobile },
+ });
+ },
+
+ /** 绑定或更换手机号 */
+ bindOrChangeMobile(data: MobileUpdateForm) {
+ return request({
+ url: `${USER_BASE_URL}/mobile`,
+ method: "put",
+ data: data,
+ });
+ },
+
+ /** 发送邮箱验证码(绑定或更换邮箱)*/
+ sendEmailCode(email: string) {
+ return request({
+ url: `${USER_BASE_URL}/email/code`,
+ method: "post",
+ params: { email: email },
+ });
+ },
+
+ /** 绑定或更换邮箱 */
+ bindOrChangeEmail(data: EmailUpdateForm) {
+ return request({
+ url: `${USER_BASE_URL}/email`,
+ method: "put",
+ data: data,
+ });
+ },
+
+ /**
+ * 获取用户下拉列表
+ */
+ getOptions() {
+ return request({
+ url: `${USER_BASE_URL}/options`,
+ method: "get",
+ });
+ },
+};
+
+export default UserAPI;
+
+/** 登录用户信息 */
+export interface UserInfo {
+ /** 用户ID */
+ userId?: number;
+
+ /** 用户名 */
+ username?: string;
+
+ /** 昵称 */
+ nickname?: string;
+
+ /** 头像URL */
+ avatar?: string;
+
+ /** 角色 */
+ roles: string[];
+
+ /** 权限 */
+ perms: string[];
+}
+
+/**
+ * 用户分页查询对象
+ */
+export interface UserPageQuery extends PageQuery {
+ /** 搜索关键字 */
+ keywords?: string;
+
+ /** 用户状态 */
+ status?: number;
+
+ /** 部门ID */
+ deptId?: number;
+
+ /** 开始时间 */
+ createTime?: [string, string];
+}
+
+/** 用户分页对象 */
+export interface UserPageVO {
+ /** 用户ID */
+ id: number;
+ /** 用户头像URL */
+ avatar?: string;
+ /** 创建时间 */
+ createTime?: Date;
+ /** 部门名称 */
+ deptName?: string;
+ /** 用户邮箱 */
+ email?: string;
+ /** 性别 */
+ gender?: number;
+ /** 手机号 */
+ mobile?: string;
+ /** 用户昵称 */
+ nickname?: string;
+ /** 角色名称,多个使用英文逗号(,)分割 */
+ roleNames?: string;
+ /** 用户状态(1:启用;0:禁用) */
+ status?: number;
+ /** 用户名 */
+ username?: string;
+}
+
+/** 用户表单类型 */
+export interface UserForm {
+ /** 用户头像 */
+ avatar?: string;
+ /** 部门ID */
+ deptId?: number;
+ /** 邮箱 */
+ email?: string;
+ /** 性别 */
+ gender?: number;
+ /** 用户ID */
+ id?: number;
+ /** 手机号 */
+ mobile?: string;
+ /** 昵称 */
+ nickname?: string;
+ /** 角色ID集合 */
+ roleIds?: number[];
+ /** 用户状态(1:正常;0:禁用) */
+ status?: number;
+ /** 用户名 */
+ username?: string;
+}
+
+/** 个人中心用户信息 */
+export interface UserProfileVO {
+ /** 用户ID */
+ id?: number;
+
+ /** 用户名 */
+ username?: string;
+
+ /** 昵称 */
+ nickname?: string;
+
+ /** 头像URL */
+ avatar?: string;
+
+ /** 性别 */
+ gender?: number;
+
+ /** 手机号 */
+ mobile?: string;
+
+ /** 邮箱 */
+ email?: string;
+
+ /** 部门名称 */
+ deptName?: string;
+
+ /** 角色名称,多个使用英文逗号(,)分割 */
+ roleNames?: string;
+
+ /** 创建时间 */
+ createTime?: Date;
+}
+
+/** 个人中心用户信息表单 */
+export interface UserProfileForm {
+ /** 用户ID */
+ id?: number;
+
+ /** 用户名 */
+ username?: string;
+
+ /** 昵称 */
+ nickname?: string;
+
+ /** 头像URL */
+ avatar?: string;
+
+ /** 性别 */
+ gender?: number;
+
+ /** 手机号 */
+ mobile?: string;
+
+ /** 邮箱 */
+ email?: string;
+}
+
+/** 修改密码表单 */
+export interface PasswordChangeForm {
+ /** 原密码 */
+ oldPassword?: string;
+ /** 新密码 */
+ newPassword?: string;
+ /** 确认新密码 */
+ confirmPassword?: string;
+}
+
+/** 修改手机表单 */
+export interface MobileUpdateForm {
+ /** 手机号 */
+ mobile?: string;
+ /** 验证码 */
+ code?: string;
+}
+
+/** 修改邮箱表单 */
+export interface EmailUpdateForm {
+ /** 邮箱 */
+ email?: string;
+ /** 验证码 */
+ code?: string;
+}
diff --git a/src/assets/icons/api.svg b/src/assets/icons/api.svg
new file mode 100644
index 0000000..0181bdd
--- /dev/null
+++ b/src/assets/icons/api.svg
@@ -0,0 +1 @@
+
\ No newline at end of file
diff --git a/src/assets/icons/backtop.svg b/src/assets/icons/backtop.svg
new file mode 100644
index 0000000..f8e6aa0
--- /dev/null
+++ b/src/assets/icons/backtop.svg
@@ -0,0 +1 @@
+
\ No newline at end of file
diff --git a/src/assets/icons/bilibili.svg b/src/assets/icons/bilibili.svg
new file mode 100644
index 0000000..b86747c
--- /dev/null
+++ b/src/assets/icons/bilibili.svg
@@ -0,0 +1 @@
+
diff --git a/src/assets/icons/browser.svg b/src/assets/icons/browser.svg
new file mode 100644
index 0000000..15c3927
--- /dev/null
+++ b/src/assets/icons/browser.svg
@@ -0,0 +1 @@
+
diff --git a/src/assets/icons/captcha.svg b/src/assets/icons/captcha.svg
new file mode 100644
index 0000000..8b1da30
--- /dev/null
+++ b/src/assets/icons/captcha.svg
@@ -0,0 +1 @@
+
\ No newline at end of file
diff --git a/src/assets/icons/cascader.svg b/src/assets/icons/cascader.svg
new file mode 100644
index 0000000..57209bf
--- /dev/null
+++ b/src/assets/icons/cascader.svg
@@ -0,0 +1 @@
+
\ No newline at end of file
diff --git a/src/assets/icons/client.svg b/src/assets/icons/client.svg
new file mode 100644
index 0000000..7373b3d
--- /dev/null
+++ b/src/assets/icons/client.svg
@@ -0,0 +1 @@
+
\ No newline at end of file
diff --git a/src/assets/icons/close.svg b/src/assets/icons/close.svg
new file mode 100644
index 0000000..e99c978
--- /dev/null
+++ b/src/assets/icons/close.svg
@@ -0,0 +1 @@
+
\ No newline at end of file
diff --git a/src/assets/icons/close_all.svg b/src/assets/icons/close_all.svg
new file mode 100644
index 0000000..2005198
--- /dev/null
+++ b/src/assets/icons/close_all.svg
@@ -0,0 +1 @@
+
\ No newline at end of file
diff --git a/src/assets/icons/close_left.svg b/src/assets/icons/close_left.svg
new file mode 100644
index 0000000..fc5cf71
--- /dev/null
+++ b/src/assets/icons/close_left.svg
@@ -0,0 +1 @@
+
\ No newline at end of file
diff --git a/src/assets/icons/close_other.svg b/src/assets/icons/close_other.svg
new file mode 100644
index 0000000..27ffc32
--- /dev/null
+++ b/src/assets/icons/close_other.svg
@@ -0,0 +1 @@
+
\ No newline at end of file
diff --git a/src/assets/icons/close_right.svg b/src/assets/icons/close_right.svg
new file mode 100644
index 0000000..b96dc1c
--- /dev/null
+++ b/src/assets/icons/close_right.svg
@@ -0,0 +1 @@
+
\ No newline at end of file
diff --git a/src/assets/icons/cnblogs.svg b/src/assets/icons/cnblogs.svg
new file mode 100644
index 0000000..4920a4c
--- /dev/null
+++ b/src/assets/icons/cnblogs.svg
@@ -0,0 +1 @@
+
diff --git a/src/assets/icons/code.svg b/src/assets/icons/code.svg
new file mode 100644
index 0000000..d8b546c
--- /dev/null
+++ b/src/assets/icons/code.svg
@@ -0,0 +1 @@
+
diff --git a/src/assets/icons/collapse.svg b/src/assets/icons/collapse.svg
new file mode 100644
index 0000000..1507568
--- /dev/null
+++ b/src/assets/icons/collapse.svg
@@ -0,0 +1 @@
+
\ No newline at end of file
diff --git a/src/assets/icons/csdn.svg b/src/assets/icons/csdn.svg
new file mode 100644
index 0000000..e16bad0
--- /dev/null
+++ b/src/assets/icons/csdn.svg
@@ -0,0 +1,6 @@
+
+ ic/csdn
+
+
+
+
diff --git a/src/assets/icons/data_statistics.svg b/src/assets/icons/data_statistics.svg
new file mode 100644
index 0000000..74a7adf
--- /dev/null
+++ b/src/assets/icons/data_statistics.svg
@@ -0,0 +1 @@
+
\ No newline at end of file
diff --git a/src/assets/icons/dict.svg b/src/assets/icons/dict.svg
new file mode 100644
index 0000000..db60220
--- /dev/null
+++ b/src/assets/icons/dict.svg
@@ -0,0 +1 @@
+
\ No newline at end of file
diff --git a/src/assets/icons/document.svg b/src/assets/icons/document.svg
new file mode 100644
index 0000000..aaa0574
--- /dev/null
+++ b/src/assets/icons/document.svg
@@ -0,0 +1 @@
+
\ No newline at end of file
diff --git a/src/assets/icons/down.svg b/src/assets/icons/down.svg
new file mode 100644
index 0000000..5fc8b88
--- /dev/null
+++ b/src/assets/icons/down.svg
@@ -0,0 +1 @@
+
diff --git a/src/assets/icons/download.svg b/src/assets/icons/download.svg
new file mode 100644
index 0000000..a8077dc
--- /dev/null
+++ b/src/assets/icons/download.svg
@@ -0,0 +1 @@
+
\ No newline at end of file
diff --git a/src/assets/icons/enter.svg b/src/assets/icons/enter.svg
new file mode 100644
index 0000000..9e199df
--- /dev/null
+++ b/src/assets/icons/enter.svg
@@ -0,0 +1 @@
+
diff --git a/src/assets/icons/esc.svg b/src/assets/icons/esc.svg
new file mode 100644
index 0000000..2f85dd2
--- /dev/null
+++ b/src/assets/icons/esc.svg
@@ -0,0 +1 @@
+
diff --git a/src/assets/icons/file.svg b/src/assets/icons/file.svg
new file mode 100644
index 0000000..fac9bf0
--- /dev/null
+++ b/src/assets/icons/file.svg
@@ -0,0 +1 @@
+
diff --git a/src/assets/icons/fullscreen-exit.svg b/src/assets/icons/fullscreen-exit.svg
new file mode 100644
index 0000000..2452f2b
--- /dev/null
+++ b/src/assets/icons/fullscreen-exit.svg
@@ -0,0 +1 @@
+
\ No newline at end of file
diff --git a/src/assets/icons/fullscreen.svg b/src/assets/icons/fullscreen.svg
new file mode 100644
index 0000000..4b6ee11
--- /dev/null
+++ b/src/assets/icons/fullscreen.svg
@@ -0,0 +1 @@
+
\ No newline at end of file
diff --git a/src/assets/icons/gitcode.svg b/src/assets/icons/gitcode.svg
new file mode 100644
index 0000000..7a02760
--- /dev/null
+++ b/src/assets/icons/gitcode.svg
@@ -0,0 +1 @@
+
diff --git a/src/assets/icons/gitee.svg b/src/assets/icons/gitee.svg
new file mode 100644
index 0000000..c799c2f
--- /dev/null
+++ b/src/assets/icons/gitee.svg
@@ -0,0 +1 @@
+
diff --git a/src/assets/icons/github.svg b/src/assets/icons/github.svg
new file mode 100644
index 0000000..1adfa4e
--- /dev/null
+++ b/src/assets/icons/github.svg
@@ -0,0 +1 @@
+
\ No newline at end of file
diff --git a/src/assets/icons/homepage.svg b/src/assets/icons/homepage.svg
new file mode 100644
index 0000000..1e1feab
--- /dev/null
+++ b/src/assets/icons/homepage.svg
@@ -0,0 +1 @@
+
\ No newline at end of file
diff --git a/src/assets/icons/java.svg b/src/assets/icons/java.svg
new file mode 100644
index 0000000..eaa93db
--- /dev/null
+++ b/src/assets/icons/java.svg
@@ -0,0 +1 @@
+
diff --git a/src/assets/icons/juejin.svg b/src/assets/icons/juejin.svg
new file mode 100644
index 0000000..937ace3
--- /dev/null
+++ b/src/assets/icons/juejin.svg
@@ -0,0 +1 @@
+
diff --git a/src/assets/icons/language.svg b/src/assets/icons/language.svg
new file mode 100644
index 0000000..e754062
--- /dev/null
+++ b/src/assets/icons/language.svg
@@ -0,0 +1 @@
+
\ No newline at end of file
diff --git a/src/assets/icons/menu.svg b/src/assets/icons/menu.svg
new file mode 100644
index 0000000..f5875d3
--- /dev/null
+++ b/src/assets/icons/menu.svg
@@ -0,0 +1 @@
+
\ No newline at end of file
diff --git a/src/assets/icons/message.svg b/src/assets/icons/message.svg
new file mode 100644
index 0000000..deacdc3
--- /dev/null
+++ b/src/assets/icons/message.svg
@@ -0,0 +1 @@
+
diff --git a/src/assets/icons/monitor.svg b/src/assets/icons/monitor.svg
new file mode 100644
index 0000000..f153b9c
--- /dev/null
+++ b/src/assets/icons/monitor.svg
@@ -0,0 +1 @@
+
\ No newline at end of file
diff --git a/src/assets/icons/project.svg b/src/assets/icons/project.svg
new file mode 100644
index 0000000..eaf6a12
--- /dev/null
+++ b/src/assets/icons/project.svg
@@ -0,0 +1 @@
+
diff --git a/src/assets/icons/qq.svg b/src/assets/icons/qq.svg
new file mode 100644
index 0000000..a59086b
--- /dev/null
+++ b/src/assets/icons/qq.svg
@@ -0,0 +1 @@
+
diff --git a/src/assets/icons/refresh.svg b/src/assets/icons/refresh.svg
new file mode 100644
index 0000000..e598ed1
--- /dev/null
+++ b/src/assets/icons/refresh.svg
@@ -0,0 +1 @@
+
\ No newline at end of file
diff --git a/src/assets/icons/role.svg b/src/assets/icons/role.svg
new file mode 100644
index 0000000..5d25278
--- /dev/null
+++ b/src/assets/icons/role.svg
@@ -0,0 +1 @@
+
\ No newline at end of file
diff --git a/src/assets/icons/search.svg b/src/assets/icons/search.svg
new file mode 100644
index 0000000..2312daf
--- /dev/null
+++ b/src/assets/icons/search.svg
@@ -0,0 +1 @@
+
diff --git a/src/assets/icons/setting.svg b/src/assets/icons/setting.svg
new file mode 100644
index 0000000..fbc4945
--- /dev/null
+++ b/src/assets/icons/setting.svg
@@ -0,0 +1 @@
+
\ No newline at end of file
diff --git a/src/assets/icons/size.svg b/src/assets/icons/size.svg
new file mode 100644
index 0000000..f92f852
--- /dev/null
+++ b/src/assets/icons/size.svg
@@ -0,0 +1 @@
+
\ No newline at end of file
diff --git a/src/assets/icons/system.svg b/src/assets/icons/system.svg
new file mode 100644
index 0000000..2e6045b
--- /dev/null
+++ b/src/assets/icons/system.svg
@@ -0,0 +1 @@
+
\ No newline at end of file
diff --git a/src/assets/icons/table.svg b/src/assets/icons/table.svg
new file mode 100644
index 0000000..1a16abb
--- /dev/null
+++ b/src/assets/icons/table.svg
@@ -0,0 +1 @@
+
\ No newline at end of file
diff --git a/src/assets/icons/todo.svg b/src/assets/icons/todo.svg
new file mode 100644
index 0000000..f48e667
--- /dev/null
+++ b/src/assets/icons/todo.svg
@@ -0,0 +1 @@
+
diff --git a/src/assets/icons/tree.svg b/src/assets/icons/tree.svg
new file mode 100644
index 0000000..51aea8f
--- /dev/null
+++ b/src/assets/icons/tree.svg
@@ -0,0 +1 @@
+
\ No newline at end of file
diff --git a/src/assets/icons/typescript.svg b/src/assets/icons/typescript.svg
new file mode 100644
index 0000000..781d6f8
--- /dev/null
+++ b/src/assets/icons/typescript.svg
@@ -0,0 +1 @@
+
diff --git a/src/assets/icons/up.svg b/src/assets/icons/up.svg
new file mode 100644
index 0000000..3b6c535
--- /dev/null
+++ b/src/assets/icons/up.svg
@@ -0,0 +1 @@
+
diff --git a/src/assets/icons/user.svg b/src/assets/icons/user.svg
new file mode 100644
index 0000000..8e693ec
--- /dev/null
+++ b/src/assets/icons/user.svg
@@ -0,0 +1 @@
+
diff --git a/src/assets/icons/visitor.svg b/src/assets/icons/visitor.svg
new file mode 100644
index 0000000..1fd8dbe
--- /dev/null
+++ b/src/assets/icons/visitor.svg
@@ -0,0 +1 @@
+
diff --git a/src/assets/icons/vue.svg b/src/assets/icons/vue.svg
new file mode 100644
index 0000000..456f876
--- /dev/null
+++ b/src/assets/icons/vue.svg
@@ -0,0 +1 @@
+
diff --git a/src/assets/icons/wechat.svg b/src/assets/icons/wechat.svg
new file mode 100644
index 0000000..2fc5803
--- /dev/null
+++ b/src/assets/icons/wechat.svg
@@ -0,0 +1 @@
+
diff --git a/src/assets/icons/xml.svg b/src/assets/icons/xml.svg
new file mode 100644
index 0000000..f041213
--- /dev/null
+++ b/src/assets/icons/xml.svg
@@ -0,0 +1 @@
+
diff --git a/src/assets/images/1024.png b/src/assets/images/1024.png
new file mode 100644
index 0000000000000000000000000000000000000000..716381bea35f0257e570c507df1703578a0e8634
GIT binary patch
literal 1797
zcmeAS@N?(olHy`uVBq!ia0y~yV4MKL985qF{<{lj11Zh|kH}&M2EHR8%s5q>Pnv;&
zZM&z7V@SoEw|5--t~dy|I`;hf7oNAR!?jyzvc-Liq~2Llx2slN-GA@>cm9T^`Q`Eq
z5Ar$1CMZTYv?VY~HnFB~(13F+`gpj8pu+e@=<
z$DP0YZIYjDuKnrnKev)>#);B)c9M0;YSXe=%8pzPM`t%P-%z*Ye*!_t_+m
z)YK4j`M3KA{cUe63rUa6xN`Xi@xJd>=l=V*<>hDp_nx`c@wdNi-Zr=Xb|uLk0!GK&
zRqvyH|Nq0w2j2fam;k@
z`YmU_)Lq#2{O6XvOP1fH5@9snrv)RX8zZw4D-u8FF<@@%w|G(U~-dq2~X1;Ci
zdy>;yPj=P%_UfzOXV=W1@B2P6xB4H+5#94X(f9pVo3q<$7v;X+{;y`ve`At!FDSzO
zZ+~l8=KJ0g6j9%|zI}gpoB8&)f0mORQ{XII_5Z~cVE6S3gUd)WY!FhJ8C<)@r=_y5+A9$pQ1JI^yzg>r!A|
zT0ZaFzE|gqZ~rbLwM=+C>)!5}_n50@<^FbA?zg`^xBmT?pWEN212Yvlkoo>XS3j3^P6
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/src/assets/images/404.svg b/src/assets/images/404.svg
new file mode 100644
index 0000000..b9bf23c
--- /dev/null
+++ b/src/assets/images/404.svg
@@ -0,0 +1,340 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/src/assets/images/avatar.png b/src/assets/images/avatar.png
new file mode 100644
index 0000000000000000000000000000000000000000..997732a452cdf6273ec7cd444dab69f1a4db4e48
GIT binary patch
literal 1865
zcmZ`)c{tQtAODTza*@X1N<*aVWTtDs?k%!jYoY}WLiX#HElY(Ex>s3Bwz^c7C|g8?
z;b+D&Gxl}JF!ptZGRzore|q14-ap>wIp_QRe3#GXoaa2}IVai9#_Wi|aRC4Tj#!u*
z+e7}&;pFFrtneos1#zhSr7vQ1w`F-<=nVWj)O8tpOlX0RUp+0e}gmVwM0P6a@e)
zZUCVF8~`N#%=>iF003aFc2*81P{Z`}G!{o_>mXq}h^-w&$bKIL2)A|+4l$mH#r-!=
zAP}%P5*8%1f&?;|j0K@Q)Y%JxR*=|^?}cnDp45&%q~mbI{}JSHIN0t%NUE)aaEL*u
z)OV;162Q(w1!#3h5DLF1ziWa7QvcZO+|nj(>)Q@ve}5k;O-)nx_Lz{u(C8#%caJi=
zu)MO)V(rh)EwO(}N?EEEcPqIJCY&TU)1%
zO-#4rNxQpy)cNHe(lDESu(-UovbxdH)i*Ug$6)Nzx4v#|f9oF{pPrfD+Wy)}8067L
z2}4KxbBMjoC4gVuA*!&xV_X2x>YyQew!>{XkiHnIhi8sMb%8Z5G#cQ99jChoe)cI+=tDU$U#pbw*z+#CC8P=$~EobKL
z=6ZU6$+Z$o5Lef3Y+t_pF>ZCMx;fa2`
zW7kY|HT}}U2o-W?N)FJfO;ph&q$qW*DMJMTgr=v3sUS>^;Hg$sG#j%tlyR}-ZLAO?
zN%GH~=ZvI
z%3ipaV_Ex(faW=BW~>Y$cTlJk##zBHsnU!qe2@Kd7ND;W@gMtlm8a0)YTv~*
zuUQ~+tkg92wzP86iLl@TH${)%UPs#?DefE99ukizBJlGmODZDaHT>EKPp-X%J<~ME
zu%EL<$WOkF(hx<5J2gG{Xx`MRkgAt(Ub0!P>fyy_+ZHd+8l(&iVEy>-fSB6mslu_b
zcjl>#yN@cXc>L}ex+338UtY^H5xm6xaQ1@_TkUzBlkQX9^tgJ&(+g6LPV?}OXYof3
zX-WZJFYvjxTf2(3FRR33Gqn?4gMPhV;7fKMGb67yNGBFd97_skgaju9mn9EG{UQWX
zN=Xv@^eQQn8jYDtMSP)y~~w$$XsMrgg@)x+Kd)0m`id<
zCn(vd&gN^;W7$)m#3d+K@QR@3*b@cYFR;WA!*sCdkIt6bp1h#w@|K6Ee~ff87(k~`
z?`8W-)VJ-!p8Qm5M4_D;ub{XPnTns_|5&_a$C)~W-&xNpFcR(FfN
z9jeTdW$_E7nXqiMpbO4Br^zw;wM^dqlVtl~a?Y>CKSmZNG{`nwP|vDy;@+~iB45cj
zFV8sPdcxic*;Rz`mD)9}D&_HL4sN>;y_bK3z`jgC@tjk7!@oV-qa|}I);=1ot~0M?
zBbC-;T{IKj}RiTaFBbVESTn0$!I
z)eukj5HEd?+g^|Z&Z5wI>Sz=+98hQVwe<8gbyZO)eH1Fd=cU8{5CVfdeQ!qme}c4@
SBJ}40fQ5;T@p~h;xPJj%?#LPd
literal 0
HcmV?d00001
diff --git a/src/assets/images/background.webp b/src/assets/images/background.webp
new file mode 100644
index 0000000000000000000000000000000000000000..3ad73c8b209ac4563e2ad6bb7cab860c87e7fa32
GIT binary patch
literal 156066
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