refactor: migrate demo applications to playground (#4116)
* chore: detail adjustment * refactor: Migrate demo applications to playground * perf: logic optimization * chore: update docs * chore: update docspull/48/MERGE
							parent
							
								
									654bf90c0d
								
							
						
					
					
						commit
						b464b87ac5
					
				|  | @ -1,5 +1,42 @@ | |||
| # Contributing Guide | ||||
| # Vben Admin Contributing Guide | ||||
| 
 | ||||
| 1. Make sure you put things in the right category! | ||||
| 2. Always add your items to the end of a list. To be fair, the order is first-come-first-serve. | ||||
| 3. If you think something belongs in the wrong category, or think there needs to be a new category, feel free to edit things too. | ||||
| Hi! We're really excited that you are interested in contributing to Vben Admin. Before submitting your contribution, please make sure to take a moment and read through the following guidelines: | ||||
| 
 | ||||
| - [Pull Request Guidelines](#pull-request-guidelines) | ||||
| 
 | ||||
| ## Contributor Code of Conduct | ||||
| 
 | ||||
| As contributors and maintainers of this project, we pledge to respect all people who contribute through reporting issues, posting feature requests, updating documentation, submitting pull requests or patches, and other activities. | ||||
| 
 | ||||
| We are committed to making participation in this project a harassment-free experience for everyone, regardless of the level of experience, gender, gender identity and expression, sexual orientation, disability, personal appearance, body size, race, age, or religion. | ||||
| 
 | ||||
| Examples of unacceptable behavior by participants include the use of sexual language or imagery, derogatory comments or personal attacks, trolling, public or private harassment, insults, or other unprofessional conduct. | ||||
| 
 | ||||
| Project maintainers have the right and responsibility to remove, edit, or reject comments, commits, code, wiki edits, issues, and other contributions that are not aligned to this Code of Conduct. Project maintainers who do not follow the Code of Conduct may be removed from the project team. | ||||
| 
 | ||||
| ## Pull Request Guidelines | ||||
| 
 | ||||
| - Checkout a topic branch from the relevant branch, e.g. main, and merge back against that branch. | ||||
| 
 | ||||
| - If adding a new feature: | ||||
| 
 | ||||
|   - Provide a convincing reason to add this feature. Ideally, you should open a suggestion issue first and have it approved before working on it. | ||||
| 
 | ||||
| - If fixing bug: | ||||
| 
 | ||||
|   - Provide a detailed description of the bug in the PR. Live demo preferred. | ||||
| 
 | ||||
| - It's OK to have multiple small commits as you work on the PR - GitHub can automatically squash them before merging. | ||||
| 
 | ||||
| ## Development Setup | ||||
| 
 | ||||
| You will need [pnpm](https://pnpm.io/) | ||||
| 
 | ||||
| After cloning the repo, run: | ||||
| 
 | ||||
| ```bash | ||||
| # install the dependencies of the project | ||||
| $ pnpm install | ||||
| # start the project | ||||
| $ pnpm run dev | ||||
| ``` | ||||
|  |  | |||
|  | @ -25,7 +25,7 @@ categories: | |||
|       - "perf" | ||||
|       - "chore" | ||||
|       - "dependencies" | ||||
|     collapse-after: 5 | ||||
|     # collapse-after: 12 | ||||
|   - title: 🚦 Tests | ||||
|     labels: | ||||
|       - "tests" | ||||
|  | @ -40,11 +40,10 @@ version-resolver: | |||
|   minor: | ||||
|     labels: | ||||
|       - "minor" | ||||
|       # - "feature" | ||||
|       - "feature" | ||||
|   patch: | ||||
|     labels: | ||||
|       - "patch" | ||||
|       - "feature" | ||||
|       - "bug" | ||||
|       - "maintenance" | ||||
|       - "docs" | ||||
|  |  | |||
|  | @ -6,7 +6,46 @@ on: | |||
|       - main | ||||
| 
 | ||||
| jobs: | ||||
|   deploy-push-ftp: | ||||
|   deploy-push-playground-ftp: | ||||
|     name: Deploy Push Ftp | ||||
|     if: github.actor != 'dependabot[bot]' && !contains(github.event.head_commit.message, '[skip ci]') | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - name: Checkout code | ||||
|         uses: actions/checkout@v4 | ||||
|         with: | ||||
|           fetch-depth: 0 | ||||
| 
 | ||||
|       - name: Sed Config Base | ||||
|         shell: bash | ||||
|         run: | | ||||
|           sed -i  "s#VITE_COMPRESS\s*=.*#VITE_COMPRESS = gzip#g" ./playground/.env.production | ||||
|           sed -i  "s#VITE_PWA\s*=.*#VITE_PWA = true#g" ./playground/.env.production | ||||
|           cat ./playground/.env.production | ||||
| 
 | ||||
|       - name: Setup Node | ||||
|         uses: ./.github/actions/setup-node | ||||
| 
 | ||||
|       - name: Build | ||||
|         run: pnpm run build | ||||
| 
 | ||||
|       - name: Sync Playground files | ||||
|         uses: SamKirkland/FTP-Deploy-Action@v4.3.5 | ||||
|         with: | ||||
|           server: ${{ secrets.PRO_FTP_HOST }} | ||||
|           username: ${{ secrets.WEB_PLAYGROUND_FTP_ACCOUNT }} | ||||
|           password: ${{ secrets.WEB_PLAYGROUND_FTP_PWSSWORD }} | ||||
|           local-dir: ./playground/dist/ | ||||
| 
 | ||||
|       - name: Sync Docs files | ||||
|         uses: SamKirkland/FTP-Deploy-Action@v4.3.5 | ||||
|         with: | ||||
|           server: ${{ secrets.PRO_FTP_HOST }} | ||||
|           username: ${{ secrets.WEBSITE_FTP_ACCOUNT }} | ||||
|           password: ${{ secrets.WEBSITE_FTP_PASSWORD }} | ||||
|           local-dir: ./docs/.vitepress/dist/ | ||||
| 
 | ||||
|   deploy-push-antd-ftp: | ||||
|     name: Deploy Push Ftp | ||||
|     if: github.actor != 'dependabot[bot]' && !contains(github.event.head_commit.message, '[skip ci]') | ||||
|     runs-on: ubuntu-latest | ||||
|  | @ -22,9 +61,65 @@ jobs: | |||
|           sed -i  "s#VITE_COMPRESS\s*=.*#VITE_COMPRESS = gzip#g" ./apps/web-antd/.env.production | ||||
|           sed -i  "s#VITE_PWA\s*=.*#VITE_PWA = true#g" ./apps/web-antd/.env.production | ||||
|           cat ./apps/web-antd/.env.production | ||||
| 
 | ||||
|       - name: Setup Node | ||||
|         uses: ./.github/actions/setup-node | ||||
| 
 | ||||
|       - name: Build | ||||
|         run: pnpm run build | ||||
| 
 | ||||
|       - name: Sync files | ||||
|         uses: SamKirkland/FTP-Deploy-Action@v4.3.5 | ||||
|         with: | ||||
|           server: ${{ secrets.PRO_FTP_HOST }} | ||||
|           username: ${{ secrets.WEB_ANTD_FTP_ACCOUNT }} | ||||
|           password: ${{ secrets.WEB_ANTD_FTP_PASSWORD }} | ||||
|           local-dir: ./apps/web-antd/dist/ | ||||
| 
 | ||||
|   deploy-push-ele-ftp: | ||||
|     name: Deploy Push Ftp | ||||
|     if: github.actor != 'dependabot[bot]' && !contains(github.event.head_commit.message, '[skip ci]') | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - name: Checkout code | ||||
|         uses: actions/checkout@v4 | ||||
|         with: | ||||
|           fetch-depth: 0 | ||||
| 
 | ||||
|       - name: Sed Config Base | ||||
|         shell: bash | ||||
|         run: | | ||||
|           sed -i  "s#VITE_COMPRESS\s*=.*#VITE_COMPRESS = gzip#g" ./apps/web-ele/.env.production | ||||
|           sed -i  "s#VITE_PWA\s*=.*#VITE_PWA = true#g" ./apps/web-ele/.env.production | ||||
|           cat ./apps/web-ele/.env.production | ||||
| 
 | ||||
|       - name: Setup Node | ||||
|         uses: ./.github/actions/setup-node | ||||
| 
 | ||||
|       - name: Build | ||||
|         run: pnpm run build | ||||
| 
 | ||||
|       - name: Sync files | ||||
|         uses: SamKirkland/FTP-Deploy-Action@v4.3.5 | ||||
|         with: | ||||
|           server: ${{ secrets.PRO_FTP_HOST }} | ||||
|           username: ${{ secrets.WEB_ELE_FTP_ACCOUNT }} | ||||
|           password: ${{ secrets.WEB_ELE_FTP_PASSWORD }} | ||||
|           local-dir: ./apps/web-ele/dist/ | ||||
| 
 | ||||
|   deploy-push-naive-ftp: | ||||
|     name: Deploy Push Ftp | ||||
|     if: github.actor != 'dependabot[bot]' && !contains(github.event.head_commit.message, '[skip ci]') | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - name: Checkout code | ||||
|         uses: actions/checkout@v4 | ||||
|         with: | ||||
|           fetch-depth: 0 | ||||
| 
 | ||||
|       - name: Sed Config Base | ||||
|         shell: bash | ||||
|         run: | | ||||
|           sed -i  "s#VITE_COMPRESS\s*=.*#VITE_COMPRESS = gzip#g" ./apps/web-naive/.env.production | ||||
|           sed -i  "s#VITE_PWA\s*=.*#VITE_PWA = true#g" ./apps/web-naive/.env.production | ||||
|           cat ./apps/web-naive/.env.production | ||||
|  | @ -35,34 +130,10 @@ jobs: | |||
|       - name: Build | ||||
|         run: pnpm run build | ||||
| 
 | ||||
|       - name: Sync Web Antd files | ||||
|         uses: SamKirkland/FTP-Deploy-Action@v4.3.5 | ||||
|         with: | ||||
|           server: ${{ secrets.PRO_FTP_HOST }} | ||||
|           username: ${{ secrets.WEB_ANTD_FTP_ACCOUNT }} | ||||
|           password: ${{ secrets.WEB_ANTD_FTP_PASSWORD }} | ||||
|           local-dir: ./apps/web-antd/dist/ | ||||
| 
 | ||||
|       - name: Sync Web Naive files | ||||
|       - name: Sync files | ||||
|         uses: SamKirkland/FTP-Deploy-Action@v4.3.5 | ||||
|         with: | ||||
|           server: ${{ secrets.PRO_FTP_HOST }} | ||||
|           username: ${{ secrets.WEB_NAIVE_FTP_ACCOUNT }} | ||||
|           password: ${{ secrets.WEB_NAIVE_FTP_PASSWORD }} | ||||
|           local-dir: ./apps/web-naive/dist/ | ||||
| 
 | ||||
|       - name: Sync Web Ele files | ||||
|         uses: SamKirkland/FTP-Deploy-Action@v4.3.5 | ||||
|         with: | ||||
|           server: ${{ secrets.PRO_FTP_HOST }} | ||||
|           username: ${{ secrets.WEB_ELE_FTP_ACCOUNT }} | ||||
|           password: ${{ secrets.WEB_ELE_FTP_PASSWORD }} | ||||
|           local-dir: ./apps/web-ele/dist/ | ||||
| 
 | ||||
|       - name: Sync Docs files | ||||
|         uses: SamKirkland/FTP-Deploy-Action@v4.3.5 | ||||
|         with: | ||||
|           server: ${{ secrets.PRO_FTP_HOST }} | ||||
|           username: ${{ secrets.WEBSITE_FTP_ACCOUNT }} | ||||
|           password: ${{ secrets.WEBSITE_FTP_PASSWORD }} | ||||
|           local-dir: ./docs/.vitepress/dist/ | ||||
|  |  | |||
|  | @ -10,10 +10,6 @@ | |||
|     "esbenp.prettier-vscode", | ||||
|     // 支持 dotenv 文件语法 | ||||
|     "mikestead.dotenv", | ||||
|     // 获取每个 CSS 属性的初始值。 | ||||
|     "dzhavat.css-initial-value", | ||||
|     // 使 VSCode 中的 TypeScript 错误更漂亮、更易于理解 | ||||
|     "yoavbls.pretty-ts-errors", | ||||
|     // 源代码的拼写检查器 | ||||
|     "streetsidesoftware.code-spell-checker", | ||||
|     // Tailwind CSS 的官方 VS Code 插件 | ||||
|  |  | |||
|  | @ -4,18 +4,27 @@ | |||
|   "configurations": [ | ||||
|     { | ||||
|       "type": "chrome", | ||||
|       "name": "vben admin antd dev", | ||||
|       "name": "vben admin playground dev", | ||||
|       "request": "launch", | ||||
|       "url": "http://localhost:5555", | ||||
|       "env": { "NODE_ENV": "development" }, | ||||
|       "sourceMaps": true, | ||||
|       "webRoot": "${workspaceFolder}/playground/src" | ||||
|     }, | ||||
|     { | ||||
|       "type": "chrome", | ||||
|       "name": "vben admin antd dev", | ||||
|       "request": "launch", | ||||
|       "url": "http://localhost:5666", | ||||
|       "env": { "NODE_ENV": "development" }, | ||||
|       "sourceMaps": true, | ||||
|       "webRoot": "${workspaceFolder}/apps/web-antd/src" | ||||
|     }, | ||||
|     { | ||||
|       "type": "chrome", | ||||
|       "name": "vben admin ele dev", | ||||
|       "request": "launch", | ||||
|       "url": "http://localhost:5666", | ||||
|       "url": "http://localhost:5777", | ||||
|       "env": { "NODE_ENV": "development" }, | ||||
|       "sourceMaps": true, | ||||
|       "webRoot": "${workspaceFolder}/apps/web-ele/src" | ||||
|  | @ -24,7 +33,7 @@ | |||
|       "type": "chrome", | ||||
|       "name": "vben admin naive dev", | ||||
|       "request": "launch", | ||||
|       "url": "http://localhost:5777", | ||||
|       "url": "http://localhost:5888", | ||||
|       "env": { "NODE_ENV": "development" }, | ||||
|       "sourceMaps": true, | ||||
|       "webRoot": "${workspaceFolder}/apps/web-naive/src" | ||||
|  |  | |||
|  | @ -167,6 +167,7 @@ | |||
| 
 | ||||
|   "i18n-ally.localesPaths": [ | ||||
|     "packages/locales/src/langs", | ||||
|     "playground/src/langs", | ||||
|     "apps/*/src/locales/langs" | ||||
|   ], | ||||
|   "i18n-ally.enabledParsers": ["json", "ts", "js", "yaml"], | ||||
|  |  | |||
|  | @ -21,7 +21,7 @@ RUN echo "Builder Success 🎉" | |||
| FROM nginx:stable-alpine as production | ||||
| 
 | ||||
| RUN echo "types { application/javascript js mjs; }" > /etc/nginx/conf.d/mjs.conf | ||||
| COPY --from=builder /app/apps/web-antd/dist /usr/share/nginx/html | ||||
| COPY --from=builder /app/playground/dist /usr/share/nginx/html | ||||
| 
 | ||||
| COPY ./nginx.conf /etc/nginx/nginx.conf | ||||
| 
 | ||||
|  |  | |||
|  | @ -134,7 +134,8 @@ pnpm build | |||
| ## 貢献者 | ||||
| 
 | ||||
| <a href="https://github.com/vbenjs/vue-vben-admin/graphs/contributors"> | ||||
|   <img src="https://contrib.rocks/image?repo=vbenjs/vue-vben-admin" /> | ||||
|  <img alt="Contributors" | ||||
|         src="https://opencollective.com/vbenjs/contributors.svg?button=false" /> | ||||
| </a> | ||||
| 
 | ||||
| ## Discord | ||||
|  |  | |||
|  | @ -133,7 +133,8 @@ If you think this project is helpful to you, you can help the author buy a cup o | |||
| ## Contributor | ||||
| 
 | ||||
| <a href="https://github.com/vbenjs/vue-vben-admin/graphs/contributors"> | ||||
|   <img src="https://contrib.rocks/image?repo=vbenjs/vue-vben-admin" /> | ||||
|   <img alt="Contributors" | ||||
|         src="https://opencollective.com/vbenjs/contributors.svg?button=false" /> | ||||
| </a> | ||||
| 
 | ||||
| ## Discord | ||||
|  |  | |||
|  | @ -126,6 +126,13 @@ pnpm build | |||
| 
 | ||||
| <a style="display: block;width: 100px;height: 50px;line-height: 50px; color: #fff;text-align: center; background: #408aed;border-radius: 4px;" href="https://www.paypal.com/paypalme/cvvben">Paypal Me</a> | ||||
| 
 | ||||
| ## Contributor | ||||
| 
 | ||||
| <a href="https://github.com/vbenjs/vue-vben-admin/graphs/contributors"> | ||||
|   <img alt="Contributors" | ||||
|         src="https://opencollective.com/vbenjs/contributors.svg?button=false" /> | ||||
| </a> | ||||
| 
 | ||||
| ## Discord | ||||
| 
 | ||||
| - [Github Discussions](https://github.com/anncwb/vue-vben-admin/discussions) | ||||
|  |  | |||
|  | @ -1,5 +1,5 @@ | |||
| # 应用标题 | ||||
| VITE_APP_TITLE=Vben Admin | ||||
| VITE_APP_TITLE=Vben Admin Antd | ||||
| 
 | ||||
| # 应用命名空间,用于缓存、store等功能的前缀,确保隔离 | ||||
| VITE_APP_NAMESPACE=vben-web-antd | ||||
|  |  | |||
|  | @ -1,5 +1,5 @@ | |||
| # 端口号 | ||||
| VITE_PORT=5555 | ||||
| VITE_PORT=5666 | ||||
| 
 | ||||
| VITE_BASE=/ | ||||
| 
 | ||||
|  |  | |||
|  | @ -21,7 +21,7 @@ | |||
|         (function () { | ||||
|           var hm = document.createElement('script'); | ||||
|           hm.src = | ||||
|             'https://hm.baidu.com/hm.js?d20a01273820422b6aa2ee41b6c9414d'; | ||||
|             'https://hm.baidu.com/hm.js?b38e689f40558f20a9a686d7f6f33edf'; | ||||
|           var s = document.getElementsByTagName('script')[0]; | ||||
|           s.parentNode.insertBefore(hm, s); | ||||
|         })(); | ||||
|  |  | |||
|  | @ -1,2 +1 @@ | |||
| export * from './core'; | ||||
| export * from './demos'; | ||||
|  |  | |||
|  | @ -59,7 +59,7 @@ function createRequestClient(baseURL: string) { | |||
|     if (status >= 200 && status < 400 && code === 0) { | ||||
|       return data; | ||||
|     } | ||||
|     throw new Error(msg); | ||||
|     throw new Error(`Error ${status}: ${msg}`); | ||||
|   }); | ||||
|   return client; | ||||
| } | ||||
|  |  | |||
|  | @ -58,7 +58,11 @@ async function loadDayjsLocale(lang: SupportedLanguagesType) { | |||
|       locale = await import('dayjs/locale/en'); | ||||
|     } | ||||
|   } | ||||
|   if (locale) { | ||||
|     dayjs.locale(locale); | ||||
|   } else { | ||||
|     console.error(`Failed to load dayjs locale for ${lang}`); | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  |  | |||
|  | @ -2,66 +2,7 @@ | |||
|   "page": { | ||||
|     "demos": { | ||||
|       "title": "Demos", | ||||
|       "access": { | ||||
|         "frontendPermissions": "Frontend Permissions", | ||||
|         "backendPermissions": "Backend Permissions", | ||||
|         "pageAccess": "Page Access", | ||||
|         "buttonControl": "Button Control", | ||||
|         "menuVisible403": "Menu Visible(403)", | ||||
|         "superVisible": "Visible to Super", | ||||
|         "adminVisible": "Visible to Admin", | ||||
|         "userVisible": "Visible to User" | ||||
|       }, | ||||
|       "nested": { | ||||
|         "title": "Nested Menu", | ||||
|         "menu1": "Menu 1", | ||||
|         "menu2": "Menu 2", | ||||
|         "menu2_1": "Menu 2-1", | ||||
|         "menu3": "Menu 3", | ||||
|         "menu3_1": "Menu 3-1", | ||||
|         "menu3_2": "Menu 3-2", | ||||
|         "menu3_2_1": "Menu 3-2-1" | ||||
|       }, | ||||
|       "outside": { | ||||
|         "title": "External Pages", | ||||
|         "embedded": "Embedded", | ||||
|         "externalLink": "External Link" | ||||
|       }, | ||||
|       "badge": { | ||||
|         "title": "Menu Badge", | ||||
|         "dot": "Dot Badge", | ||||
|         "text": "Text Badge", | ||||
|         "color": "Badge Color" | ||||
|       }, | ||||
|       "activeIcon": { | ||||
|         "title": "Active Menu Icon", | ||||
|         "children": "Children Active Icon" | ||||
|       }, | ||||
|       "fallback": { | ||||
|         "title": "Fallback Page" | ||||
|       }, | ||||
|       "features": { | ||||
|         "title": "Features", | ||||
|         "hideChildrenInMenu": "Hide Menu Children", | ||||
|         "loginExpired": "Login Expired", | ||||
|         "icons": "Icons", | ||||
|         "watermark": "Watermark", | ||||
|         "tabs": "Tabs", | ||||
|         "tabDetail": "Tab Detail Page" | ||||
|       }, | ||||
|       "breadcrumb": { | ||||
|         "navigation": "Breadcrumb Navigation", | ||||
|         "lateral": "Lateral Mode", | ||||
|         "lateralDetail": "Lateral Mode Detail", | ||||
|         "level": "Level Mode", | ||||
|         "levelDetail": "Level Mode Detail" | ||||
|       } | ||||
|     }, | ||||
|     "examples": { | ||||
|       "title": "Examples", | ||||
|       "ellipsis": { | ||||
|         "title": "EllipsisText" | ||||
|       } | ||||
|       "antd": "Ant Design Vue" | ||||
|     } | ||||
|   } | ||||
| } | ||||
|  |  | |||
|  | @ -2,66 +2,7 @@ | |||
|   "page": { | ||||
|     "demos": { | ||||
|       "title": "演示", | ||||
|       "access": { | ||||
|         "frontendPermissions": "前端权限", | ||||
|         "backendPermissions": "后端权限", | ||||
|         "pageAccess": "页面访问", | ||||
|         "buttonControl": "按钮控制", | ||||
|         "menuVisible403": "菜单可见(403)", | ||||
|         "superVisible": "Super 可见", | ||||
|         "adminVisible": "Admin 可见", | ||||
|         "userVisible": "User 可见" | ||||
|       }, | ||||
|       "nested": { | ||||
|         "title": "嵌套菜单", | ||||
|         "menu1": "菜单 1", | ||||
|         "menu2": "菜单 2", | ||||
|         "menu2_1": "菜单 2-1", | ||||
|         "menu3": "菜单 3", | ||||
|         "menu3_1": "菜单 3-1", | ||||
|         "menu3_2": "菜单 3-2", | ||||
|         "menu3_2_1": "菜单 3-2-1" | ||||
|       }, | ||||
|       "outside": { | ||||
|         "title": "外部页面", | ||||
|         "embedded": "内嵌", | ||||
|         "externalLink": "外链" | ||||
|       }, | ||||
|       "badge": { | ||||
|         "title": "菜单徽标", | ||||
|         "dot": "点徽标", | ||||
|         "text": "文本徽标", | ||||
|         "color": "徽标颜色" | ||||
|       }, | ||||
|       "activeIcon": { | ||||
|         "title": "菜单激活图标", | ||||
|         "children": "子级激活图标" | ||||
|       }, | ||||
|       "fallback": { | ||||
|         "title": "缺省页" | ||||
|       }, | ||||
|       "features": { | ||||
|         "title": "功能", | ||||
|         "hideChildrenInMenu": "隐藏子菜单", | ||||
|         "loginExpired": "登录过期", | ||||
|         "icons": "图标", | ||||
|         "watermark": "水印", | ||||
|         "tabs": "标签页", | ||||
|         "tabDetail": "标签详情页" | ||||
|       }, | ||||
|       "breadcrumb": { | ||||
|         "navigation": "面包屑导航", | ||||
|         "lateral": "平级模式", | ||||
|         "level": "层级模式", | ||||
|         "levelDetail": "层级模式详情", | ||||
|         "lateralDetail": "平级模式详情" | ||||
|       } | ||||
|     }, | ||||
|     "examples": { | ||||
|       "title": "示例", | ||||
|       "ellipsis": { | ||||
|         "title": "文本省略" | ||||
|       } | ||||
|       "antd": "Ant Design Vue" | ||||
|     } | ||||
|   } | ||||
| } | ||||
|  |  | |||
|  | @ -1,6 +1,6 @@ | |||
| import type { RouteRecordRaw } from 'vue-router'; | ||||
| 
 | ||||
| import { BasicLayout, IFrameView } from '#/layouts'; | ||||
| import { BasicLayout } from '#/layouts'; | ||||
| import { $t } from '#/locales'; | ||||
| 
 | ||||
| const routes: RouteRecordRaw[] = [ | ||||
|  | @ -15,477 +15,13 @@ const routes: RouteRecordRaw[] = [ | |||
|     name: 'Demos', | ||||
|     path: '/demos', | ||||
|     children: [ | ||||
|       // 权限控制
 | ||||
|       { | ||||
|         meta: { | ||||
|           icon: 'mdi:shield-key-outline', | ||||
|           title: $t('page.demos.access.frontendPermissions'), | ||||
|           title: $t('page.demos.antd'), | ||||
|         }, | ||||
|         name: 'AccessDemos', | ||||
|         path: '/demos/access', | ||||
|         children: [ | ||||
|           { | ||||
|             name: 'AccessPageControlDemo', | ||||
|             path: '/demos/access/page-control', | ||||
|             component: () => import('#/views/demos/access/index.vue'), | ||||
|             meta: { | ||||
|               icon: 'mdi:page-previous-outline', | ||||
|               title: $t('page.demos.access.pageAccess'), | ||||
|             }, | ||||
|           }, | ||||
|           { | ||||
|             name: 'AccessButtonControlDemo', | ||||
|             path: '/demos/access/button-control', | ||||
|             component: () => import('#/views/demos/access/button-control.vue'), | ||||
|             meta: { | ||||
|               icon: 'mdi:button-cursor', | ||||
|               title: $t('page.demos.access.buttonControl'), | ||||
|             }, | ||||
|           }, | ||||
|           { | ||||
|             name: 'AccessMenuVisible403Demo', | ||||
|             path: '/demos/access/menu-visible-403', | ||||
|             component: () => | ||||
|               import('#/views/demos/access/menu-visible-403.vue'), | ||||
|             meta: { | ||||
|               authority: ['no-body'], | ||||
|               icon: 'mdi:button-cursor', | ||||
|               menuVisibleWithForbidden: true, | ||||
|               title: $t('page.demos.access.menuVisible403'), | ||||
|             }, | ||||
|           }, | ||||
|           { | ||||
|             name: 'AccessSuperVisibleDemo', | ||||
|             path: '/demos/access/super-visible', | ||||
|             component: () => import('#/views/demos/access/super-visible.vue'), | ||||
|             meta: { | ||||
|               authority: ['super'], | ||||
|               icon: 'mdi:button-cursor', | ||||
|               title: $t('page.demos.access.superVisible'), | ||||
|             }, | ||||
|           }, | ||||
|           { | ||||
|             name: 'AccessAdminVisibleDemo', | ||||
|             path: '/demos/access/admin-visible', | ||||
|             component: () => import('#/views/demos/access/admin-visible.vue'), | ||||
|             meta: { | ||||
|               authority: ['admin'], | ||||
|               icon: 'mdi:button-cursor', | ||||
|               title: $t('page.demos.access.adminVisible'), | ||||
|             }, | ||||
|           }, | ||||
|           { | ||||
|             name: 'AccessUserVisibleDemo', | ||||
|             path: '/demos/access/user-visible', | ||||
|             component: () => import('#/views/demos/access/user-visible.vue'), | ||||
|             meta: { | ||||
|               authority: ['user'], | ||||
|               icon: 'mdi:button-cursor', | ||||
|               title: $t('page.demos.access.userVisible'), | ||||
|             }, | ||||
|           }, | ||||
|         ], | ||||
|       }, | ||||
|       // 功能
 | ||||
|       { | ||||
|         meta: { | ||||
|           icon: 'mdi:feature-highlight', | ||||
|           title: $t('page.demos.features.title'), | ||||
|         }, | ||||
|         name: 'FeaturesDemos', | ||||
|         path: '/demos/features', | ||||
|         children: [ | ||||
|           { | ||||
|             name: 'LoginExpiredDemo', | ||||
|             path: '/demos/features/login-expired', | ||||
|             component: () => | ||||
|               import('#/views/demos/features/login-expired/index.vue'), | ||||
|             meta: { | ||||
|               icon: 'mdi:encryption-expiration', | ||||
|               title: $t('page.demos.features.loginExpired'), | ||||
|             }, | ||||
|           }, | ||||
|           { | ||||
|             name: 'IconsDemo', | ||||
|             path: '/demos/features/icons', | ||||
|             component: () => import('#/views/demos/features/icons/index.vue'), | ||||
|             meta: { | ||||
|               title: $t('page.demos.features.icons'), | ||||
|             }, | ||||
|           }, | ||||
|           { | ||||
|             name: 'WatermarkDemo', | ||||
|             path: '/demos/features/watermark', | ||||
|             component: () => | ||||
|               import('#/views/demos/features/watermark/index.vue'), | ||||
|             meta: { | ||||
|               title: $t('page.demos.features.watermark'), | ||||
|             }, | ||||
|           }, | ||||
|           { | ||||
|             name: 'FeatureTabsDemo', | ||||
|             path: '/demos/features/tabs', | ||||
|             component: () => import('#/views/demos/features/tabs/index.vue'), | ||||
|             meta: { | ||||
|               icon: 'lucide:app-window', | ||||
|               title: $t('page.demos.features.tabs'), | ||||
|             }, | ||||
|           }, | ||||
|           { | ||||
|             name: 'FeatureTabDetailDemo', | ||||
|             path: '/demos/features/tabs/detail/:id', | ||||
|             component: () => | ||||
|               import('#/views/demos/features/tabs/tab-detail.vue'), | ||||
|             meta: { | ||||
|               activePath: '/demos/features/tabs', | ||||
|               hideInMenu: true, | ||||
|               maxNumOfOpenTab: 3, | ||||
|               title: $t('page.demos.features.tabDetail'), | ||||
|             }, | ||||
|           }, | ||||
|           { | ||||
|             name: 'HideChildrenInMenuParentDemo', | ||||
|             path: '/demos/features/hide-menu-children', | ||||
|             component: () => | ||||
|               import('#/views/demos/features/hide-menu-children/parent.vue'), | ||||
|             meta: { | ||||
|               hideChildrenInMenu: true, | ||||
|               icon: 'ic:round-menu', | ||||
|               title: $t('page.demos.features.hideChildrenInMenu'), | ||||
|             }, | ||||
|             children: [ | ||||
|               { | ||||
|                 name: 'HideChildrenInMenuChildrenDemo', | ||||
|                 path: '/demos/features/hide-menu-children/children', | ||||
|                 component: () => | ||||
|                   import( | ||||
|                     '#/views/demos/features/hide-menu-children/children.vue' | ||||
|                   ), | ||||
|                 meta: { title: 'HideChildrenInMenuChildrenDemo' }, | ||||
|               }, | ||||
|             ], | ||||
|           }, | ||||
|         ], | ||||
|       }, | ||||
|       // 面包屑导航
 | ||||
|       { | ||||
|         name: 'BreadcrumbDemos', | ||||
|         path: '/demos/breadcrumb', | ||||
|         meta: { | ||||
|           icon: 'lucide:navigation', | ||||
|           title: $t('page.demos.breadcrumb.navigation'), | ||||
|         }, | ||||
|         children: [ | ||||
|           { | ||||
|             name: 'BreadcrumbLateralDemo', | ||||
|             path: '/demos/breadcrumb/lateral', | ||||
|             component: () => import('#/views/demos/breadcrumb/lateral.vue'), | ||||
|             meta: { | ||||
|               icon: 'lucide:navigation', | ||||
|               title: $t('page.demos.breadcrumb.lateral'), | ||||
|             }, | ||||
|           }, | ||||
|           { | ||||
|             name: 'BreadcrumbLateralDetailDemo', | ||||
|             path: '/demos/breadcrumb/lateral-detail', | ||||
|             component: () => | ||||
|               import('#/views/demos/breadcrumb/lateral-detail.vue'), | ||||
|             meta: { | ||||
|               activePath: '/demos/breadcrumb/lateral', | ||||
|               hideInMenu: true, | ||||
|               title: $t('page.demos.breadcrumb.lateralDetail'), | ||||
|             }, | ||||
|           }, | ||||
|           { | ||||
|             name: 'BreadcrumbLevelDemo', | ||||
|             path: '/demos/breadcrumb/level', | ||||
|             meta: { | ||||
|               icon: 'lucide:navigation', | ||||
|               title: $t('page.demos.breadcrumb.level'), | ||||
|             }, | ||||
|             children: [ | ||||
|               { | ||||
|                 name: 'BreadcrumbLevelDetailDemo', | ||||
|                 path: '/demos/breadcrumb/level/detail', | ||||
|                 component: () => | ||||
|                   import('#/views/demos/breadcrumb/level-detail.vue'), | ||||
|                 meta: { | ||||
|                   title: $t('page.demos.breadcrumb.levelDetail'), | ||||
|                 }, | ||||
|               }, | ||||
|             ], | ||||
|           }, | ||||
|         ], | ||||
|       }, | ||||
|       // 缺省页
 | ||||
|       { | ||||
|         meta: { | ||||
|           icon: 'mdi:lightbulb-error-outline', | ||||
|           title: $t('page.demos.fallback.title'), | ||||
|         }, | ||||
|         name: 'FallbackDemos', | ||||
|         path: '/demos/fallback', | ||||
|         children: [ | ||||
|           { | ||||
|             name: 'Fallback403Demo', | ||||
|             path: '/demos/fallback/403', | ||||
|             component: () => import('#/views/_core/fallback/forbidden.vue'), | ||||
|             meta: { | ||||
|               icon: 'mdi:do-not-disturb-alt', | ||||
|               title: '403', | ||||
|             }, | ||||
|           }, | ||||
|           { | ||||
|             name: 'Fallback404Demo', | ||||
|             path: '/demos/fallback/404', | ||||
|             component: () => import('#/views/_core/fallback/not-found.vue'), | ||||
|             meta: { | ||||
|               icon: 'mdi:table-off', | ||||
|               title: '404', | ||||
|             }, | ||||
|           }, | ||||
|           { | ||||
|             name: 'Fallback500Demo', | ||||
|             path: '/demos/fallback/500', | ||||
|             component: () => | ||||
|               import('#/views/_core/fallback/internal-error.vue'), | ||||
|             meta: { | ||||
|               icon: 'mdi:server-network-off', | ||||
|               title: '500', | ||||
|             }, | ||||
|           }, | ||||
|           { | ||||
|             name: 'FallbackOfflineDemo', | ||||
|             path: '/demos/fallback/offline', | ||||
|             component: () => import('#/views/_core/fallback/offline.vue'), | ||||
|             meta: { | ||||
|               icon: 'mdi:offline', | ||||
|               title: $t('fallback.offline'), | ||||
|             }, | ||||
|           }, | ||||
|         ], | ||||
|       }, | ||||
|       // 菜单徽标
 | ||||
|       { | ||||
|         meta: { | ||||
|           badgeType: 'dot', | ||||
|           badgeVariants: 'destructive', | ||||
|           icon: 'lucide:circle-dot', | ||||
|           title: $t('page.demos.badge.title'), | ||||
|         }, | ||||
|         name: 'BadgeDemos', | ||||
|         path: '/demos/badge', | ||||
|         children: [ | ||||
|           { | ||||
|             name: 'BadgeDotDemo', | ||||
|             component: () => import('#/views/demos/badge/index.vue'), | ||||
|             path: '/demos/badge/dot', | ||||
|             meta: { | ||||
|               badgeType: 'dot', | ||||
|               icon: 'lucide:square-dot', | ||||
|               title: $t('page.demos.badge.dot'), | ||||
|             }, | ||||
|           }, | ||||
|           { | ||||
|             name: 'BadgeTextDemo', | ||||
|             component: () => import('#/views/demos/badge/index.vue'), | ||||
|             path: '/demos/badge/text', | ||||
|             meta: { | ||||
|               badge: '10', | ||||
|               icon: 'lucide:square-dot', | ||||
|               title: $t('page.demos.badge.text'), | ||||
|             }, | ||||
|           }, | ||||
|           { | ||||
|             name: 'BadgeColorDemo', | ||||
|             component: () => import('#/views/demos/badge/index.vue'), | ||||
|             path: '/demos/badge/color', | ||||
|             meta: { | ||||
|               badge: 'Hot', | ||||
|               badgeVariants: 'destructive', | ||||
|               icon: 'lucide:square-dot', | ||||
|               title: $t('page.demos.badge.color'), | ||||
|             }, | ||||
|           }, | ||||
|         ], | ||||
|       }, | ||||
|       // 菜单激活图标
 | ||||
|       { | ||||
|         meta: { | ||||
|           activeIcon: 'fluent-emoji:radioactive', | ||||
|           icon: 'bi:radioactive', | ||||
|           title: $t('page.demos.activeIcon.title'), | ||||
|         }, | ||||
|         name: 'ActiveIconDemos', | ||||
|         path: '/demos/active-icon', | ||||
|         children: [ | ||||
|           { | ||||
|             name: 'ActiveIconDemo', | ||||
|             component: () => import('#/views/demos/active-icon/index.vue'), | ||||
|             path: '/demos/active-icon/children', | ||||
|             meta: { | ||||
|               activeIcon: 'fluent-emoji:radioactive', | ||||
|               icon: 'bi:radioactive', | ||||
|               title: $t('page.demos.activeIcon.children'), | ||||
|             }, | ||||
|           }, | ||||
|         ], | ||||
|       }, | ||||
|       // 外部链接
 | ||||
|       { | ||||
|         meta: { | ||||
|           icon: 'ic:round-settings-input-composite', | ||||
|           title: $t('page.demos.outside.title'), | ||||
|         }, | ||||
|         name: 'OutsideDemos', | ||||
|         path: '/demos/outside', | ||||
|         children: [ | ||||
|           { | ||||
|             name: 'IframeDemos', | ||||
|             path: '/demos/outside/iframe', | ||||
|             meta: { | ||||
|               icon: 'mdi:newspaper-variant-outline', | ||||
|               title: $t('page.demos.outside.embedded'), | ||||
|             }, | ||||
|             children: [ | ||||
|               { | ||||
|                 name: 'VueDocumentDemo', | ||||
|                 path: '/demos/outside/iframe/vue-document', | ||||
|                 component: IFrameView, | ||||
|                 meta: { | ||||
|                   icon: 'logos:vue', | ||||
|                   iframeSrc: 'https://cn.vuejs.org/', | ||||
|                   keepAlive: true, | ||||
|                   title: 'Vue', | ||||
|                 }, | ||||
|               }, | ||||
|               { | ||||
|                 name: 'TailwindcssDemo', | ||||
|                 path: '/demos/outside/iframe/tailwindcss', | ||||
|                 component: IFrameView, | ||||
|                 meta: { | ||||
|                   icon: 'devicon:tailwindcss', | ||||
|                   iframeSrc: 'https://tailwindcss.com/', | ||||
|                   // keepAlive: true,
 | ||||
|                   title: 'Tailwindcss', | ||||
|                 }, | ||||
|               }, | ||||
|             ], | ||||
|           }, | ||||
|           { | ||||
|             name: 'ExternalLinkDemos', | ||||
|             path: '/demos/outside/external-link', | ||||
|             meta: { | ||||
|               icon: 'mdi:newspaper-variant-multiple-outline', | ||||
|               title: $t('page.demos.outside.externalLink'), | ||||
|             }, | ||||
|             children: [ | ||||
|               { | ||||
|                 name: 'ViteDemo', | ||||
|                 path: '/demos/outside/external-link/vite', | ||||
|                 component: IFrameView, | ||||
|                 meta: { | ||||
|                   icon: 'logos:vitejs', | ||||
|                   link: 'https://vitejs.dev/', | ||||
|                   title: 'Vite', | ||||
|                 }, | ||||
|               }, | ||||
|               { | ||||
|                 name: 'VueUseDemo', | ||||
|                 path: '/demos/outside/external-link/vue-use', | ||||
|                 component: IFrameView, | ||||
|                 meta: { | ||||
|                   icon: 'logos:vueuse', | ||||
|                   link: 'https://vueuse.org', | ||||
|                   title: 'VueUse', | ||||
|                 }, | ||||
|               }, | ||||
|             ], | ||||
|           }, | ||||
|         ], | ||||
|       }, | ||||
|       // 嵌套菜单
 | ||||
|       { | ||||
|         meta: { | ||||
|           icon: 'ic:round-menu', | ||||
|           title: $t('page.demos.nested.title'), | ||||
|         }, | ||||
|         name: 'NestedDemos', | ||||
|         path: '/demos/nested', | ||||
|         children: [ | ||||
|           { | ||||
|             name: 'Menu1Demo', | ||||
|             path: '/demos/nested/menu1', | ||||
|             component: () => import('#/views/demos/nested/menu-1.vue'), | ||||
|             meta: { | ||||
|               icon: 'ic:round-menu', | ||||
|               keepAlive: true, | ||||
|               title: $t('page.demos.nested.menu1'), | ||||
|             }, | ||||
|           }, | ||||
|           { | ||||
|             name: 'Menu2Demo', | ||||
|             path: '/demos/nested/menu2', | ||||
|             meta: { | ||||
|               icon: 'ic:round-menu', | ||||
|               keepAlive: true, | ||||
|               title: $t('page.demos.nested.menu2'), | ||||
|             }, | ||||
|             children: [ | ||||
|               { | ||||
|                 name: 'Menu21Demo', | ||||
|                 path: '/demos/nested/menu2/menu2-1', | ||||
|                 component: () => import('#/views/demos/nested/menu-2-1.vue'), | ||||
|                 meta: { | ||||
|                   icon: 'ic:round-menu', | ||||
|                   keepAlive: true, | ||||
|                   title: $t('page.demos.nested.menu2_1'), | ||||
|                 }, | ||||
|               }, | ||||
|             ], | ||||
|           }, | ||||
|           { | ||||
|             name: 'Menu3Demo', | ||||
|             path: '/demos/nested/menu3', | ||||
|             meta: { | ||||
|               icon: 'ic:round-menu', | ||||
|               title: $t('page.demos.nested.menu3'), | ||||
|             }, | ||||
|             children: [ | ||||
|               { | ||||
|                 name: 'Menu31Demo', | ||||
|                 path: 'menu3-1', | ||||
|                 component: () => import('#/views/demos/nested/menu-3-1.vue'), | ||||
|                 meta: { | ||||
|                   icon: 'ic:round-menu', | ||||
|                   keepAlive: true, | ||||
|                   title: $t('page.demos.nested.menu3_1'), | ||||
|                 }, | ||||
|               }, | ||||
|               { | ||||
|                 name: 'Menu32Demo', | ||||
|                 path: 'menu3-2', | ||||
|                 meta: { | ||||
|                   icon: 'ic:round-menu', | ||||
|                   title: $t('page.demos.nested.menu3_2'), | ||||
|                 }, | ||||
|                 children: [ | ||||
|                   { | ||||
|                     name: 'Menu321Demo', | ||||
|                     path: '/demos/nested/menu3/menu3-2/menu3-2-1', | ||||
|                     component: () => | ||||
|                       import('#/views/demos/nested/menu-3-2-1.vue'), | ||||
|                     meta: { | ||||
|                       icon: 'ic:round-menu', | ||||
|                       keepAlive: true, | ||||
|                       title: $t('page.demos.nested.menu3_2_1'), | ||||
|                     }, | ||||
|                   }, | ||||
|                 ], | ||||
|               }, | ||||
|             ], | ||||
|           }, | ||||
|         ], | ||||
|         name: 'AntDesignDemos', | ||||
|         path: '/demos/ant-design', | ||||
|         component: () => import('#/views/demos/antd/index.vue'), | ||||
|       }, | ||||
|     ], | ||||
|   }, | ||||
|  |  | |||
|  | @ -38,8 +38,7 @@ const routes: RouteRecordRaw[] = [ | |||
|         component: IFrameView, | ||||
|         meta: { | ||||
|           icon: 'lucide:book-open-text', | ||||
|           iframeSrc: VBEN_DOC_URL, | ||||
|           keepAlive: true, | ||||
|           link: VBEN_DOC_URL, | ||||
|           title: $t('page.vben.document'), | ||||
|         }, | ||||
|       }, | ||||
|  |  | |||
|  | @ -0,0 +1,86 @@ | |||
| <script lang="ts" setup> | ||||
| import { Button, Card, message, notification, Space } from 'ant-design-vue'; | ||||
| 
 | ||||
| type NotificationType = 'error' | 'info' | 'success' | 'warning'; | ||||
| 
 | ||||
| function info() { | ||||
|   message.info('How many roads must a man walk down'); | ||||
| } | ||||
| 
 | ||||
| function error() { | ||||
|   message.error({ | ||||
|     content: 'Once upon a time you dressed so fine', | ||||
|     duration: 2500, | ||||
|   }); | ||||
| } | ||||
| 
 | ||||
| function warning() { | ||||
|   message.warning('How many roads must a man walk down'); | ||||
| } | ||||
| function success() { | ||||
|   message.success('Cause you walked hand in hand With another man in my place'); | ||||
| } | ||||
| 
 | ||||
| function notify(type: NotificationType) { | ||||
|   notification[type]({ | ||||
|     duration: 2500, | ||||
|     message: '说点啥呢', | ||||
|     type, | ||||
|   }); | ||||
| } | ||||
| </script> | ||||
| 
 | ||||
| <template> | ||||
|   <div class="p-5"> | ||||
|     <div class="card-box p-5"> | ||||
|       <h1 class="text-xl font-semibold">Ant Design Vue组件使用演示</h1> | ||||
|       <div class="text-foreground/80 mt-2">支持多语言,主题功能集成切换等</div> | ||||
|     </div> | ||||
| 
 | ||||
|     <div class="card-box mt-5 p-5"> | ||||
|       <div class="mb-3"> | ||||
|         <span class="text-lg font-semibold">按钮</span> | ||||
|       </div> | ||||
|       <div> | ||||
|         <Space> | ||||
|           <Button>Default</Button> | ||||
|           <Button type="primary"> Primary </Button> | ||||
|           <Button> Info </Button> | ||||
|           <Button danger> Error </Button> | ||||
|         </Space> | ||||
|       </div> | ||||
|     </div> | ||||
| 
 | ||||
|     <div class="card-box mt-5 p-5"> | ||||
|       <div class="mb-3"> | ||||
|         <span class="text-lg font-semibold">卡片</span> | ||||
|       </div> | ||||
|       <div> | ||||
|         <Card title="卡片"> 卡片内容 </Card> | ||||
|       </div> | ||||
|     </div> | ||||
|     <div class="card-box mt-5 p-5"> | ||||
|       <div class="mb-3"> | ||||
|         <span class="text-lg font-semibold">信息 Message </span> | ||||
|       </div> | ||||
|       <div class="flex gap-3"> | ||||
|         <Button @click="info"> 信息 </Button> | ||||
|         <Button danger @click="error"> 错误 </Button> | ||||
|         <Button @click="warning"> 警告 </Button> | ||||
|         <Button @click="success"> 成功 </Button> | ||||
|       </div> | ||||
|     </div> | ||||
| 
 | ||||
|     <div class="card-box mt-5 p-5"> | ||||
|       <div class="mb-3"> | ||||
|         <span class="text-lg font-semibold">通知 Notification </span> | ||||
|       </div> | ||||
|       <div class="flex gap-3"> | ||||
|         <Button @click="notify('info')"> 信息 </Button> | ||||
|         <Button danger @click="notify('error')"> 错误 </Button> | ||||
|         <Button @click="notify('warning')"> 警告 </Button> | ||||
|         <Button @click="notify('success')"> 成功 </Button> | ||||
|       </div> | ||||
|     </div> | ||||
|   </div> | ||||
| </template> | ||||
|  | @ -1,5 +1,5 @@ | |||
| # 端口号 | ||||
| VITE_PORT=5666 | ||||
| VITE_PORT=5777 | ||||
| 
 | ||||
| VITE_BASE=/ | ||||
| 
 | ||||
|  |  | |||
|  | @ -59,7 +59,7 @@ function createRequestClient(baseURL: string) { | |||
|     if (status >= 200 && status < 400 && code === 0) { | ||||
|       return data; | ||||
|     } | ||||
|     throw new Error(msg); | ||||
|     throw new Error(`Error ${status}: ${msg}`); | ||||
|   }); | ||||
|   return client; | ||||
| } | ||||
|  |  | |||
|  | @ -58,7 +58,11 @@ async function loadDayjsLocale(lang: SupportedLanguagesType) { | |||
|       locale = await import('dayjs/locale/en'); | ||||
|     } | ||||
|   } | ||||
|   if (locale) { | ||||
|     dayjs.locale(locale); | ||||
|   } else { | ||||
|     console.error(`Failed to load dayjs locale for ${lang}`); | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  |  | |||
|  | @ -17,7 +17,6 @@ const routes: RouteRecordRaw[] = [ | |||
|     children: [ | ||||
|       { | ||||
|         meta: { | ||||
|           icon: 'mdi:shield-key-outline', | ||||
|           title: $t('page.demos.element-plus'), | ||||
|         }, | ||||
|         name: 'NaiveDemos', | ||||
|  |  | |||
|  | @ -1,11 +1,11 @@ | |||
| import type { RouteRecordRaw } from 'vue-router'; | ||||
| 
 | ||||
| import { | ||||
|   VBEN_ANT_PREVIEW_URL, | ||||
|   VBEN_DOC_URL, | ||||
|   VBEN_GITHUB_URL, | ||||
|   VBEN_LOGO_URL, | ||||
|   VBEN_NAIVE_PREVIEW_URL, | ||||
|   VBEN_PREVIEW_URL, | ||||
| } from '@vben/constants'; | ||||
| 
 | ||||
| import { BasicLayout, IFrameView } from '#/layouts'; | ||||
|  | @ -38,8 +38,7 @@ const routes: RouteRecordRaw[] = [ | |||
|         component: IFrameView, | ||||
|         meta: { | ||||
|           icon: 'lucide:book-open-text', | ||||
|           iframeSrc: VBEN_DOC_URL, | ||||
|           keepAlive: true, | ||||
|           link: VBEN_DOC_URL, | ||||
|           title: $t('page.vben.document'), | ||||
|         }, | ||||
|       }, | ||||
|  | @ -69,7 +68,7 @@ const routes: RouteRecordRaw[] = [ | |||
|         component: IFrameView, | ||||
|         meta: { | ||||
|           badgeType: 'dot', | ||||
|           link: VBEN_PREVIEW_URL, | ||||
|           link: VBEN_ANT_PREVIEW_URL, | ||||
|           title: $t('page.vben.antdv'), | ||||
|         }, | ||||
|       }, | ||||
|  |  | |||
|  | @ -1,5 +1,5 @@ | |||
| # 端口号 | ||||
| VITE_PORT=5777 | ||||
| VITE_PORT=5888 | ||||
| 
 | ||||
| VITE_BASE=/ | ||||
| 
 | ||||
|  |  | |||
|  | @ -58,7 +58,7 @@ function createRequestClient(baseURL: string) { | |||
|     if (status >= 200 && status < 400 && code === 0) { | ||||
|       return data; | ||||
|     } | ||||
|     throw new Error(msg); | ||||
|     throw new Error(`Error ${status}: ${msg}`); | ||||
|   }); | ||||
|   return client; | ||||
| } | ||||
|  |  | |||
|  | @ -17,7 +17,6 @@ const routes: RouteRecordRaw[] = [ | |||
|     children: [ | ||||
|       { | ||||
|         meta: { | ||||
|           icon: 'mdi:shield-key-outline', | ||||
|           title: $t('page.demos.naive'), | ||||
|         }, | ||||
|         name: 'NaiveDemos', | ||||
|  |  | |||
|  | @ -1,11 +1,11 @@ | |||
| import type { RouteRecordRaw } from 'vue-router'; | ||||
| 
 | ||||
| import { | ||||
|   VBEN_ANT_PREVIEW_URL, | ||||
|   VBEN_DOC_URL, | ||||
|   VBEN_ELE_PREVIEW_URL, | ||||
|   VBEN_GITHUB_URL, | ||||
|   VBEN_LOGO_URL, | ||||
|   VBEN_PREVIEW_URL, | ||||
| } from '@vben/constants'; | ||||
| 
 | ||||
| import { BasicLayout, IFrameView } from '#/layouts'; | ||||
|  | @ -38,8 +38,7 @@ const routes: RouteRecordRaw[] = [ | |||
|         component: IFrameView, | ||||
|         meta: { | ||||
|           icon: 'lucide:book-open-text', | ||||
|           iframeSrc: VBEN_DOC_URL, | ||||
|           keepAlive: true, | ||||
|           link: VBEN_DOC_URL, | ||||
|           title: $t('page.vben.document'), | ||||
|         }, | ||||
|       }, | ||||
|  | @ -59,7 +58,7 @@ const routes: RouteRecordRaw[] = [ | |||
|         component: IFrameView, | ||||
|         meta: { | ||||
|           badgeType: 'dot', | ||||
|           link: VBEN_PREVIEW_URL, | ||||
|           link: VBEN_ANT_PREVIEW_URL, | ||||
|           title: $t('page.vben.antdv'), | ||||
|         }, | ||||
|       }, | ||||
|  |  | |||
|  | @ -162,7 +162,11 @@ function nav(): DefaultTheme.NavItem[] { | |||
|           items: [ | ||||
|             { | ||||
|               link: 'https://www.vben.pro', | ||||
|               text: 'Ant Design Vue 版本(默认)', | ||||
|               text: '演示版本', | ||||
|             }, | ||||
|             { | ||||
|               link: 'https://ant.vben.pro', | ||||
|               text: 'Ant Design Vue 版本', | ||||
|             }, | ||||
|             { | ||||
|               link: 'https://naive.vben.pro', | ||||
|  | @ -250,6 +254,7 @@ function sidebarGuide(): DefaultTheme.SidebarItem[] { | |||
|           text: '为什么选择我们?', | ||||
|         }, | ||||
|         { link: 'introduction/quick-start', text: '快速开始' }, | ||||
|         { link: 'introduction/thin', text: '精简版本' }, | ||||
|       ], | ||||
|     }, | ||||
|     { | ||||
|  | @ -284,6 +289,7 @@ function sidebarGuide(): DefaultTheme.SidebarItem[] { | |||
|       items: [ | ||||
|         { link: 'project/standard', text: '规范' }, | ||||
|         { link: 'project/cli', text: 'CLI' }, | ||||
|         { link: 'project/dir', text: '目录说明' }, | ||||
|         { link: 'project/test', text: '单元测试' }, | ||||
|         { link: 'project/tailwindcss', text: 'Tailwind CSS' }, | ||||
|         { link: 'project/changeset', text: 'Changeset' }, | ||||
|  |  | |||
|  | @ -4,7 +4,10 @@ | |||
|   <div class="vp-doc vben-contributors"> | ||||
|     <p>Contributors</p> | ||||
|     <a href="https://github.com/vbenjs/vue-vben-admin/graphs/contributors"> | ||||
|       <img src="https://contrib.rocks/image?repo=vbenjs/vue-vben-admin" /> | ||||
|       <img | ||||
|         alt="Contributors" | ||||
|         src="https://opencollective.com/vbenjs/contributors.svg?button=false" | ||||
|       /> | ||||
|     </a> | ||||
|   </div> | ||||
| </template> | ||||
|  |  | |||
|  | @ -78,6 +78,8 @@ npm 脚本是项目常见的配置,用于执行一些常见的任务,比如 | |||
|     "dev:docs": "pnpm -F @vben/docs run dev", | ||||
|     // 启动web-ele应用 | ||||
|     "dev:ele": "pnpm -F @vben/web-ele run dev", | ||||
|     // 启动演示应用 | ||||
|     "dev:play": "pnpm -F @vben/playground run dev", | ||||
|     // 启动web-naive应用 | ||||
|     "dev:naive": "pnpm -F @vben/web-naive run dev", | ||||
|     // 格式化代码 | ||||
|  |  | |||
|  | @ -221,7 +221,7 @@ function createRequestClient(baseURL: string) { | |||
|     if (status >= 200 && status < 400 && code === 0) { | ||||
|       return data; | ||||
|     } | ||||
|     throw new Error(msg); | ||||
|     throw new Error(`Error ${status}: ${msg}`); | ||||
|   }); | ||||
|   return client; | ||||
| } | ||||
|  |  | |||
|  | @ -198,7 +198,11 @@ async function loadDayjsLocale(lang: SupportedLanguagesType) { | |||
|       locale = await import('dayjs/locale/en'); | ||||
|     } | ||||
|   } | ||||
|   if (locale) { | ||||
|     dayjs.locale(locale); | ||||
|   } else { | ||||
|     console.error(`Failed to load dayjs locale for ${lang}`); | ||||
|   } | ||||
| } | ||||
| ``` | ||||
| 
 | ||||
|  |  | |||
|  | @ -72,11 +72,24 @@ pnpm install | |||
| 
 | ||||
| ### 运行项目 | ||||
| 
 | ||||
| 执行以下命令即可运行项目: | ||||
| 执行以下命运行项目: | ||||
| 
 | ||||
| ```bash | ||||
| # 启动项目 | ||||
| pnpm dev | ||||
| ``` | ||||
| 
 | ||||
| 此时,你会看到类似如下的输出,选择你需要运行的项目: | ||||
| 
 | ||||
| ```bash | ||||
| │ | ||||
| ◆  Select the app you need to run [dev]: | ||||
| │  ● @vben/web-antd | ||||
| │  ○ @vben/web-ele | ||||
| │  ○ @vben/web-naive | ||||
| │  ○ @vben/docs | ||||
| │  ○ @vben/playground | ||||
| └ | ||||
| ``` | ||||
| 
 | ||||
| 现在,你可以在浏览器访问 `http://localhost:5555` 查看项目。 | ||||
|  |  | |||
|  | @ -0,0 +1,62 @@ | |||
| # 精简版本 | ||||
| 
 | ||||
| 从 `5.0` 版本开始,我们不再提供精简的仓库或者分支。我们的目标是提供一个更加一致的开发体验,同时减少维护成本。在这里,我们将如何介绍自己的项目,如何去精简以及移除不需要的功能。 | ||||
| 
 | ||||
| ## 应用精简 | ||||
| 
 | ||||
| 首先,确认你需要的 `UI` 组件库版本,然后删除对应的应用,比如你选择使用 `Ant Design Vue`,那么你可以删除其他应用, 只需要删除下面两个文件夹即可: | ||||
| 
 | ||||
| ```bash | ||||
| apps/web-ele | ||||
| apps/web-native | ||||
| 
 | ||||
| ``` | ||||
| 
 | ||||
| ::: tip | ||||
| 
 | ||||
| 如果项目没有内置你需要的 `UI` 组件库应用,你可以直接全部删除其他应用。然后自行新建应用即可。 | ||||
| 
 | ||||
| ::: | ||||
| 
 | ||||
| ## 演示代码精简 | ||||
| 
 | ||||
| 如果你不需要演示代码,你可以直接删除的`playground`文件夹。 | ||||
| 
 | ||||
| ## 文档精简 | ||||
| 
 | ||||
| 如果你不需要文档,你可以直接删除`docs`文件夹。 | ||||
| 
 | ||||
| ## Mock 服务精简 | ||||
| 
 | ||||
| 如果你不需要`Mock`服务,你可以直接删除`apps/backend-mock`文件夹。同时在你的应用下`.env.development`文件中删除`VITE_NITRO_MOCK`变量。 | ||||
| 
 | ||||
| ```bash | ||||
| # 是否开启 Nitro Mock服务,true 为开启,false 为关闭 | ||||
| VITE_NITRO_MOCK=false | ||||
| ``` | ||||
| 
 | ||||
| ## 安装依赖 | ||||
| 
 | ||||
| 到这里,你已经完成了精简操作,接下来你可以安装依赖,并启动你的项目: | ||||
| 
 | ||||
| ```bash | ||||
| # 根目录下执行 | ||||
| pnpm install | ||||
| 
 | ||||
| ``` | ||||
| 
 | ||||
| ## 命令调整 | ||||
| 
 | ||||
| 在精简后,你可能需要根据你的项目调整命令,在根目录下的`package.json`文件中,你可以调整`scripts`字段,移除你不需要的命令。 | ||||
| 
 | ||||
| ```json | ||||
| { | ||||
|   "scripts": { | ||||
|     "dev:antd": "pnpm -F @vben/web-antd run dev", | ||||
|     "dev:docs": "pnpm -F @vben/docs run dev", | ||||
|     "dev:ele": "pnpm -F @vben/web-ele run dev", | ||||
|     "dev:play": "pnpm -F @vben/playground run dev", | ||||
|     "dev:naive": "pnpm -F @vben/web-naive run dev" | ||||
|   } | ||||
| } | ||||
| ``` | ||||
|  | @ -0,0 +1,68 @@ | |||
| # 目录说明 | ||||
| 
 | ||||
| 目录使用 Monorepo 管理,项目结构如下: | ||||
| 
 | ||||
| ```bash | ||||
| . | ||||
| ├── Dockerfile # Docker 镜像构建文件 | ||||
| ├── README.md # 项目说明文档 | ||||
| ├── apps # 项目应用目录 | ||||
| │   ├── backend-mock # 后端模拟服务应用 | ||||
| │   ├── web-antd # 基于 Ant Design Vue 的前端应用 | ||||
| │   ├── web-ele # 基于 Element Plus 的前端应用 | ||||
| │   └── web-naive # 基于 Naive UI 的前端应用 | ||||
| ├── build-local-docker-image.sh # 本地构建 Docker 镜像脚本 | ||||
| ├── cspell.json # CSpell 配置文件 | ||||
| ├── docs # 项目文档目录 | ||||
| ├── eslint.config.mjs # ESLint 配置文件 | ||||
| ├── internal # 内部工具目录 | ||||
| │   ├── lint-configs # 代码检查配置 | ||||
| │   │   ├── commitlint-config # Commitlint 配置 | ||||
| │   │   ├── eslint-config # ESLint 配置 | ||||
| │   │   ├── prettier-config # Prettier 配置 | ||||
| │   │   └── stylelint-config # Stylelint 配置 | ||||
| │   ├── node-utils # Node.js 工具 | ||||
| │   ├── tailwind-config # Tailwind 配置 | ||||
| │   ├── tsconfig # 通用 tsconfig 配置 | ||||
| │   └── vite-config # 通用Vite 配置 | ||||
| ├── package.json # 项目依赖配置 | ||||
| ├── packages # 项目包目录 | ||||
| │   ├── @core # 核心包 | ||||
| │   │   ├── base # 基础包 | ||||
| │   │   │   ├── design # 设计相关 | ||||
| │   │   │   ├── icons # 图标 | ||||
| │   │   │   ├── shared # 共享 | ||||
| │   │   │   └── typings # 类型定义 | ||||
| │   │   ├── composables # 组合式 API | ||||
| │   │   ├── preferences # 偏好设置 | ||||
| │   │   └── ui-kit # UI 组件集合 | ||||
| │   │       ├── layout-ui # 布局 UI | ||||
| │   │       ├── menu-ui  # 菜单 UI | ||||
| │   │       ├── shadcn-ui # shadcn UI | ||||
| │   │       └── tabs-ui # 标签页 UI | ||||
| │   ├── constants # 常量 | ||||
| │   ├── effects # 副作用相关包 | ||||
| │   │   ├── access # 访问控制 | ||||
| │   │   ├── chart-ui # 图表 UI | ||||
| │   │   ├── common-ui # 通用 UI | ||||
| │   │   ├── hooks # 组合式 API | ||||
| │   │   ├── layouts # 布局 | ||||
| │   │   └── request # 请求 | ||||
| │   ├── icons # 图标 | ||||
| │   ├── locales # 国际化 | ||||
| │   ├── preferences  # 偏好设置 | ||||
| │   ├── stores # 状态管理 | ||||
| │   ├── styles # 样式 | ||||
| │   ├── types # 类型定义 | ||||
| │   └── utils # 工具 | ||||
| ├── playground # 演示目录 | ||||
| ├── pnpm-lock.yaml # pnpm 锁定文件 | ||||
| ├── pnpm-workspace.yaml # pnpm 工作区配置文件 | ||||
| ├── scripts # 脚本目录 | ||||
| │   ├── turbo-run # Turbo 运行脚本 | ||||
| │   └── vsh # VSH 脚本 | ||||
| ├── stylelint.config.mjs # Stylelint 配置文件 | ||||
| ├── turbo.json # Turbo 配置文件 | ||||
| ├── vben-admin.code-workspace # VS Code 工作区配置文件 | ||||
| └── vitest.config.ts # Vite 配置文件 | ||||
| ``` | ||||
|  | @ -13,7 +13,6 @@ | |||
| 
 | ||||
|     "moduleResolution": "node", | ||||
|     "resolveJsonModule": true, | ||||
|     "allowImportingTsExtensions": true, | ||||
| 
 | ||||
|     "strict": true, | ||||
|     "strictNullChecks": true, | ||||
|  |  | |||
|  | @ -41,6 +41,7 @@ | |||
|     "dev:antd": "pnpm -F @vben/web-antd run dev", | ||||
|     "dev:docs": "pnpm -F @vben/docs run dev", | ||||
|     "dev:ele": "pnpm -F @vben/web-ele run dev", | ||||
|     "dev:play": "pnpm -F @vben/playground run dev", | ||||
|     "dev:naive": "pnpm -F @vben/web-naive run dev", | ||||
|     "format": "vsh lint --format", | ||||
|     "lint": "vsh lint", | ||||
|  |  | |||
|  | @ -35,7 +35,9 @@ | |||
|     min-height: 100vh; | ||||
| 
 | ||||
|     /* overflow: overlay; */ | ||||
| 
 | ||||
|     -webkit-font-smoothing: antialiased; | ||||
|     -moz-osx-font-smoothing: grayscale; | ||||
|   } | ||||
| 
 | ||||
|   a, | ||||
|  |  | |||
|  | @ -8,7 +8,7 @@ | |||
| 
 | ||||
|   /* 主体区域背景色 */ | ||||
|   --background-deep: 210 11.11% 96.47%; | ||||
|   --foreground: 210 6% 21%; | ||||
|   --foreground: 222 84% 5%; | ||||
| 
 | ||||
|   /* Background color for <Card /> */ | ||||
|   --card: 0 0% 100%; | ||||
|  |  | |||
|  | @ -22,3 +22,5 @@ export const VBEN_PREVIEW_URL = 'https://www.vben.pro'; | |||
| export const VBEN_ELE_PREVIEW_URL = 'https://ele.vben.pro'; | ||||
| 
 | ||||
| export const VBEN_NAIVE_PREVIEW_URL = 'https://naive.vben.pro'; | ||||
| 
 | ||||
| export const VBEN_ANT_PREVIEW_URL = 'https://ant.vben.pro'; | ||||
|  |  | |||
|  | @ -125,7 +125,7 @@ const devDependenciesItems = Object.keys(devDependencies).map((key) => ({ | |||
|         <h3 class="text-foreground text-2xl font-semibold leading-7"> | ||||
|           {{ title }} | ||||
|         </h3> | ||||
|         <p class="text-foreground/80 mt-3 text-sm leading-6"> | ||||
|         <p class="text-foreground mt-3 text-sm leading-6"> | ||||
|           <VbenLink :href="VBEN_GITHUB_URL"> | ||||
|             {{ name }} | ||||
|           </VbenLink> | ||||
|  | @ -139,7 +139,7 @@ const devDependenciesItems = Object.keys(devDependencies).map((key) => ({ | |||
|               <dt class="text-foreground text-sm font-medium leading-6"> | ||||
|                 {{ item.title }} | ||||
|               </dt> | ||||
|               <dd class="text-foreground/80 mt-1 text-sm leading-6 sm:mt-2"> | ||||
|               <dd class="text-foreground mt-1 text-sm leading-6 sm:mt-2"> | ||||
|                 <VbenRenderContent :content="item.content" /> | ||||
|               </dd> | ||||
|             </div> | ||||
|  | @ -159,7 +159,7 @@ const devDependenciesItems = Object.keys(devDependencies).map((key) => ({ | |||
|               <dt class="text-foreground text-sm"> | ||||
|                 {{ item.title }} | ||||
|               </dt> | ||||
|               <dd class="text-foreground/60 mt-1 text-sm sm:mt-2"> | ||||
|               <dd class="text-foreground/80 mt-1 text-sm sm:mt-2"> | ||||
|                 <VbenRenderContent :content="item.content" /> | ||||
|               </dd> | ||||
|             </div> | ||||
|  | @ -178,7 +178,7 @@ const devDependenciesItems = Object.keys(devDependencies).map((key) => ({ | |||
|               <dt class="text-foreground text-sm"> | ||||
|                 {{ item.title }} | ||||
|               </dt> | ||||
|               <dd class="text-foreground/60 mt-1 text-sm sm:mt-2"> | ||||
|               <dd class="text-foreground/80 mt-1 text-sm sm:mt-2"> | ||||
|                 <VbenRenderContent :content="item.content" /> | ||||
|               </dd> | ||||
|             </div> | ||||
|  |  | |||
|  | @ -100,15 +100,11 @@ export function useTabbar() { | |||
|   watch( | ||||
|     () => route.path, | ||||
|     () => { | ||||
|       // 这里不能用route,用route时,vue-router会自动将父级meta进行合并
 | ||||
|       const routes = router.getRoutes(); | ||||
|       const currentRoute = routes.find((item) => item.path === route.path); | ||||
|       if (currentRoute) { | ||||
|       const meta = route.matched?.[route.matched.length - 1]?.meta; | ||||
|       tabbarStore.addTab({ | ||||
|         ...route, | ||||
|           meta: currentRoute.meta, | ||||
|         } as unknown as RouteLocationNormalizedGeneric); | ||||
|       } | ||||
|         meta: meta || route.meta, | ||||
|       }); | ||||
|     }, | ||||
|     { immediate: true }, | ||||
|   ); | ||||
|  |  | |||
|  | @ -0,0 +1,5 @@ | |||
| # 应用标题 | ||||
| VITE_APP_TITLE=Vben Admin | ||||
| 
 | ||||
| # 应用命名空间,用于缓存、store等功能的前缀,确保隔离 | ||||
| VITE_APP_NAMESPACE=vben-web-antd | ||||
|  | @ -0,0 +1,7 @@ | |||
| # public path | ||||
| VITE_BASE=/ | ||||
| 
 | ||||
| # Basic interface address SPA | ||||
| VITE_GLOB_API_URL=/api | ||||
| 
 | ||||
| VITE_VISUALIZER=true | ||||
|  | @ -0,0 +1,16 @@ | |||
| # 端口号 | ||||
| VITE_PORT=5555 | ||||
| 
 | ||||
| VITE_BASE=/ | ||||
| 
 | ||||
| # 接口地址 | ||||
| VITE_GLOB_API_URL=/api | ||||
| 
 | ||||
| # 是否开启 Nitro Mock服务,true 为开启,false 为关闭 | ||||
| VITE_NITRO_MOCK=true | ||||
| 
 | ||||
| # 是否打开 devtools,true 为打开,false 为关闭 | ||||
| VITE_DEVTOOLS=false | ||||
| 
 | ||||
| # 是否注入全局loading | ||||
| VITE_INJECT_APP_LOADING=true | ||||
|  | @ -0,0 +1,16 @@ | |||
| VITE_BASE=/ | ||||
| 
 | ||||
| # 接口地址 | ||||
| VITE_GLOB_API_URL=https://mock-napi.vben.pro/api | ||||
| 
 | ||||
| # 是否开启压缩,可以设置为 none, brotli, gzip | ||||
| VITE_COMPRESS=none | ||||
| 
 | ||||
| # 是否开启 PWA | ||||
| VITE_PWA=false | ||||
| 
 | ||||
| # vue-router 的模式 | ||||
| VITE_ROUTER_HISTORY=hash | ||||
| 
 | ||||
| # 是否注入全局loading | ||||
| VITE_INJECT_APP_LOADING=true | ||||
|  | @ -0,0 +1,35 @@ | |||
| <!doctype html> | ||||
| <html lang="zh"> | ||||
|   <head> | ||||
|     <meta charset="UTF-8" /> | ||||
|     <meta http-equiv="X-UA-Compatible" content="IE=edge,chrome=1" /> | ||||
|     <meta name="renderer" content="webkit" /> | ||||
|     <meta name="description" content="A Modern Back-end Management System" /> | ||||
|     <meta name="keywords" content="Vben Admin Vue3 Vite" /> | ||||
|     <meta name="author" content="Vben" /> | ||||
|     <meta | ||||
|       name="viewport" | ||||
|       content="width=device-width,initial-scale=1.0,minimum-scale=1.0,maximum-scale=1.0,user-scalable=0" | ||||
|     /> | ||||
|     <!-- 由 vite 注入 VITE_APP_TITLE 变量,在 .env 文件内配置 --> | ||||
|     <title><%= VITE_APP_TITLE %></title> | ||||
|     <link rel="icon" href="/favicon.ico" /> | ||||
|     <script> | ||||
|       // 生产环境下注入百度统计 | ||||
|       if (window._VBEN_ADMIN_PRO_APP_CONF_) { | ||||
|         var _hmt = _hmt || []; | ||||
|         (function () { | ||||
|           var hm = document.createElement('script'); | ||||
|           hm.src = | ||||
|             'https://hm.baidu.com/hm.js?d20a01273820422b6aa2ee41b6c9414d'; | ||||
|           var s = document.getElementsByTagName('script')[0]; | ||||
|           s.parentNode.insertBefore(hm, s); | ||||
|         })(); | ||||
|       } | ||||
|     </script> | ||||
|   </head> | ||||
|   <body> | ||||
|     <div id="app"></div> | ||||
|     <script type="module" src="/src/main.ts"></script> | ||||
|   </body> | ||||
| </html> | ||||
|  | @ -0,0 +1,50 @@ | |||
| { | ||||
|   "name": "@vben/playground", | ||||
|   "version": "5.0.1", | ||||
|   "homepage": "https://vben.pro", | ||||
|   "bugs": "https://github.com/vbenjs/vue-vben-admin/issues", | ||||
|   "repository": { | ||||
|     "type": "git", | ||||
|     "url": "git+https://github.com/vbenjs/vue-vben-admin.git", | ||||
|     "directory": "playground" | ||||
|   }, | ||||
|   "license": "MIT", | ||||
|   "author": { | ||||
|     "name": "vben", | ||||
|     "email": "ann.vben@gmail.com", | ||||
|     "url": "https://github.com/anncwb" | ||||
|   }, | ||||
|   "type": "module", | ||||
|   "scripts": { | ||||
|     "build": "pnpm vite build --mode production", | ||||
|     "build:analyze": "pnpm vite build --mode analyze", | ||||
|     "dev": "pnpm vite --mode development", | ||||
|     "preview": "vite preview", | ||||
|     "typecheck": "vue-tsc --noEmit --skipLibCheck" | ||||
|   }, | ||||
|   "imports": { | ||||
|     "#/*": "./src/*" | ||||
|   }, | ||||
|   "dependencies": { | ||||
|     "@vben/access": "workspace:*", | ||||
|     "@vben/chart-ui": "workspace:*", | ||||
|     "@vben/common-ui": "workspace:*", | ||||
|     "@vben/constants": "workspace:*", | ||||
|     "@vben/hooks": "workspace:*", | ||||
|     "@vben/icons": "workspace:*", | ||||
|     "@vben/layouts": "workspace:*", | ||||
|     "@vben/locales": "workspace:*", | ||||
|     "@vben/preferences": "workspace:*", | ||||
|     "@vben/request": "workspace:*", | ||||
|     "@vben/stores": "workspace:*", | ||||
|     "@vben/styles": "workspace:*", | ||||
|     "@vben/types": "workspace:*", | ||||
|     "@vben/utils": "workspace:*", | ||||
|     "@vueuse/core": "^10.11.1", | ||||
|     "ant-design-vue": "^4.2.3", | ||||
|     "dayjs": "^1.11.12", | ||||
|     "pinia": "2.2.1", | ||||
|     "vue": "^3.4.37", | ||||
|     "vue-router": "^4.4.3" | ||||
|   } | ||||
| } | ||||
|  | @ -0,0 +1 @@ | |||
| export { default } from '@vben/tailwind-config/postcss'; | ||||
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 5.3 KiB | 
|  | @ -0,0 +1,33 @@ | |||
| import { requestClient } from '#/api/request'; | ||||
| 
 | ||||
| export namespace AuthApi { | ||||
|   /** 登录接口参数 */ | ||||
|   export interface LoginParams { | ||||
|     password: string; | ||||
|     username: string; | ||||
|   } | ||||
| 
 | ||||
|   /** 登录接口返回值 */ | ||||
|   export interface LoginResult { | ||||
|     accessToken: string; | ||||
|     desc: string; | ||||
|     realName: string; | ||||
|     refreshToken: string; | ||||
|     userId: string; | ||||
|     username: string; | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * 登录 | ||||
|  */ | ||||
| export async function loginApi(data: AuthApi.LoginParams) { | ||||
|   return requestClient.post<AuthApi.LoginResult>('/auth/login', data); | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * 获取用户权限码 | ||||
|  */ | ||||
| export async function getAccessCodesApi() { | ||||
|   return requestClient.get<string[]>('/auth/codes'); | ||||
| } | ||||
|  | @ -0,0 +1,3 @@ | |||
| export * from './auth'; | ||||
| export * from './menu'; | ||||
| export * from './user'; | ||||
|  | @ -0,0 +1,10 @@ | |||
| import type { RouteRecordStringComponent } from '@vben/types'; | ||||
| 
 | ||||
| import { requestClient } from '#/api/request'; | ||||
| 
 | ||||
| /** | ||||
|  * 获取用户所有菜单 | ||||
|  */ | ||||
| export async function getAllMenusApi() { | ||||
|   return requestClient.get<RouteRecordStringComponent[]>('/menu/all'); | ||||
| } | ||||
|  | @ -0,0 +1,10 @@ | |||
| import type { UserInfo } from '@vben/types'; | ||||
| 
 | ||||
| import { requestClient } from '#/api/request'; | ||||
| 
 | ||||
| /** | ||||
|  * 获取用户信息 | ||||
|  */ | ||||
| export async function getUserInfoApi() { | ||||
|   return requestClient.get<UserInfo>('/user/info'); | ||||
| } | ||||
|  | @ -0,0 +1,2 @@ | |||
| export * from './core'; | ||||
| export * from './demos'; | ||||
|  | @ -0,0 +1,67 @@ | |||
| /** | ||||
|  * 该文件可自行根据业务逻辑进行调整 | ||||
|  */ | ||||
| import type { HttpResponse } from '@vben/request'; | ||||
| 
 | ||||
| import { useAppConfig } from '@vben/hooks'; | ||||
| import { preferences } from '@vben/preferences'; | ||||
| import { RequestClient } from '@vben/request'; | ||||
| import { useAccessStore } from '@vben/stores'; | ||||
| 
 | ||||
| import { message } from 'ant-design-vue'; | ||||
| 
 | ||||
| import { useAuthStore } from '#/store'; | ||||
| 
 | ||||
| const { apiURL } = useAppConfig(import.meta.env, import.meta.env.PROD); | ||||
| 
 | ||||
| function createRequestClient(baseURL: string) { | ||||
|   const client = new RequestClient({ | ||||
|     baseURL, | ||||
|     // 为每个请求携带 Authorization
 | ||||
|     makeAuthorization: () => { | ||||
|       return { | ||||
|         // 默认
 | ||||
|         key: 'Authorization', | ||||
|         tokenHandler: () => { | ||||
|           const accessStore = useAccessStore(); | ||||
|           return { | ||||
|             refreshToken: `${accessStore.refreshToken}`, | ||||
|             token: `${accessStore.accessToken}`, | ||||
|           }; | ||||
|         }, | ||||
|         unAuthorizedHandler: async () => { | ||||
|           const accessStore = useAccessStore(); | ||||
|           const authStore = useAuthStore(); | ||||
|           accessStore.setAccessToken(null); | ||||
| 
 | ||||
|           if (preferences.app.loginExpiredMode === 'modal') { | ||||
|             accessStore.setLoginExpired(true); | ||||
|           } else { | ||||
|             // 退出登录
 | ||||
|             await authStore.logout(); | ||||
|           } | ||||
|         }, | ||||
|       }; | ||||
|     }, | ||||
|     makeErrorMessage: (msg) => message.error(msg), | ||||
| 
 | ||||
|     makeRequestHeaders: () => { | ||||
|       return { | ||||
|         // 为每个请求携带 Accept-Language
 | ||||
|         'Accept-Language': preferences.app.locale, | ||||
|       }; | ||||
|     }, | ||||
|   }); | ||||
|   client.addResponseInterceptor<HttpResponse>((response) => { | ||||
|     const { data: responseData, status } = response; | ||||
| 
 | ||||
|     const { code, data, message: msg } = responseData; | ||||
|     if (status >= 200 && status < 400 && code === 0) { | ||||
|       return data; | ||||
|     } | ||||
|     throw new Error(`Error ${status}: ${msg}`); | ||||
|   }); | ||||
|   return client; | ||||
| } | ||||
| 
 | ||||
| export const requestClient = createRequestClient(apiURL); | ||||
|  | @ -0,0 +1,39 @@ | |||
| <script lang="ts" setup> | ||||
| import { computed } from 'vue'; | ||||
| 
 | ||||
| import { useAntdDesignTokens } from '@vben/hooks'; | ||||
| import { preferences, usePreferences } from '@vben/preferences'; | ||||
| 
 | ||||
| import { App, ConfigProvider, theme } from 'ant-design-vue'; | ||||
| 
 | ||||
| import { antdLocale } from '#/locales'; | ||||
| 
 | ||||
| defineOptions({ name: 'App' }); | ||||
| 
 | ||||
| const { isDark } = usePreferences(); | ||||
| const { tokens } = useAntdDesignTokens(); | ||||
| 
 | ||||
| const tokenTheme = computed(() => { | ||||
|   const algorithm = isDark.value | ||||
|     ? [theme.darkAlgorithm] | ||||
|     : [theme.defaultAlgorithm]; | ||||
| 
 | ||||
|   // antd 紧凑模式算法 | ||||
|   if (preferences.app.compact) { | ||||
|     algorithm.push(theme.compactAlgorithm); | ||||
|   } | ||||
| 
 | ||||
|   return { | ||||
|     algorithm, | ||||
|     token: tokens, | ||||
|   }; | ||||
| }); | ||||
| </script> | ||||
| 
 | ||||
| <template> | ||||
|   <ConfigProvider :locale="antdLocale" :theme="tokenTheme"> | ||||
|     <App> | ||||
|       <RouterView /> | ||||
|     </App> | ||||
|   </ConfigProvider> | ||||
| </template> | ||||
|  | @ -0,0 +1,31 @@ | |||
| import { createApp } from 'vue'; | ||||
| 
 | ||||
| import { registerAccessDirective } from '@vben/access'; | ||||
| import { initStores } from '@vben/stores'; | ||||
| import '@vben/styles'; | ||||
| import '@vben/styles/antd'; | ||||
| 
 | ||||
| import { setupI18n } from '#/locales'; | ||||
| 
 | ||||
| import App from './app.vue'; | ||||
| import { router } from './router'; | ||||
| 
 | ||||
| async function bootstrap(namespace: string) { | ||||
|   const app = createApp(App); | ||||
| 
 | ||||
|   // 国际化 i18n 配置
 | ||||
|   await setupI18n(app); | ||||
| 
 | ||||
|   // 配置 pinia-tore
 | ||||
|   await initStores(app, { namespace }); | ||||
| 
 | ||||
|   // 安装权限指令
 | ||||
|   registerAccessDirective(app); | ||||
| 
 | ||||
|   // 配置路由及路由守卫
 | ||||
|   app.use(router); | ||||
| 
 | ||||
|   app.mount('#app'); | ||||
| } | ||||
| 
 | ||||
| export { bootstrap }; | ||||
|  | @ -0,0 +1,154 @@ | |||
| <script lang="ts" setup> | ||||
| import type { NotificationItem } from '@vben/layouts'; | ||||
| 
 | ||||
| import { computed, ref } from 'vue'; | ||||
| import { useRouter } from 'vue-router'; | ||||
| 
 | ||||
| import { AuthenticationLoginExpiredModal } from '@vben/common-ui'; | ||||
| import { LOGIN_PATH, VBEN_DOC_URL, VBEN_GITHUB_URL } from '@vben/constants'; | ||||
| import { BookOpenText, CircleHelp, MdiGithub } from '@vben/icons'; | ||||
| import { | ||||
|   BasicLayout, | ||||
|   LockScreen, | ||||
|   Notification, | ||||
|   UserDropdown, | ||||
| } from '@vben/layouts'; | ||||
| import { preferences } from '@vben/preferences'; | ||||
| import { | ||||
|   resetAllStores, | ||||
|   storeToRefs, | ||||
|   useAccessStore, | ||||
|   useUserStore, | ||||
| } from '@vben/stores'; | ||||
| import { openWindow } from '@vben/utils'; | ||||
| 
 | ||||
| import { $t } from '#/locales'; | ||||
| import { resetRoutes } from '#/router'; | ||||
| import { useAuthStore } from '#/store'; | ||||
| 
 | ||||
| const notifications = ref<NotificationItem[]>([ | ||||
|   { | ||||
|     avatar: 'https://avatar.vercel.sh/vercel.svg?text=VB', | ||||
|     date: '3小时前', | ||||
|     isRead: true, | ||||
|     message: '描述信息描述信息描述信息', | ||||
|     title: '收到了 14 份新周报', | ||||
|   }, | ||||
|   { | ||||
|     avatar: 'https://avatar.vercel.sh/1', | ||||
|     date: '刚刚', | ||||
|     isRead: false, | ||||
|     message: '描述信息描述信息描述信息', | ||||
|     title: '朱偏右 回复了你', | ||||
|   }, | ||||
|   { | ||||
|     avatar: 'https://avatar.vercel.sh/1', | ||||
|     date: '2024-01-01', | ||||
|     isRead: false, | ||||
|     message: '描述信息描述信息描述信息', | ||||
|     title: '曲丽丽 评论了你', | ||||
|   }, | ||||
|   { | ||||
|     avatar: 'https://avatar.vercel.sh/satori', | ||||
|     date: '1天前', | ||||
|     isRead: false, | ||||
|     message: '描述信息描述信息描述信息', | ||||
|     title: '代办提醒', | ||||
|   }, | ||||
| ]); | ||||
| 
 | ||||
| const userStore = useUserStore(); | ||||
| const authStore = useAuthStore(); | ||||
| const accessStore = useAccessStore(); | ||||
| const showDot = computed(() => | ||||
|   notifications.value.some((item) => !item.isRead), | ||||
| ); | ||||
| 
 | ||||
| const menus = computed(() => [ | ||||
|   { | ||||
|     handler: () => { | ||||
|       openWindow(VBEN_DOC_URL, { | ||||
|         target: '_blank', | ||||
|       }); | ||||
|     }, | ||||
|     icon: BookOpenText, | ||||
|     text: $t('widgets.document'), | ||||
|   }, | ||||
|   { | ||||
|     handler: () => { | ||||
|       openWindow(VBEN_GITHUB_URL, { | ||||
|         target: '_blank', | ||||
|       }); | ||||
|     }, | ||||
|     icon: MdiGithub, | ||||
|     text: 'GitHub', | ||||
|   }, | ||||
|   { | ||||
|     handler: () => { | ||||
|       openWindow(`${VBEN_GITHUB_URL}/issues`, { | ||||
|         target: '_blank', | ||||
|       }); | ||||
|     }, | ||||
|     icon: CircleHelp, | ||||
|     text: $t('widgets.qa'), | ||||
|   }, | ||||
| ]); | ||||
| 
 | ||||
| const { loginLoading } = storeToRefs(authStore); | ||||
| 
 | ||||
| const avatar = computed(() => { | ||||
|   return userStore.userInfo?.avatar ?? preferences.app.defaultAvatar; | ||||
| }); | ||||
| 
 | ||||
| const router = useRouter(); | ||||
| 
 | ||||
| async function handleLogout() { | ||||
|   resetAllStores(); | ||||
|   resetRoutes(); | ||||
|   await router.replace(LOGIN_PATH); | ||||
| } | ||||
| 
 | ||||
| function handleNoticeClear() { | ||||
|   notifications.value = []; | ||||
| } | ||||
| 
 | ||||
| function handleMakeAll() { | ||||
|   notifications.value.forEach((item) => (item.isRead = true)); | ||||
| } | ||||
| </script> | ||||
| 
 | ||||
| <template> | ||||
|   <BasicLayout @clear-preferences-and-logout="handleLogout"> | ||||
|     <template #user-dropdown> | ||||
|       <UserDropdown | ||||
|         :avatar | ||||
|         :menus | ||||
|         :text="userStore.userInfo?.realName" | ||||
|         description="ann.vben@gmail.com" | ||||
|         tag-text="Pro" | ||||
|         @logout="handleLogout" | ||||
|       /> | ||||
|     </template> | ||||
|     <template #notification> | ||||
|       <Notification | ||||
|         :dot="showDot" | ||||
|         :notifications="notifications" | ||||
|         @clear="handleNoticeClear" | ||||
|         @make-all="handleMakeAll" | ||||
|       /> | ||||
|     </template> | ||||
|     <template #extra> | ||||
|       <AuthenticationLoginExpiredModal | ||||
|         v-model:open="accessStore.loginExpired" | ||||
|         :avatar | ||||
|         :loading="loginLoading" | ||||
|         password-placeholder="123456" | ||||
|         username-placeholder="vben" | ||||
|         @submit="authStore.authLogin" | ||||
|       /> | ||||
|     </template> | ||||
|     <template #lock-screen> | ||||
|       <LockScreen :avatar @to-login="handleLogout" /> | ||||
|     </template> | ||||
|   </BasicLayout> | ||||
| </template> | ||||
|  | @ -0,0 +1,8 @@ | |||
| const BasicLayout = () => import('./basic.vue'); | ||||
| 
 | ||||
| const IFrameView = () => import('@vben/layouts').then((m) => m.IFrameView); | ||||
| 
 | ||||
| const AuthPageLayout = () => | ||||
|   import('@vben/layouts').then((m) => m.AuthPageLayout); | ||||
| 
 | ||||
| export { AuthPageLayout, BasicLayout, IFrameView }; | ||||
|  | @ -0,0 +1,3 @@ | |||
| # locale | ||||
| 
 | ||||
| 每个app使用的国际化可能不同,这里用于扩展国际化的功能,例如扩展 dayjs、antd组件库的多语言切换,以及app本身的国际化文件。 | ||||
|  | @ -0,0 +1,94 @@ | |||
| import type { LocaleSetupOptions, SupportedLanguagesType } from '@vben/locales'; | ||||
| import type { Locale } from 'ant-design-vue/es/locale'; | ||||
| 
 | ||||
| import type { App } from 'vue'; | ||||
| import { ref } from 'vue'; | ||||
| 
 | ||||
| import { $t, setupI18n as coreSetup, loadLocalesMap } from '@vben/locales'; | ||||
| import { preferences } from '@vben/preferences'; | ||||
| 
 | ||||
| import antdEnLocale from 'ant-design-vue/es/locale/en_US'; | ||||
| import antdDefaultLocale from 'ant-design-vue/es/locale/zh_CN'; | ||||
| import dayjs from 'dayjs'; | ||||
| 
 | ||||
| const antdLocale = ref<Locale>(antdDefaultLocale); | ||||
| 
 | ||||
| const modules = import.meta.glob('./langs/*.json'); | ||||
| 
 | ||||
| const localesMap = loadLocalesMap(modules); | ||||
| 
 | ||||
| /** | ||||
|  * 加载应用特有的语言包 | ||||
|  * 这里也可以改造为从服务端获取翻译数据 | ||||
|  * @param lang | ||||
|  */ | ||||
| async function loadMessages(lang: SupportedLanguagesType) { | ||||
|   const [appLocaleMessages] = await Promise.all([ | ||||
|     localesMap[lang]?.(), | ||||
|     loadThirdPartyMessage(lang), | ||||
|   ]); | ||||
|   return appLocaleMessages?.default; | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * 加载第三方组件库的语言包 | ||||
|  * @param lang | ||||
|  */ | ||||
| async function loadThirdPartyMessage(lang: SupportedLanguagesType) { | ||||
|   await Promise.all([loadAntdLocale(lang), loadDayjsLocale(lang)]); | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * 加载dayjs的语言包 | ||||
|  * @param lang | ||||
|  */ | ||||
| async function loadDayjsLocale(lang: SupportedLanguagesType) { | ||||
|   let locale; | ||||
|   switch (lang) { | ||||
|     case 'zh-CN': { | ||||
|       locale = await import('dayjs/locale/zh-cn'); | ||||
|       break; | ||||
|     } | ||||
|     case 'en-US': { | ||||
|       locale = await import('dayjs/locale/en'); | ||||
|       break; | ||||
|     } | ||||
|     // 默认使用英语
 | ||||
|     default: { | ||||
|       locale = await import('dayjs/locale/en'); | ||||
|     } | ||||
|   } | ||||
|   if (locale) { | ||||
|     dayjs.locale(locale); | ||||
|   } else { | ||||
|     console.error(`Failed to load dayjs locale for ${lang}`); | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * 加载antd的语言包 | ||||
|  * @param lang | ||||
|  */ | ||||
| async function loadAntdLocale(lang: SupportedLanguagesType) { | ||||
|   switch (lang) { | ||||
|     case 'zh-CN': { | ||||
|       antdLocale.value = antdDefaultLocale; | ||||
|       break; | ||||
|     } | ||||
|     case 'en-US': { | ||||
|       antdLocale.value = antdEnLocale; | ||||
|       break; | ||||
|     } | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| async function setupI18n(app: App, options: LocaleSetupOptions = {}) { | ||||
|   await coreSetup(app, { | ||||
|     defaultLocale: preferences.app.locale, | ||||
|     loadMessages, | ||||
|     missingWarn: !import.meta.env.PROD, | ||||
|     ...options, | ||||
|   }); | ||||
| } | ||||
| 
 | ||||
| export { $t, antdLocale, loadMessages, setupI18n }; | ||||
|  | @ -0,0 +1,67 @@ | |||
| { | ||||
|   "page": { | ||||
|     "demos": { | ||||
|       "title": "Demos", | ||||
|       "access": { | ||||
|         "frontendPermissions": "Frontend Permissions", | ||||
|         "backendPermissions": "Backend Permissions", | ||||
|         "pageAccess": "Page Access", | ||||
|         "buttonControl": "Button Control", | ||||
|         "menuVisible403": "Menu Visible(403)", | ||||
|         "superVisible": "Visible to Super", | ||||
|         "adminVisible": "Visible to Admin", | ||||
|         "userVisible": "Visible to User" | ||||
|       }, | ||||
|       "nested": { | ||||
|         "title": "Nested Menu", | ||||
|         "menu1": "Menu 1", | ||||
|         "menu2": "Menu 2", | ||||
|         "menu2_1": "Menu 2-1", | ||||
|         "menu3": "Menu 3", | ||||
|         "menu3_1": "Menu 3-1", | ||||
|         "menu3_2": "Menu 3-2", | ||||
|         "menu3_2_1": "Menu 3-2-1" | ||||
|       }, | ||||
|       "outside": { | ||||
|         "title": "External Pages", | ||||
|         "embedded": "Embedded", | ||||
|         "externalLink": "External Link" | ||||
|       }, | ||||
|       "badge": { | ||||
|         "title": "Menu Badge", | ||||
|         "dot": "Dot Badge", | ||||
|         "text": "Text Badge", | ||||
|         "color": "Badge Color" | ||||
|       }, | ||||
|       "activeIcon": { | ||||
|         "title": "Active Menu Icon", | ||||
|         "children": "Children Active Icon" | ||||
|       }, | ||||
|       "fallback": { | ||||
|         "title": "Fallback Page" | ||||
|       }, | ||||
|       "features": { | ||||
|         "title": "Features", | ||||
|         "hideChildrenInMenu": "Hide Menu Children", | ||||
|         "loginExpired": "Login Expired", | ||||
|         "icons": "Icons", | ||||
|         "watermark": "Watermark", | ||||
|         "tabs": "Tabs", | ||||
|         "tabDetail": "Tab Detail Page" | ||||
|       }, | ||||
|       "breadcrumb": { | ||||
|         "navigation": "Breadcrumb Navigation", | ||||
|         "lateral": "Lateral Mode", | ||||
|         "lateralDetail": "Lateral Mode Detail", | ||||
|         "level": "Level Mode", | ||||
|         "levelDetail": "Level Mode Detail" | ||||
|       } | ||||
|     }, | ||||
|     "examples": { | ||||
|       "title": "Examples", | ||||
|       "ellipsis": { | ||||
|         "title": "EllipsisText" | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| } | ||||
|  | @ -0,0 +1,67 @@ | |||
| { | ||||
|   "page": { | ||||
|     "demos": { | ||||
|       "title": "演示", | ||||
|       "access": { | ||||
|         "frontendPermissions": "前端权限", | ||||
|         "backendPermissions": "后端权限", | ||||
|         "pageAccess": "页面访问", | ||||
|         "buttonControl": "按钮控制", | ||||
|         "menuVisible403": "菜单可见(403)", | ||||
|         "superVisible": "Super 可见", | ||||
|         "adminVisible": "Admin 可见", | ||||
|         "userVisible": "User 可见" | ||||
|       }, | ||||
|       "nested": { | ||||
|         "title": "嵌套菜单", | ||||
|         "menu1": "菜单 1", | ||||
|         "menu2": "菜单 2", | ||||
|         "menu2_1": "菜单 2-1", | ||||
|         "menu3": "菜单 3", | ||||
|         "menu3_1": "菜单 3-1", | ||||
|         "menu3_2": "菜单 3-2", | ||||
|         "menu3_2_1": "菜单 3-2-1" | ||||
|       }, | ||||
|       "outside": { | ||||
|         "title": "外部页面", | ||||
|         "embedded": "内嵌", | ||||
|         "externalLink": "外链" | ||||
|       }, | ||||
|       "badge": { | ||||
|         "title": "菜单徽标", | ||||
|         "dot": "点徽标", | ||||
|         "text": "文本徽标", | ||||
|         "color": "徽标颜色" | ||||
|       }, | ||||
|       "activeIcon": { | ||||
|         "title": "菜单激活图标", | ||||
|         "children": "子级激活图标" | ||||
|       }, | ||||
|       "fallback": { | ||||
|         "title": "缺省页" | ||||
|       }, | ||||
|       "features": { | ||||
|         "title": "功能", | ||||
|         "hideChildrenInMenu": "隐藏子菜单", | ||||
|         "loginExpired": "登录过期", | ||||
|         "icons": "图标", | ||||
|         "watermark": "水印", | ||||
|         "tabs": "标签页", | ||||
|         "tabDetail": "标签详情页" | ||||
|       }, | ||||
|       "breadcrumb": { | ||||
|         "navigation": "面包屑导航", | ||||
|         "lateral": "平级模式", | ||||
|         "level": "层级模式", | ||||
|         "levelDetail": "层级模式详情", | ||||
|         "lateralDetail": "平级模式详情" | ||||
|       } | ||||
|     }, | ||||
|     "examples": { | ||||
|       "title": "示例", | ||||
|       "ellipsis": { | ||||
|         "title": "文本省略" | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| } | ||||
|  | @ -0,0 +1,31 @@ | |||
| import { initPreferences } from '@vben/preferences'; | ||||
| import { unmountGlobalLoading } from '@vben/utils'; | ||||
| 
 | ||||
| import { overridesPreferences } from './preferences'; | ||||
| 
 | ||||
| /** | ||||
|  * 应用初始化完成之后再进行页面加载渲染 | ||||
|  */ | ||||
| async function initApplication() { | ||||
|   // name用于指定项目唯一标识
 | ||||
|   // 用于区分不同项目的偏好设置以及存储数据的key前缀以及其他一些需要隔离的数据
 | ||||
|   const env = import.meta.env.PROD ? 'prod' : 'dev'; | ||||
|   const appVersion = import.meta.env.VITE_APP_VERSION; | ||||
|   const namespace = `${import.meta.env.VITE_APP_NAMESPACE}-${appVersion}-${env}`; | ||||
| 
 | ||||
|   // app偏好设置初始化
 | ||||
|   await initPreferences({ | ||||
|     namespace, | ||||
|     overrides: overridesPreferences, | ||||
|   }); | ||||
| 
 | ||||
|   // 启动应用并挂载
 | ||||
|   // vue应用主要逻辑及视图
 | ||||
|   const { bootstrap } = await import('./bootstrap'); | ||||
|   await bootstrap(namespace); | ||||
| 
 | ||||
|   // 移除并销毁loading
 | ||||
|   unmountGlobalLoading(); | ||||
| } | ||||
| 
 | ||||
| initApplication(); | ||||
|  | @ -0,0 +1,12 @@ | |||
| import { defineOverridesPreferences } from '@vben/preferences'; | ||||
| 
 | ||||
| /** | ||||
|  * @description 项目配置文件 | ||||
|  * 只需要覆盖项目中的一部分配置,不需要的配置不用覆盖,会自动使用默认配置 | ||||
|  */ | ||||
| export const overridesPreferences = defineOverridesPreferences({ | ||||
|   // overrides
 | ||||
|   app: { | ||||
|     name: import.meta.env.VITE_APP_TITLE, | ||||
|   }, | ||||
| }); | ||||
|  | @ -0,0 +1,42 @@ | |||
| import type { | ||||
|   ComponentRecordType, | ||||
|   GenerateMenuAndRoutesOptions, | ||||
| } from '@vben/types'; | ||||
| 
 | ||||
| import { generateAccessible } from '@vben/access'; | ||||
| import { preferences } from '@vben/preferences'; | ||||
| 
 | ||||
| import { message } from 'ant-design-vue'; | ||||
| 
 | ||||
| import { getAllMenusApi } from '#/api'; | ||||
| import { BasicLayout, IFrameView } from '#/layouts'; | ||||
| import { $t } from '#/locales'; | ||||
| 
 | ||||
| const forbiddenComponent = () => import('#/views/_core/fallback/forbidden.vue'); | ||||
| 
 | ||||
| async function generateAccess(options: GenerateMenuAndRoutesOptions) { | ||||
|   const pageMap: ComponentRecordType = import.meta.glob('../views/**/*.vue'); | ||||
| 
 | ||||
|   const layoutMap: ComponentRecordType = { | ||||
|     BasicLayout, | ||||
|     IFrameView, | ||||
|   }; | ||||
| 
 | ||||
|   return await generateAccessible(preferences.app.accessMode, { | ||||
|     ...options, | ||||
|     fetchMenuListAsync: async () => { | ||||
|       message.loading({ | ||||
|         content: `${$t('common.loadingMenu')}...`, | ||||
|         duration: 1.5, | ||||
|       }); | ||||
|       return await getAllMenusApi(); | ||||
|     }, | ||||
|     // 可以指定没有权限跳转403页面
 | ||||
|     forbiddenComponent, | ||||
|     // 如果 route.meta.menuVisibleWithForbidden = true
 | ||||
|     layoutMap, | ||||
|     pageMap, | ||||
|   }); | ||||
| } | ||||
| 
 | ||||
| export { generateAccess }; | ||||
|  | @ -0,0 +1,138 @@ | |||
| import type { Router } from 'vue-router'; | ||||
| 
 | ||||
| import { DEFAULT_HOME_PATH, LOGIN_PATH } from '@vben/constants'; | ||||
| import { preferences } from '@vben/preferences'; | ||||
| import { useAccessStore, useUserStore } from '@vben/stores'; | ||||
| import { startProgress, stopProgress } from '@vben/utils'; | ||||
| 
 | ||||
| import { useTitle } from '@vueuse/core'; | ||||
| 
 | ||||
| import { $t } from '#/locales'; | ||||
| import { coreRouteNames, dynamicRoutes } from '#/router/routes'; | ||||
| import { useAuthStore } from '#/store'; | ||||
| 
 | ||||
| import { generateAccess } from './access'; | ||||
| 
 | ||||
| /** | ||||
|  * 通用守卫配置 | ||||
|  * @param router | ||||
|  */ | ||||
| function setupCommonGuard(router: Router) { | ||||
|   // 记录已经加载的页面
 | ||||
|   const loadedPaths = new Set<string>(); | ||||
| 
 | ||||
|   router.beforeEach(async (to) => { | ||||
|     to.meta.loaded = loadedPaths.has(to.path); | ||||
| 
 | ||||
|     // 页面加载进度条
 | ||||
|     if (!to.meta.loaded && preferences.transition.progress) { | ||||
|       startProgress(); | ||||
|     } | ||||
|     return true; | ||||
|   }); | ||||
| 
 | ||||
|   router.afterEach((to) => { | ||||
|     // 记录页面是否加载,如果已经加载,后续的页面切换动画等效果不在重复执行
 | ||||
| 
 | ||||
|     if (preferences.tabbar.enable) { | ||||
|       loadedPaths.add(to.path); | ||||
|     } | ||||
| 
 | ||||
|     // 关闭页面加载进度条
 | ||||
|     if (preferences.transition.progress) { | ||||
|       stopProgress(); | ||||
|     } | ||||
| 
 | ||||
|     // 动态修改标题
 | ||||
|     if (preferences.app.dynamicTitle) { | ||||
|       const { title } = to.meta; | ||||
|       // useTitle(`${$t(title)} - ${preferences.app.name}`);
 | ||||
|       useTitle(`${$t(title)} - ${preferences.app.name}`); | ||||
|     } | ||||
|   }); | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * 权限访问守卫配置 | ||||
|  * @param router | ||||
|  */ | ||||
| function setupAccessGuard(router: Router) { | ||||
|   router.beforeEach(async (to, from) => { | ||||
|     const accessStore = useAccessStore(); | ||||
|     const userStore = useUserStore(); | ||||
|     const authStore = useAuthStore(); | ||||
| 
 | ||||
|     // 基本路由,这些路由不需要进入权限拦截
 | ||||
|     if (coreRouteNames.includes(to.name as string)) { | ||||
|       if (to.path === LOGIN_PATH && accessStore.accessToken) { | ||||
|         return decodeURIComponent( | ||||
|           (to.query?.redirect as string) || DEFAULT_HOME_PATH, | ||||
|         ); | ||||
|       } | ||||
|       return true; | ||||
|     } | ||||
| 
 | ||||
|     // accessToken 检查
 | ||||
|     if (!accessStore.accessToken) { | ||||
|       // 明确声明忽略权限访问权限,则可以访问
 | ||||
|       if (to.meta.ignoreAccess) { | ||||
|         return true; | ||||
|       } | ||||
| 
 | ||||
|       // 没有访问权限,跳转登录页面
 | ||||
|       if (to.fullPath !== LOGIN_PATH) { | ||||
|         return { | ||||
|           path: LOGIN_PATH, | ||||
|           // 如不需要,直接删除 query
 | ||||
|           query: { redirect: encodeURIComponent(to.fullPath) }, | ||||
|           // 携带当前跳转的页面,登录后重新跳转该页面
 | ||||
|           replace: true, | ||||
|         }; | ||||
|       } | ||||
|       return to; | ||||
|     } | ||||
| 
 | ||||
|     const accessRoutes = accessStore.accessRoutes; | ||||
| 
 | ||||
|     // 是否已经生成过动态路由
 | ||||
|     if (accessRoutes && accessRoutes.length > 0) { | ||||
|       return true; | ||||
|     } | ||||
| 
 | ||||
|     // 生成路由表
 | ||||
|     // 当前登录用户拥有的角色标识列表
 | ||||
|     const userInfo = userStore.userInfo || (await authStore.fetchUserInfo()); | ||||
|     const userRoles = userInfo.roles ?? []; | ||||
| 
 | ||||
|     // 生成菜单和路由
 | ||||
|     const { accessibleMenus, accessibleRoutes } = await generateAccess({ | ||||
|       roles: userRoles, | ||||
|       router, | ||||
|       // 则会在菜单中显示,但是访问会被重定向到403
 | ||||
|       routes: dynamicRoutes, | ||||
|     }); | ||||
| 
 | ||||
|     // 保存菜单信息和路由信息
 | ||||
|     accessStore.setAccessMenus(accessibleMenus); | ||||
|     accessStore.setAccessRoutes(accessibleRoutes); | ||||
|     const redirectPath = (from.query.redirect ?? to.fullPath) as string; | ||||
| 
 | ||||
|     return { | ||||
|       ...router.resolve(decodeURIComponent(redirectPath)), | ||||
|       replace: true, | ||||
|     }; | ||||
|   }); | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * 项目守卫配置 | ||||
|  * @param router | ||||
|  */ | ||||
| function createRouterGuard(router: Router) { | ||||
|   /** 通用 */ | ||||
|   setupCommonGuard(router); | ||||
|   /** 权限访问 */ | ||||
|   setupAccessGuard(router); | ||||
| } | ||||
| 
 | ||||
| export { createRouterGuard }; | ||||
|  | @ -0,0 +1,32 @@ | |||
| import { | ||||
|   createRouter, | ||||
|   createWebHashHistory, | ||||
|   createWebHistory, | ||||
| } from 'vue-router'; | ||||
| 
 | ||||
| import { resetStaticRoutes } from '@vben/utils'; | ||||
| 
 | ||||
| import { createRouterGuard } from './guard'; | ||||
| import { routes } from './routes'; | ||||
| 
 | ||||
| /** | ||||
|  *  @zh_CN 创建vue-router实例 | ||||
|  */ | ||||
| const router = createRouter({ | ||||
|   history: | ||||
|     import.meta.env.VITE_ROUTER_HISTORY === 'hash' | ||||
|       ? createWebHashHistory(import.meta.env.VITE_BASE) | ||||
|       : createWebHistory(import.meta.env.VITE_BASE), | ||||
|   // 应该添加到路由的初始路由列表。
 | ||||
|   routes, | ||||
|   scrollBehavior: () => ({ left: 0, top: 0 }), | ||||
|   // 是否应该禁止尾部斜杠。
 | ||||
|   // strict: true,
 | ||||
| }); | ||||
| 
 | ||||
| const resetRoutes = () => resetStaticRoutes(router, routes); | ||||
| 
 | ||||
| // 创建路由守卫
 | ||||
| createRouterGuard(router); | ||||
| 
 | ||||
| export { resetRoutes, router }; | ||||
|  | @ -0,0 +1,86 @@ | |||
| import type { RouteRecordRaw } from 'vue-router'; | ||||
| 
 | ||||
| import { DEFAULT_HOME_PATH } from '@vben/constants'; | ||||
| 
 | ||||
| import { AuthPageLayout } from '#/layouts'; | ||||
| import { $t } from '#/locales'; | ||||
| import Login from '#/views/_core/authentication/login.vue'; | ||||
| 
 | ||||
| /** 全局404页面 */ | ||||
| const fallbackNotFoundRoute: RouteRecordRaw = { | ||||
|   component: () => import('#/views/_core/fallback/not-found.vue'), | ||||
|   meta: { | ||||
|     hideInBreadcrumb: true, | ||||
|     hideInMenu: true, | ||||
|     hideInTab: true, | ||||
|     title: '404', | ||||
|   }, | ||||
|   name: 'FallbackNotFound', | ||||
|   path: '/:path(.*)*', | ||||
| }; | ||||
| 
 | ||||
| /** 基本路由,这些路由是必须存在的 */ | ||||
| const coreRoutes: RouteRecordRaw[] = [ | ||||
|   { | ||||
|     meta: { | ||||
|       title: 'Root', | ||||
|     }, | ||||
|     name: 'Root', | ||||
|     path: '/', | ||||
|     redirect: DEFAULT_HOME_PATH, | ||||
|   }, | ||||
|   { | ||||
|     component: AuthPageLayout, | ||||
|     meta: { | ||||
|       title: 'Authentication', | ||||
|     }, | ||||
|     name: 'Authentication', | ||||
|     path: '/auth', | ||||
|     children: [ | ||||
|       { | ||||
|         name: 'Login', | ||||
|         path: 'login', | ||||
|         component: Login, | ||||
|         meta: { | ||||
|           title: $t('page.core.login'), | ||||
|         }, | ||||
|       }, | ||||
|       { | ||||
|         name: 'CodeLogin', | ||||
|         path: 'code-login', | ||||
|         component: () => import('#/views/_core/authentication/code-login.vue'), | ||||
|         meta: { | ||||
|           title: $t('page.core.codeLogin'), | ||||
|         }, | ||||
|       }, | ||||
|       { | ||||
|         name: 'QrCodeLogin', | ||||
|         path: 'qrcode-login', | ||||
|         component: () => | ||||
|           import('#/views/_core/authentication/qrcode-login.vue'), | ||||
|         meta: { | ||||
|           title: $t('page.core.qrcodeLogin'), | ||||
|         }, | ||||
|       }, | ||||
|       { | ||||
|         name: 'ForgetPassword', | ||||
|         path: 'forget-password', | ||||
|         component: () => | ||||
|           import('#/views/_core/authentication/forget-password.vue'), | ||||
|         meta: { | ||||
|           title: $t('page.core.forgetPassword'), | ||||
|         }, | ||||
|       }, | ||||
|       { | ||||
|         name: 'Register', | ||||
|         path: 'register', | ||||
|         component: () => import('#/views/_core/authentication/register.vue'), | ||||
|         meta: { | ||||
|           title: $t('page.core.register'), | ||||
|         }, | ||||
|       }, | ||||
|     ], | ||||
|   }, | ||||
| ]; | ||||
| 
 | ||||
| export { coreRoutes, fallbackNotFoundRoute }; | ||||
|  | @ -0,0 +1,31 @@ | |||
| import type { RouteRecordRaw } from 'vue-router'; | ||||
| 
 | ||||
| import { mergeRouteModules, traverseTreeValues } from '@vben/utils'; | ||||
| 
 | ||||
| import { coreRoutes, fallbackNotFoundRoute } from './core'; | ||||
| 
 | ||||
| const dynamicRouteFiles = import.meta.glob('./modules/**/*.ts', { | ||||
|   eager: true, | ||||
| }); | ||||
| 
 | ||||
| // 有需要可以自行打开注释,并创建文件夹
 | ||||
| // const staticRouteFiles = import.meta.glob('./static/**/*.ts', { eager: true });
 | ||||
| 
 | ||||
| /** 动态路由 */ | ||||
| const dynamicRoutes: RouteRecordRaw[] = mergeRouteModules(dynamicRouteFiles); | ||||
| 
 | ||||
| /** 静态路由列表,访问这些页面可以不需要权限 */ | ||||
| // const staticRoutes: RouteRecordRaw[] = mergeRouteModules(staticRouteFiles);
 | ||||
| const staticRoutes: RouteRecordRaw[] = []; | ||||
| 
 | ||||
| /** 路由列表,由基本路由+静态路由组成 */ | ||||
| const routes: RouteRecordRaw[] = [ | ||||
|   ...coreRoutes, | ||||
|   ...staticRoutes, | ||||
|   fallbackNotFoundRoute, | ||||
| ]; | ||||
| 
 | ||||
| /** 基本路由列表,这些路由不需要进入权限拦截 */ | ||||
| const coreRouteNames = traverseTreeValues(coreRoutes, (route) => route.name); | ||||
| 
 | ||||
| export { coreRouteNames, dynamicRoutes, routes }; | ||||
|  | @ -0,0 +1,39 @@ | |||
| import type { RouteRecordRaw } from 'vue-router'; | ||||
| 
 | ||||
| import { BasicLayout } from '#/layouts'; | ||||
| import { $t } from '#/locales'; | ||||
| 
 | ||||
| const routes: RouteRecordRaw[] = [ | ||||
|   { | ||||
|     component: BasicLayout, | ||||
|     meta: { | ||||
|       icon: 'lucide:layout-dashboard', | ||||
|       order: -1, | ||||
|       title: $t('page.dashboard.title'), | ||||
|     }, | ||||
|     name: 'Dashboard', | ||||
|     path: '/', | ||||
|     children: [ | ||||
|       { | ||||
|         name: 'Analytics', | ||||
|         path: '/analytics', | ||||
|         component: () => import('#/views/dashboard/analytics/index.vue'), | ||||
|         meta: { | ||||
|           affixTab: true, | ||||
|           icon: 'lucide:area-chart', | ||||
|           title: $t('page.dashboard.analytics'), | ||||
|         }, | ||||
|       }, | ||||
|       { | ||||
|         name: 'Workspace', | ||||
|         path: '/workspace', | ||||
|         component: () => import('#/views/dashboard/workspace/index.vue'), | ||||
|         meta: { | ||||
|           title: $t('page.dashboard.workspace'), | ||||
|         }, | ||||
|       }, | ||||
|     ], | ||||
|   }, | ||||
| ]; | ||||
| 
 | ||||
| export default routes; | ||||
|  | @ -0,0 +1,494 @@ | |||
| import type { RouteRecordRaw } from 'vue-router'; | ||||
| 
 | ||||
| import { BasicLayout, IFrameView } from '#/layouts'; | ||||
| import { $t } from '#/locales'; | ||||
| 
 | ||||
| const routes: RouteRecordRaw[] = [ | ||||
|   { | ||||
|     component: BasicLayout, | ||||
|     meta: { | ||||
|       icon: 'ic:baseline-view-in-ar', | ||||
|       keepAlive: true, | ||||
|       order: 1000, | ||||
|       title: $t('page.demos.title'), | ||||
|     }, | ||||
|     name: 'Demos', | ||||
|     path: '/demos', | ||||
|     children: [ | ||||
|       // 权限控制
 | ||||
|       { | ||||
|         meta: { | ||||
|           icon: 'mdi:shield-key-outline', | ||||
|           title: $t('page.demos.access.frontendPermissions'), | ||||
|         }, | ||||
|         name: 'AccessDemos', | ||||
|         path: '/demos/access', | ||||
|         children: [ | ||||
|           { | ||||
|             name: 'AccessPageControlDemo', | ||||
|             path: '/demos/access/page-control', | ||||
|             component: () => import('#/views/demos/access/index.vue'), | ||||
|             meta: { | ||||
|               icon: 'mdi:page-previous-outline', | ||||
|               title: $t('page.demos.access.pageAccess'), | ||||
|             }, | ||||
|           }, | ||||
|           { | ||||
|             name: 'AccessButtonControlDemo', | ||||
|             path: '/demos/access/button-control', | ||||
|             component: () => import('#/views/demos/access/button-control.vue'), | ||||
|             meta: { | ||||
|               icon: 'mdi:button-cursor', | ||||
|               title: $t('page.demos.access.buttonControl'), | ||||
|             }, | ||||
|           }, | ||||
|           { | ||||
|             name: 'AccessMenuVisible403Demo', | ||||
|             path: '/demos/access/menu-visible-403', | ||||
|             component: () => | ||||
|               import('#/views/demos/access/menu-visible-403.vue'), | ||||
|             meta: { | ||||
|               authority: ['no-body'], | ||||
|               icon: 'mdi:button-cursor', | ||||
|               menuVisibleWithForbidden: true, | ||||
|               title: $t('page.demos.access.menuVisible403'), | ||||
|             }, | ||||
|           }, | ||||
|           { | ||||
|             name: 'AccessSuperVisibleDemo', | ||||
|             path: '/demos/access/super-visible', | ||||
|             component: () => import('#/views/demos/access/super-visible.vue'), | ||||
|             meta: { | ||||
|               authority: ['super'], | ||||
|               icon: 'mdi:button-cursor', | ||||
|               title: $t('page.demos.access.superVisible'), | ||||
|             }, | ||||
|           }, | ||||
|           { | ||||
|             name: 'AccessAdminVisibleDemo', | ||||
|             path: '/demos/access/admin-visible', | ||||
|             component: () => import('#/views/demos/access/admin-visible.vue'), | ||||
|             meta: { | ||||
|               authority: ['admin'], | ||||
|               icon: 'mdi:button-cursor', | ||||
|               title: $t('page.demos.access.adminVisible'), | ||||
|             }, | ||||
|           }, | ||||
|           { | ||||
|             name: 'AccessUserVisibleDemo', | ||||
|             path: '/demos/access/user-visible', | ||||
|             component: () => import('#/views/demos/access/user-visible.vue'), | ||||
|             meta: { | ||||
|               authority: ['user'], | ||||
|               icon: 'mdi:button-cursor', | ||||
|               title: $t('page.demos.access.userVisible'), | ||||
|             }, | ||||
|           }, | ||||
|         ], | ||||
|       }, | ||||
|       // 功能
 | ||||
|       { | ||||
|         meta: { | ||||
|           icon: 'mdi:feature-highlight', | ||||
|           title: $t('page.demos.features.title'), | ||||
|         }, | ||||
|         name: 'FeaturesDemos', | ||||
|         path: '/demos/features', | ||||
|         children: [ | ||||
|           { | ||||
|             name: 'LoginExpiredDemo', | ||||
|             path: '/demos/features/login-expired', | ||||
|             component: () => | ||||
|               import('#/views/demos/features/login-expired/index.vue'), | ||||
|             meta: { | ||||
|               icon: 'mdi:encryption-expiration', | ||||
|               title: $t('page.demos.features.loginExpired'), | ||||
|             }, | ||||
|           }, | ||||
|           { | ||||
|             name: 'IconsDemo', | ||||
|             path: '/demos/features/icons', | ||||
|             component: () => import('#/views/demos/features/icons/index.vue'), | ||||
|             meta: { | ||||
|               title: $t('page.demos.features.icons'), | ||||
|             }, | ||||
|           }, | ||||
|           { | ||||
|             name: 'WatermarkDemo', | ||||
|             path: '/demos/features/watermark', | ||||
|             component: () => | ||||
|               import('#/views/demos/features/watermark/index.vue'), | ||||
|             meta: { | ||||
|               title: $t('page.demos.features.watermark'), | ||||
|             }, | ||||
|           }, | ||||
|           { | ||||
|             name: 'FeatureTabsDemo', | ||||
|             path: '/demos/features/tabs', | ||||
|             component: () => import('#/views/demos/features/tabs/index.vue'), | ||||
|             meta: { | ||||
|               icon: 'lucide:app-window', | ||||
|               title: $t('page.demos.features.tabs'), | ||||
|             }, | ||||
|           }, | ||||
|           { | ||||
|             name: 'FeatureTabDetailDemo', | ||||
|             path: '/demos/features/tabs/detail/:id', | ||||
|             component: () => | ||||
|               import('#/views/demos/features/tabs/tab-detail.vue'), | ||||
|             meta: { | ||||
|               activePath: '/demos/features/tabs', | ||||
|               hideInMenu: true, | ||||
|               maxNumOfOpenTab: 3, | ||||
|               title: $t('page.demos.features.tabDetail'), | ||||
|             }, | ||||
|           }, | ||||
|           { | ||||
|             name: 'HideChildrenInMenuParentDemo', | ||||
|             path: '/demos/features/hide-menu-children', | ||||
|             component: () => | ||||
|               import('#/views/demos/features/hide-menu-children/parent.vue'), | ||||
|             meta: { | ||||
|               hideChildrenInMenu: true, | ||||
|               icon: 'ic:round-menu', | ||||
|               title: $t('page.demos.features.hideChildrenInMenu'), | ||||
|             }, | ||||
|             children: [ | ||||
|               { | ||||
|                 name: 'HideChildrenInMenuChildrenDemo', | ||||
|                 path: '/demos/features/hide-menu-children/children', | ||||
|                 component: () => | ||||
|                   import( | ||||
|                     '#/views/demos/features/hide-menu-children/children.vue' | ||||
|                   ), | ||||
|                 meta: { title: 'HideChildrenInMenuChildrenDemo' }, | ||||
|               }, | ||||
|             ], | ||||
|           }, | ||||
|         ], | ||||
|       }, | ||||
|       // 面包屑导航
 | ||||
|       { | ||||
|         name: 'BreadcrumbDemos', | ||||
|         path: '/demos/breadcrumb', | ||||
|         meta: { | ||||
|           icon: 'lucide:navigation', | ||||
|           title: $t('page.demos.breadcrumb.navigation'), | ||||
|         }, | ||||
|         children: [ | ||||
|           { | ||||
|             name: 'BreadcrumbLateralDemo', | ||||
|             path: '/demos/breadcrumb/lateral', | ||||
|             component: () => import('#/views/demos/breadcrumb/lateral.vue'), | ||||
|             meta: { | ||||
|               icon: 'lucide:navigation', | ||||
|               title: $t('page.demos.breadcrumb.lateral'), | ||||
|             }, | ||||
|           }, | ||||
|           { | ||||
|             name: 'BreadcrumbLateralDetailDemo', | ||||
|             path: '/demos/breadcrumb/lateral-detail', | ||||
|             component: () => | ||||
|               import('#/views/demos/breadcrumb/lateral-detail.vue'), | ||||
|             meta: { | ||||
|               activePath: '/demos/breadcrumb/lateral', | ||||
|               hideInMenu: true, | ||||
|               title: $t('page.demos.breadcrumb.lateralDetail'), | ||||
|             }, | ||||
|           }, | ||||
|           { | ||||
|             name: 'BreadcrumbLevelDemo', | ||||
|             path: '/demos/breadcrumb/level', | ||||
|             meta: { | ||||
|               icon: 'lucide:navigation', | ||||
|               title: $t('page.demos.breadcrumb.level'), | ||||
|             }, | ||||
|             children: [ | ||||
|               { | ||||
|                 name: 'BreadcrumbLevelDetailDemo', | ||||
|                 path: '/demos/breadcrumb/level/detail', | ||||
|                 component: () => | ||||
|                   import('#/views/demos/breadcrumb/level-detail.vue'), | ||||
|                 meta: { | ||||
|                   title: $t('page.demos.breadcrumb.levelDetail'), | ||||
|                 }, | ||||
|               }, | ||||
|             ], | ||||
|           }, | ||||
|         ], | ||||
|       }, | ||||
|       // 缺省页
 | ||||
|       { | ||||
|         meta: { | ||||
|           icon: 'mdi:lightbulb-error-outline', | ||||
|           title: $t('page.demos.fallback.title'), | ||||
|         }, | ||||
|         name: 'FallbackDemos', | ||||
|         path: '/demos/fallback', | ||||
|         children: [ | ||||
|           { | ||||
|             name: 'Fallback403Demo', | ||||
|             path: '/demos/fallback/403', | ||||
|             component: () => import('#/views/_core/fallback/forbidden.vue'), | ||||
|             meta: { | ||||
|               icon: 'mdi:do-not-disturb-alt', | ||||
|               title: '403', | ||||
|             }, | ||||
|           }, | ||||
|           { | ||||
|             name: 'Fallback404Demo', | ||||
|             path: '/demos/fallback/404', | ||||
|             component: () => import('#/views/_core/fallback/not-found.vue'), | ||||
|             meta: { | ||||
|               icon: 'mdi:table-off', | ||||
|               title: '404', | ||||
|             }, | ||||
|           }, | ||||
|           { | ||||
|             name: 'Fallback500Demo', | ||||
|             path: '/demos/fallback/500', | ||||
|             component: () => | ||||
|               import('#/views/_core/fallback/internal-error.vue'), | ||||
|             meta: { | ||||
|               icon: 'mdi:server-network-off', | ||||
|               title: '500', | ||||
|             }, | ||||
|           }, | ||||
|           { | ||||
|             name: 'FallbackOfflineDemo', | ||||
|             path: '/demos/fallback/offline', | ||||
|             component: () => import('#/views/_core/fallback/offline.vue'), | ||||
|             meta: { | ||||
|               icon: 'mdi:offline', | ||||
|               title: $t('fallback.offline'), | ||||
|             }, | ||||
|           }, | ||||
|         ], | ||||
|       }, | ||||
|       // 菜单徽标
 | ||||
|       { | ||||
|         meta: { | ||||
|           badgeType: 'dot', | ||||
|           badgeVariants: 'destructive', | ||||
|           icon: 'lucide:circle-dot', | ||||
|           title: $t('page.demos.badge.title'), | ||||
|         }, | ||||
|         name: 'BadgeDemos', | ||||
|         path: '/demos/badge', | ||||
|         children: [ | ||||
|           { | ||||
|             name: 'BadgeDotDemo', | ||||
|             component: () => import('#/views/demos/badge/index.vue'), | ||||
|             path: '/demos/badge/dot', | ||||
|             meta: { | ||||
|               badgeType: 'dot', | ||||
|               icon: 'lucide:square-dot', | ||||
|               title: $t('page.demos.badge.dot'), | ||||
|             }, | ||||
|           }, | ||||
|           { | ||||
|             name: 'BadgeTextDemo', | ||||
|             component: () => import('#/views/demos/badge/index.vue'), | ||||
|             path: '/demos/badge/text', | ||||
|             meta: { | ||||
|               badge: '10', | ||||
|               icon: 'lucide:square-dot', | ||||
|               title: $t('page.demos.badge.text'), | ||||
|             }, | ||||
|           }, | ||||
|           { | ||||
|             name: 'BadgeColorDemo', | ||||
|             component: () => import('#/views/demos/badge/index.vue'), | ||||
|             path: '/demos/badge/color', | ||||
|             meta: { | ||||
|               badge: 'Hot', | ||||
|               badgeVariants: 'destructive', | ||||
|               icon: 'lucide:square-dot', | ||||
|               title: $t('page.demos.badge.color'), | ||||
|             }, | ||||
|           }, | ||||
|         ], | ||||
|       }, | ||||
|       // 菜单激活图标
 | ||||
|       { | ||||
|         meta: { | ||||
|           activeIcon: 'fluent-emoji:radioactive', | ||||
|           icon: 'bi:radioactive', | ||||
|           title: $t('page.demos.activeIcon.title'), | ||||
|         }, | ||||
|         name: 'ActiveIconDemos', | ||||
|         path: '/demos/active-icon', | ||||
|         children: [ | ||||
|           { | ||||
|             name: 'ActiveIconDemo', | ||||
|             component: () => import('#/views/demos/active-icon/index.vue'), | ||||
|             path: '/demos/active-icon/children', | ||||
|             meta: { | ||||
|               activeIcon: 'fluent-emoji:radioactive', | ||||
|               icon: 'bi:radioactive', | ||||
|               title: $t('page.demos.activeIcon.children'), | ||||
|             }, | ||||
|           }, | ||||
|         ], | ||||
|       }, | ||||
|       // 外部链接
 | ||||
|       { | ||||
|         meta: { | ||||
|           icon: 'ic:round-settings-input-composite', | ||||
|           title: $t('page.demos.outside.title'), | ||||
|         }, | ||||
|         name: 'OutsideDemos', | ||||
|         path: '/demos/outside', | ||||
|         children: [ | ||||
|           { | ||||
|             name: 'IframeDemos', | ||||
|             path: '/demos/outside/iframe', | ||||
|             meta: { | ||||
|               icon: 'mdi:newspaper-variant-outline', | ||||
|               title: $t('page.demos.outside.embedded'), | ||||
|             }, | ||||
|             children: [ | ||||
|               { | ||||
|                 name: 'VueDocumentDemo', | ||||
|                 path: '/demos/outside/iframe/vue-document', | ||||
|                 component: IFrameView, | ||||
|                 meta: { | ||||
|                   icon: 'logos:vue', | ||||
|                   iframeSrc: 'https://cn.vuejs.org/', | ||||
|                   keepAlive: true, | ||||
|                   title: 'Vue', | ||||
|                 }, | ||||
|               }, | ||||
|               { | ||||
|                 name: 'TailwindcssDemo', | ||||
|                 path: '/demos/outside/iframe/tailwindcss', | ||||
|                 component: IFrameView, | ||||
|                 meta: { | ||||
|                   icon: 'devicon:tailwindcss', | ||||
|                   iframeSrc: 'https://tailwindcss.com/', | ||||
|                   // keepAlive: true,
 | ||||
|                   title: 'Tailwindcss', | ||||
|                 }, | ||||
|               }, | ||||
|             ], | ||||
|           }, | ||||
|           { | ||||
|             name: 'ExternalLinkDemos', | ||||
|             path: '/demos/outside/external-link', | ||||
|             meta: { | ||||
|               icon: 'mdi:newspaper-variant-multiple-outline', | ||||
|               title: $t('page.demos.outside.externalLink'), | ||||
|             }, | ||||
|             children: [ | ||||
|               { | ||||
|                 name: 'ViteDemo', | ||||
|                 path: '/demos/outside/external-link/vite', | ||||
|                 component: IFrameView, | ||||
|                 meta: { | ||||
|                   icon: 'logos:vitejs', | ||||
|                   link: 'https://vitejs.dev/', | ||||
|                   title: 'Vite', | ||||
|                 }, | ||||
|               }, | ||||
|               { | ||||
|                 name: 'VueUseDemo', | ||||
|                 path: '/demos/outside/external-link/vue-use', | ||||
|                 component: IFrameView, | ||||
|                 meta: { | ||||
|                   icon: 'logos:vueuse', | ||||
|                   link: 'https://vueuse.org', | ||||
|                   title: 'VueUse', | ||||
|                 }, | ||||
|               }, | ||||
|             ], | ||||
|           }, | ||||
|         ], | ||||
|       }, | ||||
|       // 嵌套菜单
 | ||||
|       { | ||||
|         meta: { | ||||
|           icon: 'ic:round-menu', | ||||
|           title: $t('page.demos.nested.title'), | ||||
|         }, | ||||
|         name: 'NestedDemos', | ||||
|         path: '/demos/nested', | ||||
|         children: [ | ||||
|           { | ||||
|             name: 'Menu1Demo', | ||||
|             path: '/demos/nested/menu1', | ||||
|             component: () => import('#/views/demos/nested/menu-1.vue'), | ||||
|             meta: { | ||||
|               icon: 'ic:round-menu', | ||||
|               keepAlive: true, | ||||
|               title: $t('page.demos.nested.menu1'), | ||||
|             }, | ||||
|           }, | ||||
|           { | ||||
|             name: 'Menu2Demo', | ||||
|             path: '/demos/nested/menu2', | ||||
|             meta: { | ||||
|               icon: 'ic:round-menu', | ||||
|               keepAlive: true, | ||||
|               title: $t('page.demos.nested.menu2'), | ||||
|             }, | ||||
|             children: [ | ||||
|               { | ||||
|                 name: 'Menu21Demo', | ||||
|                 path: '/demos/nested/menu2/menu2-1', | ||||
|                 component: () => import('#/views/demos/nested/menu-2-1.vue'), | ||||
|                 meta: { | ||||
|                   icon: 'ic:round-menu', | ||||
|                   keepAlive: true, | ||||
|                   title: $t('page.demos.nested.menu2_1'), | ||||
|                 }, | ||||
|               }, | ||||
|             ], | ||||
|           }, | ||||
|           { | ||||
|             name: 'Menu3Demo', | ||||
|             path: '/demos/nested/menu3', | ||||
|             meta: { | ||||
|               icon: 'ic:round-menu', | ||||
|               title: $t('page.demos.nested.menu3'), | ||||
|             }, | ||||
|             children: [ | ||||
|               { | ||||
|                 name: 'Menu31Demo', | ||||
|                 path: 'menu3-1', | ||||
|                 component: () => import('#/views/demos/nested/menu-3-1.vue'), | ||||
|                 meta: { | ||||
|                   icon: 'ic:round-menu', | ||||
|                   keepAlive: true, | ||||
|                   title: $t('page.demos.nested.menu3_1'), | ||||
|                 }, | ||||
|               }, | ||||
|               { | ||||
|                 name: 'Menu32Demo', | ||||
|                 path: 'menu3-2', | ||||
|                 meta: { | ||||
|                   icon: 'ic:round-menu', | ||||
|                   title: $t('page.demos.nested.menu3_2'), | ||||
|                 }, | ||||
|                 children: [ | ||||
|                   { | ||||
|                     name: 'Menu321Demo', | ||||
|                     path: '/demos/nested/menu3/menu3-2/menu3-2-1', | ||||
|                     component: () => | ||||
|                       import('#/views/demos/nested/menu-3-2-1.vue'), | ||||
|                     meta: { | ||||
|                       icon: 'ic:round-menu', | ||||
|                       keepAlive: true, | ||||
|                       title: $t('page.demos.nested.menu3_2_1'), | ||||
|                     }, | ||||
|                   }, | ||||
|                 ], | ||||
|               }, | ||||
|             ], | ||||
|           }, | ||||
|         ], | ||||
|       }, | ||||
|     ], | ||||
|   }, | ||||
| ]; | ||||
| 
 | ||||
| export default routes; | ||||
|  | @ -0,0 +1,90 @@ | |||
| import type { RouteRecordRaw } from 'vue-router'; | ||||
| 
 | ||||
| import { | ||||
|   VBEN_ANT_PREVIEW_URL, | ||||
|   VBEN_DOC_URL, | ||||
|   VBEN_ELE_PREVIEW_URL, | ||||
|   VBEN_GITHUB_URL, | ||||
|   VBEN_LOGO_URL, | ||||
|   VBEN_NAIVE_PREVIEW_URL, | ||||
| } from '@vben/constants'; | ||||
| 
 | ||||
| import { BasicLayout, IFrameView } from '#/layouts'; | ||||
| import { $t } from '#/locales'; | ||||
| 
 | ||||
| const routes: RouteRecordRaw[] = [ | ||||
|   { | ||||
|     component: BasicLayout, | ||||
|     meta: { | ||||
|       badgeType: 'dot', | ||||
|       icon: VBEN_LOGO_URL, | ||||
|       order: 9999, | ||||
|       title: $t('page.vben.title'), | ||||
|     }, | ||||
|     name: 'VbenProject', | ||||
|     path: '/vben-admin', | ||||
|     children: [ | ||||
|       { | ||||
|         name: 'VbenAbout', | ||||
|         path: '/vben-admin/about', | ||||
|         component: () => import('#/views/_core/vben/about/index.vue'), | ||||
|         meta: { | ||||
|           icon: 'lucide:copyright', | ||||
|           title: $t('page.vben.about'), | ||||
|         }, | ||||
|       }, | ||||
|       { | ||||
|         name: 'VbenDocument', | ||||
|         path: '/vben-admin/document', | ||||
|         component: IFrameView, | ||||
|         meta: { | ||||
|           icon: 'lucide:book-open-text', | ||||
|           link: VBEN_DOC_URL, | ||||
|           title: $t('page.vben.document'), | ||||
|         }, | ||||
|       }, | ||||
|       { | ||||
|         name: 'VbenGithub', | ||||
|         path: '/vben-admin/github', | ||||
|         component: IFrameView, | ||||
|         meta: { | ||||
|           icon: 'mdi:github', | ||||
|           link: VBEN_GITHUB_URL, | ||||
|           title: 'Github', | ||||
|         }, | ||||
|       }, | ||||
|       { | ||||
|         name: 'VbenAntdv', | ||||
|         path: '/vben-admin/antdv', | ||||
|         component: IFrameView, | ||||
|         meta: { | ||||
|           badgeType: 'dot', | ||||
|           link: VBEN_ANT_PREVIEW_URL, | ||||
|           title: $t('page.vben.antdv'), | ||||
|         }, | ||||
|       }, | ||||
|       { | ||||
|         name: 'VbenNaive', | ||||
|         path: '/vben-admin/naive', | ||||
|         component: IFrameView, | ||||
|         meta: { | ||||
|           badgeType: 'dot', | ||||
|           link: VBEN_NAIVE_PREVIEW_URL, | ||||
|           title: $t('page.vben.naive-ui'), | ||||
|         }, | ||||
|       }, | ||||
|       { | ||||
|         name: 'VbenElementPlus', | ||||
|         path: '/vben-admin/ele', | ||||
|         component: IFrameView, | ||||
|         meta: { | ||||
|           badgeType: 'dot', | ||||
|           link: VBEN_ELE_PREVIEW_URL, | ||||
|           title: $t('page.vben.element-plus'), | ||||
|         }, | ||||
|       }, | ||||
|     ], | ||||
|   }, | ||||
| ]; | ||||
| 
 | ||||
| export default routes; | ||||
|  | @ -0,0 +1,111 @@ | |||
| import type { LoginAndRegisterParams } from '@vben/common-ui'; | ||||
| import type { UserInfo } from '@vben/types'; | ||||
| 
 | ||||
| import { ref } from 'vue'; | ||||
| import { useRouter } from 'vue-router'; | ||||
| 
 | ||||
| import { DEFAULT_HOME_PATH, LOGIN_PATH } from '@vben/constants'; | ||||
| import { resetAllStores, useAccessStore, useUserStore } from '@vben/stores'; | ||||
| 
 | ||||
| import { notification } from 'ant-design-vue'; | ||||
| import { defineStore } from 'pinia'; | ||||
| 
 | ||||
| import { getAccessCodesApi, getUserInfoApi, loginApi } from '#/api'; | ||||
| import { $t } from '#/locales'; | ||||
| 
 | ||||
| export const useAuthStore = defineStore('auth', () => { | ||||
|   const accessStore = useAccessStore(); | ||||
|   const userStore = useUserStore(); | ||||
|   const router = useRouter(); | ||||
| 
 | ||||
|   const loginLoading = ref(false); | ||||
| 
 | ||||
|   /** | ||||
|    * 异步处理登录操作 | ||||
|    * Asynchronously handle the login process | ||||
|    * @param params 登录表单数据 | ||||
|    */ | ||||
|   async function authLogin( | ||||
|     params: LoginAndRegisterParams, | ||||
|     onSuccess?: () => Promise<void> | void, | ||||
|   ) { | ||||
|     // 异步处理用户登录操作并获取 accessToken
 | ||||
|     let userInfo: null | UserInfo = null; | ||||
|     try { | ||||
|       loginLoading.value = true; | ||||
|       const { accessToken, refreshToken } = await loginApi(params); | ||||
| 
 | ||||
|       // 如果成功获取到 accessToken
 | ||||
|       if (accessToken) { | ||||
|         // 将 accessToken 存储到 accessStore 中
 | ||||
|         accessStore.setAccessToken(accessToken); | ||||
|         accessStore.setRefreshToken(refreshToken); | ||||
| 
 | ||||
|         // 获取用户信息并存储到 accessStore 中
 | ||||
|         const [fetchUserInfoResult, accessCodes] = await Promise.all([ | ||||
|           fetchUserInfo(), | ||||
|           getAccessCodesApi(), | ||||
|         ]); | ||||
| 
 | ||||
|         userInfo = fetchUserInfoResult; | ||||
| 
 | ||||
|         userStore.setUserInfo(userInfo); | ||||
|         accessStore.setAccessCodes(accessCodes); | ||||
| 
 | ||||
|         if (accessStore.loginExpired) { | ||||
|           accessStore.setLoginExpired(false); | ||||
|         } else { | ||||
|           onSuccess | ||||
|             ? await onSuccess?.() | ||||
|             : await router.push(userInfo.homePath || DEFAULT_HOME_PATH); | ||||
|         } | ||||
| 
 | ||||
|         if (userInfo?.realName) { | ||||
|           notification.success({ | ||||
|             description: `${$t('authentication.loginSuccessDesc')}:${userInfo?.realName}`, | ||||
|             duration: 3, | ||||
|             message: $t('authentication.loginSuccess'), | ||||
|           }); | ||||
|         } | ||||
|       } | ||||
|     } finally { | ||||
|       loginLoading.value = false; | ||||
|     } | ||||
| 
 | ||||
|     return { | ||||
|       userInfo, | ||||
|     }; | ||||
|   } | ||||
| 
 | ||||
|   async function logout() { | ||||
|     resetAllStores(); | ||||
|     accessStore.setLoginExpired(false); | ||||
| 
 | ||||
|     // 回登陆页带上当前路由地址
 | ||||
|     await router.replace({ | ||||
|       path: LOGIN_PATH, | ||||
|       query: { | ||||
|         redirect: encodeURIComponent(router.currentRoute.value.fullPath), | ||||
|       }, | ||||
|     }); | ||||
|   } | ||||
| 
 | ||||
|   async function fetchUserInfo() { | ||||
|     let userInfo: null | UserInfo = null; | ||||
|     userInfo = await getUserInfoApi(); | ||||
|     userStore.setUserInfo(userInfo); | ||||
|     return userInfo; | ||||
|   } | ||||
| 
 | ||||
|   function $reset() { | ||||
|     loginLoading.value = false; | ||||
|   } | ||||
| 
 | ||||
|   return { | ||||
|     $reset, | ||||
|     authLogin, | ||||
|     fetchUserInfo, | ||||
|     loginLoading, | ||||
|     logout, | ||||
|   }; | ||||
| }); | ||||
|  | @ -0,0 +1 @@ | |||
| export * from './auth'; | ||||
|  | @ -0,0 +1,3 @@ | |||
| # \_core | ||||
| 
 | ||||
| 此目录包含应用程序正常运行所需的基本视图。这些视图是应用程序布局中使用的视图。 | ||||
|  | @ -0,0 +1,30 @@ | |||
| <script lang="ts" setup> | ||||
| import type { LoginCodeParams } from '@vben/common-ui'; | ||||
| 
 | ||||
| import { ref } from 'vue'; | ||||
| 
 | ||||
| import { AuthenticationCodeLogin } from '@vben/common-ui'; | ||||
| import { LOGIN_PATH } from '@vben/constants'; | ||||
| 
 | ||||
| defineOptions({ name: 'CodeLogin' }); | ||||
| 
 | ||||
| const loading = ref(false); | ||||
| 
 | ||||
| /** | ||||
|  * 异步处理登录操作 | ||||
|  * Asynchronously handle the login process | ||||
|  * @param values 登录表单数据 | ||||
|  */ | ||||
| async function handleLogin(values: LoginCodeParams) { | ||||
|   // eslint-disable-next-line no-console | ||||
|   console.log(values); | ||||
| } | ||||
| </script> | ||||
| 
 | ||||
| <template> | ||||
|   <AuthenticationCodeLogin | ||||
|     :loading="loading" | ||||
|     :login-path="LOGIN_PATH" | ||||
|     @submit="handleLogin" | ||||
|   /> | ||||
| </template> | ||||
|  | @ -0,0 +1,23 @@ | |||
| <script lang="ts" setup> | ||||
| import { ref } from 'vue'; | ||||
| 
 | ||||
| import { AuthenticationForgetPassword } from '@vben/common-ui'; | ||||
| import { LOGIN_PATH } from '@vben/constants'; | ||||
| 
 | ||||
| defineOptions({ name: 'ForgetPassword' }); | ||||
| 
 | ||||
| const loading = ref(false); | ||||
| 
 | ||||
| function handleSubmit(value: string) { | ||||
|   // eslint-disable-next-line no-console | ||||
|   console.log('reset email:', value); | ||||
| } | ||||
| </script> | ||||
| 
 | ||||
| <template> | ||||
|   <AuthenticationForgetPassword | ||||
|     :loading="loading" | ||||
|     :login-path="LOGIN_PATH" | ||||
|     @submit="handleSubmit" | ||||
|   /> | ||||
| </template> | ||||
|  | @ -0,0 +1,18 @@ | |||
| <script lang="ts" setup> | ||||
| import { AuthenticationLogin } from '@vben/common-ui'; | ||||
| 
 | ||||
| import { useAuthStore } from '#/store'; | ||||
| 
 | ||||
| defineOptions({ name: 'Login' }); | ||||
| 
 | ||||
| const authStore = useAuthStore(); | ||||
| </script> | ||||
| 
 | ||||
| <template> | ||||
|   <AuthenticationLogin | ||||
|     :loading="authStore.loginLoading" | ||||
|     password-placeholder="123456" | ||||
|     username-placeholder="vben" | ||||
|     @submit="authStore.authLogin" | ||||
|   /> | ||||
| </template> | ||||
|  | @ -0,0 +1,10 @@ | |||
| <script lang="ts" setup> | ||||
| import { AuthenticationQrCodeLogin } from '@vben/common-ui'; | ||||
| import { LOGIN_PATH } from '@vben/constants'; | ||||
| 
 | ||||
| defineOptions({ name: 'QrCodeLogin' }); | ||||
| </script> | ||||
| 
 | ||||
| <template> | ||||
|   <AuthenticationQrCodeLogin :login-path="LOGIN_PATH" /> | ||||
| </template> | ||||
|  | @ -0,0 +1,25 @@ | |||
| <script lang="ts" setup> | ||||
| import type { LoginAndRegisterParams } from '@vben/common-ui'; | ||||
| 
 | ||||
| import { ref } from 'vue'; | ||||
| 
 | ||||
| import { AuthenticationRegister } from '@vben/common-ui'; | ||||
| import { LOGIN_PATH } from '@vben/constants'; | ||||
| 
 | ||||
| defineOptions({ name: 'Register' }); | ||||
| 
 | ||||
| const loading = ref(false); | ||||
| 
 | ||||
| function handleSubmit(value: LoginAndRegisterParams) { | ||||
|   // eslint-disable-next-line no-console | ||||
|   console.log('register submit:', value); | ||||
| } | ||||
| </script> | ||||
| 
 | ||||
| <template> | ||||
|   <AuthenticationRegister | ||||
|     :loading="loading" | ||||
|     :login-path="LOGIN_PATH" | ||||
|     @submit="handleSubmit" | ||||
|   /> | ||||
| </template> | ||||
|  | @ -0,0 +1,9 @@ | |||
| <script lang="ts" setup> | ||||
| import { Fallback } from '@vben/common-ui'; | ||||
| 
 | ||||
| defineOptions({ name: 'Fallback403Demo' }); | ||||
| </script> | ||||
| 
 | ||||
| <template> | ||||
|   <Fallback status="403" /> | ||||
| </template> | ||||
|  | @ -0,0 +1,9 @@ | |||
| <script lang="ts" setup> | ||||
| import { Fallback } from '@vben/common-ui'; | ||||
| 
 | ||||
| defineOptions({ name: 'Fallback500Demo' }); | ||||
| </script> | ||||
| 
 | ||||
| <template> | ||||
|   <Fallback status="500" /> | ||||
| </template> | ||||
|  | @ -0,0 +1,9 @@ | |||
| <script lang="ts" setup> | ||||
| import { Fallback } from '@vben/common-ui'; | ||||
| 
 | ||||
| defineOptions({ name: 'Fallback404Demo' }); | ||||
| </script> | ||||
| 
 | ||||
| <template> | ||||
|   <Fallback status="404" /> | ||||
| </template> | ||||
|  | @ -0,0 +1,9 @@ | |||
| <script lang="ts" setup> | ||||
| import { Fallback } from '@vben/common-ui'; | ||||
| 
 | ||||
| defineOptions({ name: 'FallbackOfflineDemo' }); | ||||
| </script> | ||||
| 
 | ||||
| <template> | ||||
|   <Fallback status="offline" /> | ||||
| </template> | ||||
|  | @ -0,0 +1,9 @@ | |||
| <script lang="ts" setup> | ||||
| import { About } from '@vben/common-ui'; | ||||
| 
 | ||||
| defineOptions({ name: 'About' }); | ||||
| </script> | ||||
| 
 | ||||
| <template> | ||||
|   <About /> | ||||
| </template> | ||||
|  | @ -0,0 +1,78 @@ | |||
| <script lang="ts" setup> | ||||
| import { onMounted, ref } from 'vue'; | ||||
| 
 | ||||
| import { EchartsUI, type EchartsUIType, useEcharts } from '@vben/chart-ui'; | ||||
| 
 | ||||
| const chartRef = ref<EchartsUIType>(); | ||||
| const { renderEcharts } = useEcharts(chartRef); | ||||
| 
 | ||||
| onMounted(() => { | ||||
|   renderEcharts({ | ||||
|     grid: { | ||||
|       bottom: 0, | ||||
|       containLabel: true, | ||||
|       left: '1%', | ||||
|       right: '1%', | ||||
|       top: '2 %', | ||||
|     }, | ||||
|     series: [ | ||||
|       { | ||||
|         areaStyle: {}, | ||||
|         data: [ | ||||
|           111, 2000, 6000, 16_000, 33_333, 55_555, 64_000, 33_333, 18_000, | ||||
|           36_000, 70_000, 42_444, 23_222, 13_000, 8000, 4000, 1200, 333, 222, | ||||
|           111, | ||||
|         ], | ||||
|         itemStyle: { | ||||
|           color: '#5ab1ef', | ||||
|         }, | ||||
|         smooth: true, | ||||
|         type: 'line', | ||||
|       }, | ||||
|       { | ||||
|         areaStyle: {}, | ||||
|         data: [ | ||||
|           33, 66, 88, 333, 3333, 6200, 20_000, 3000, 1200, 13_000, 22_000, | ||||
|           11_000, 2221, 1201, 390, 198, 60, 30, 22, 11, | ||||
|         ], | ||||
|         itemStyle: { | ||||
|           color: '#019680', | ||||
|         }, | ||||
|         smooth: true, | ||||
|         type: 'line', | ||||
|       }, | ||||
|     ], | ||||
|     tooltip: { | ||||
|       axisPointer: { | ||||
|         lineStyle: { | ||||
|           color: '#019680', | ||||
|           width: 1, | ||||
|         }, | ||||
|       }, | ||||
|       trigger: 'axis', | ||||
|     }, | ||||
|     xAxis: { | ||||
|       axisTick: { | ||||
|         show: false, | ||||
|       }, | ||||
|       boundaryGap: false, | ||||
|       data: Array.from({ length: 18 }).map((_item, index) => `${index + 6}:00`), | ||||
|       type: 'category', | ||||
|     }, | ||||
|     yAxis: [ | ||||
|       { | ||||
|         axisTick: { | ||||
|           show: false, | ||||
|         }, | ||||
|         max: 80_000, | ||||
| 
 | ||||
|         type: 'value', | ||||
|       }, | ||||
|     ], | ||||
|   }); | ||||
| }); | ||||
| </script> | ||||
| 
 | ||||
| <template> | ||||
|   <EchartsUI ref="chartRef" /> | ||||
| </template> | ||||
|  | @ -0,0 +1,80 @@ | |||
| <script lang="ts" setup> | ||||
| import { onMounted, ref } from 'vue'; | ||||
| 
 | ||||
| import { EchartsUI, type EchartsUIType, useEcharts } from '@vben/chart-ui'; | ||||
| 
 | ||||
| const chartRef = ref<EchartsUIType>(); | ||||
| const { renderEcharts } = useEcharts(chartRef); | ||||
| 
 | ||||
| onMounted(() => { | ||||
|   renderEcharts({ | ||||
|     legend: { | ||||
|       bottom: 0, | ||||
|       data: ['访问', '趋势'], | ||||
|     }, | ||||
|     radar: { | ||||
|       indicator: [ | ||||
|         { | ||||
|           name: '网页', | ||||
|         }, | ||||
|         { | ||||
|           name: '移动端', | ||||
|         }, | ||||
|         { | ||||
|           name: 'Ipad', | ||||
|         }, | ||||
|         { | ||||
|           name: '客户端', | ||||
|         }, | ||||
|         { | ||||
|           name: '第三方', | ||||
|         }, | ||||
|         { | ||||
|           name: '其它', | ||||
|         }, | ||||
|       ], | ||||
|       radius: '60%', | ||||
|       splitNumber: 8, | ||||
|     }, | ||||
|     series: [ | ||||
|       { | ||||
|         areaStyle: { | ||||
|           opacity: 1, | ||||
|           shadowBlur: 0, | ||||
|           shadowColor: 'rgba(0,0,0,.2)', | ||||
|           shadowOffsetX: 0, | ||||
|           shadowOffsetY: 10, | ||||
|         }, | ||||
|         data: [ | ||||
|           { | ||||
|             itemStyle: { | ||||
|               color: '#b6a2de', | ||||
|             }, | ||||
|             name: '访问', | ||||
|             value: [90, 50, 86, 40, 50, 20], | ||||
|           }, | ||||
|           { | ||||
|             itemStyle: { | ||||
|               color: '#5ab1ef', | ||||
|             }, | ||||
|             name: '趋势', | ||||
|             value: [70, 75, 70, 76, 20, 85], | ||||
|           }, | ||||
|         ], | ||||
|         itemStyle: { | ||||
|           // borderColor: '#fff', | ||||
|           borderRadius: 10, | ||||
|           borderWidth: 2, | ||||
|         }, | ||||
|         symbolSize: 0, | ||||
|         type: 'radar', | ||||
|       }, | ||||
|     ], | ||||
|     tooltip: {}, | ||||
|   }); | ||||
| }); | ||||
| </script> | ||||
| 
 | ||||
| <template> | ||||
|   <EchartsUI ref="chartRef" /> | ||||
| </template> | ||||
|  | @ -0,0 +1,44 @@ | |||
| <script lang="ts" setup> | ||||
| import { onMounted, ref } from 'vue'; | ||||
| 
 | ||||
| import { EchartsUI, type EchartsUIType, useEcharts } from '@vben/chart-ui'; | ||||
| 
 | ||||
| const chartRef = ref<EchartsUIType>(); | ||||
| const { renderEcharts } = useEcharts(chartRef); | ||||
| 
 | ||||
| onMounted(() => { | ||||
|   renderEcharts({ | ||||
|     series: [ | ||||
|       { | ||||
|         animationDelay() { | ||||
|           return Math.random() * 400; | ||||
|         }, | ||||
|         animationEasing: 'exponentialInOut', | ||||
|         animationType: 'scale', | ||||
|         center: ['50%', '50%'], | ||||
|         color: ['#5ab1ef', '#b6a2de', '#67e0e3', '#2ec7c9'], | ||||
|         data: [ | ||||
|           { name: '外包', value: 500 }, | ||||
|           { name: '定制', value: 310 }, | ||||
|           { name: '技术支持', value: 274 }, | ||||
|           { name: '远程', value: 400 }, | ||||
|         ].sort((a, b) => { | ||||
|           return a.value - b.value; | ||||
|         }), | ||||
|         name: '商业占比', | ||||
|         radius: '80%', | ||||
|         roseType: 'radius', | ||||
|         type: 'pie', | ||||
|       }, | ||||
|     ], | ||||
| 
 | ||||
|     tooltip: { | ||||
|       trigger: 'item', | ||||
|     }, | ||||
|   }); | ||||
| }); | ||||
| </script> | ||||
| 
 | ||||
| <template> | ||||
|   <EchartsUI ref="chartRef" /> | ||||
| </template> | ||||
|  | @ -0,0 +1,63 @@ | |||
| <script lang="ts" setup> | ||||
| import { onMounted, ref } from 'vue'; | ||||
| 
 | ||||
| import { EchartsUI, type EchartsUIType, useEcharts } from '@vben/chart-ui'; | ||||
| 
 | ||||
| const chartRef = ref<EchartsUIType>(); | ||||
| const { renderEcharts } = useEcharts(chartRef); | ||||
| 
 | ||||
| onMounted(() => { | ||||
|   renderEcharts({ | ||||
|     legend: { | ||||
|       bottom: '2%', | ||||
|       left: 'center', | ||||
|     }, | ||||
|     series: [ | ||||
|       { | ||||
|         animationDelay() { | ||||
|           return Math.random() * 100; | ||||
|         }, | ||||
|         animationEasing: 'exponentialInOut', | ||||
|         animationType: 'scale', | ||||
|         avoidLabelOverlap: false, | ||||
|         color: ['#5ab1ef', '#b6a2de', '#67e0e3', '#2ec7c9'], | ||||
|         data: [ | ||||
|           { name: '搜索引擎', value: 1048 }, | ||||
|           { name: '直接访问', value: 735 }, | ||||
|           { name: '邮件营销', value: 580 }, | ||||
|           { name: '联盟广告', value: 484 }, | ||||
|         ], | ||||
|         emphasis: { | ||||
|           label: { | ||||
|             fontSize: '12', | ||||
|             fontWeight: 'bold', | ||||
|             show: true, | ||||
|           }, | ||||
|         }, | ||||
|         itemStyle: { | ||||
|           // borderColor: '#fff', | ||||
|           borderRadius: 10, | ||||
|           borderWidth: 2, | ||||
|         }, | ||||
|         label: { | ||||
|           position: 'center', | ||||
|           show: false, | ||||
|         }, | ||||
|         labelLine: { | ||||
|           show: false, | ||||
|         }, | ||||
|         name: '访问来源', | ||||
|         radius: ['40%', '65%'], | ||||
|         type: 'pie', | ||||
|       }, | ||||
|     ], | ||||
|     tooltip: { | ||||
|       trigger: 'item', | ||||
|     }, | ||||
|   }); | ||||
| }); | ||||
| </script> | ||||
| 
 | ||||
| <template> | ||||
|   <EchartsUI ref="chartRef" /> | ||||
| </template> | ||||
|  | @ -0,0 +1,53 @@ | |||
| <script lang="ts" setup> | ||||
| import { onMounted, ref } from 'vue'; | ||||
| 
 | ||||
| import { EchartsUI, type EchartsUIType, useEcharts } from '@vben/chart-ui'; | ||||
| 
 | ||||
| const chartRef = ref<EchartsUIType>(); | ||||
| const { renderEcharts } = useEcharts(chartRef); | ||||
| 
 | ||||
| onMounted(() => { | ||||
|   renderEcharts({ | ||||
|     grid: { | ||||
|       bottom: 0, | ||||
|       containLabel: true, | ||||
|       left: '1%', | ||||
|       right: '1%', | ||||
|       top: '2 %', | ||||
|     }, | ||||
|     series: [ | ||||
|       { | ||||
|         barMaxWidth: 80, | ||||
|         // color: '#4f69fd', | ||||
|         data: [ | ||||
|           3000, 2000, 3333, 5000, 3200, 4200, 3200, 2100, 3000, 5100, 6000, | ||||
|           3200, 4800, | ||||
|         ], | ||||
|         type: 'bar', | ||||
|       }, | ||||
|     ], | ||||
|     tooltip: { | ||||
|       axisPointer: { | ||||
|         lineStyle: { | ||||
|           // color: '#4f69fd', | ||||
|           width: 1, | ||||
|         }, | ||||
|       }, | ||||
|       trigger: 'axis', | ||||
|     }, | ||||
|     xAxis: { | ||||
|       data: Array.from({ length: 12 }).map((_item, index) => `${index + 1}月`), | ||||
|       type: 'category', | ||||
|     }, | ||||
|     yAxis: { | ||||
|       max: 8000, | ||||
|       splitNumber: 4, | ||||
|       type: 'value', | ||||
|     }, | ||||
|   }); | ||||
| }); | ||||
| </script> | ||||
| 
 | ||||
| <template> | ||||
|   <EchartsUI ref="chartRef" /> | ||||
| </template> | ||||
|  | @ -0,0 +1,90 @@ | |||
| <script lang="ts" setup> | ||||
| import type { AnalysisOverviewItem } from '@vben/common-ui'; | ||||
| import type { TabOption } from '@vben/types'; | ||||
| 
 | ||||
| import { | ||||
|   AnalysisChartCard, | ||||
|   AnalysisChartsTabs, | ||||
|   AnalysisOverview, | ||||
| } from '@vben/common-ui'; | ||||
| import { | ||||
|   SvgBellIcon, | ||||
|   SvgCakeIcon, | ||||
|   SvgCardIcon, | ||||
|   SvgDownloadIcon, | ||||
| } from '@vben/icons'; | ||||
| 
 | ||||
| import AnalyticsTrends from './analytics-trends.vue'; | ||||
| import AnalyticsVisits from './analytics-visits.vue'; | ||||
| import AnalyticsVisitsData from './analytics-visits-data.vue'; | ||||
| import AnalyticsVisitsSales from './analytics-visits-sales.vue'; | ||||
| import AnalyticsVisitsSource from './analytics-visits-source.vue'; | ||||
| 
 | ||||
| const overviewItems: AnalysisOverviewItem[] = [ | ||||
|   { | ||||
|     icon: SvgCardIcon, | ||||
|     title: '用户量', | ||||
|     totalTitle: '总用户量', | ||||
|     totalValue: 120_000, | ||||
|     value: 2000, | ||||
|   }, | ||||
|   { | ||||
|     icon: SvgCakeIcon, | ||||
|     title: '访问量', | ||||
|     totalTitle: '总访问量', | ||||
|     totalValue: 500_000, | ||||
|     value: 20_000, | ||||
|   }, | ||||
|   { | ||||
|     icon: SvgDownloadIcon, | ||||
|     title: '下载量', | ||||
|     totalTitle: '总下载量', | ||||
|     totalValue: 120_000, | ||||
|     value: 8000, | ||||
|   }, | ||||
|   { | ||||
|     icon: SvgBellIcon, | ||||
|     title: '使用量', | ||||
|     totalTitle: '总使用量', | ||||
|     totalValue: 50_000, | ||||
|     value: 5000, | ||||
|   }, | ||||
| ]; | ||||
| 
 | ||||
| const chartTabs: TabOption[] = [ | ||||
|   { | ||||
|     label: '流量趋势', | ||||
|     value: 'trends', | ||||
|   }, | ||||
|   { | ||||
|     label: '月访问量', | ||||
|     value: 'visits', | ||||
|   }, | ||||
| ]; | ||||
| </script> | ||||
| 
 | ||||
| <template> | ||||
|   <div class="p-5"> | ||||
|     <AnalysisOverview :items="overviewItems" /> | ||||
|     <AnalysisChartsTabs :tabs="chartTabs" class="mt-5"> | ||||
|       <template #trends> | ||||
|         <AnalyticsTrends /> | ||||
|       </template> | ||||
|       <template #visits> | ||||
|         <AnalyticsVisits /> | ||||
|       </template> | ||||
|     </AnalysisChartsTabs> | ||||
| 
 | ||||
|     <div class="mt-5 w-full md:flex"> | ||||
|       <AnalysisChartCard class="mt-5 md:mr-4 md:mt-0 md:w-1/3" title="访问数量"> | ||||
|         <AnalyticsVisitsData /> | ||||
|       </AnalysisChartCard> | ||||
|       <AnalysisChartCard class="mt-5 md:mr-4 md:mt-0 md:w-1/3" title="访问来源"> | ||||
|         <AnalyticsVisitsSource /> | ||||
|       </AnalysisChartCard> | ||||
|       <AnalysisChartCard class="mt-5 md:mt-0 md:w-1/3" title="访问来源"> | ||||
|         <AnalyticsVisitsSales /> | ||||
|       </AnalysisChartCard> | ||||
|     </div> | ||||
|   </div> | ||||
| </template> | ||||
Some files were not shown because too many files have changed in this diff Show More
		Loading…
	
		Reference in New Issue
	
	 Vben
						Vben