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!
|
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:
|
||||||
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.
|
- [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"
|
- "perf"
|
||||||
- "chore"
|
- "chore"
|
||||||
- "dependencies"
|
- "dependencies"
|
||||||
collapse-after: 5
|
# collapse-after: 12
|
||||||
- title: 🚦 Tests
|
- title: 🚦 Tests
|
||||||
labels:
|
labels:
|
||||||
- "tests"
|
- "tests"
|
||||||
|
@ -40,11 +40,10 @@ version-resolver:
|
||||||
minor:
|
minor:
|
||||||
labels:
|
labels:
|
||||||
- "minor"
|
- "minor"
|
||||||
# - "feature"
|
- "feature"
|
||||||
patch:
|
patch:
|
||||||
labels:
|
labels:
|
||||||
- "patch"
|
- "patch"
|
||||||
- "feature"
|
|
||||||
- "bug"
|
- "bug"
|
||||||
- "maintenance"
|
- "maintenance"
|
||||||
- "docs"
|
- "docs"
|
||||||
|
|
|
@ -6,7 +6,46 @@ on:
|
||||||
- main
|
- main
|
||||||
|
|
||||||
jobs:
|
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
|
name: Deploy Push Ftp
|
||||||
if: github.actor != 'dependabot[bot]' && !contains(github.event.head_commit.message, '[skip ci]')
|
if: github.actor != 'dependabot[bot]' && !contains(github.event.head_commit.message, '[skip ci]')
|
||||||
runs-on: ubuntu-latest
|
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_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
|
sed -i "s#VITE_PWA\s*=.*#VITE_PWA = true#g" ./apps/web-antd/.env.production
|
||||||
cat ./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_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
|
sed -i "s#VITE_PWA\s*=.*#VITE_PWA = true#g" ./apps/web-ele/.env.production
|
||||||
cat ./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_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
|
sed -i "s#VITE_PWA\s*=.*#VITE_PWA = true#g" ./apps/web-naive/.env.production
|
||||||
cat ./apps/web-naive/.env.production
|
cat ./apps/web-naive/.env.production
|
||||||
|
@ -35,34 +130,10 @@ jobs:
|
||||||
- name: Build
|
- name: Build
|
||||||
run: pnpm run build
|
run: pnpm run build
|
||||||
|
|
||||||
- name: Sync Web Antd files
|
- 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/
|
|
||||||
|
|
||||||
- name: Sync Web Naive files
|
|
||||||
uses: SamKirkland/FTP-Deploy-Action@v4.3.5
|
uses: SamKirkland/FTP-Deploy-Action@v4.3.5
|
||||||
with:
|
with:
|
||||||
server: ${{ secrets.PRO_FTP_HOST }}
|
server: ${{ secrets.PRO_FTP_HOST }}
|
||||||
username: ${{ secrets.WEB_NAIVE_FTP_ACCOUNT }}
|
username: ${{ secrets.WEB_NAIVE_FTP_ACCOUNT }}
|
||||||
password: ${{ secrets.WEB_NAIVE_FTP_PASSWORD }}
|
password: ${{ secrets.WEB_NAIVE_FTP_PASSWORD }}
|
||||||
local-dir: ./apps/web-naive/dist/
|
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",
|
"esbenp.prettier-vscode",
|
||||||
// 支持 dotenv 文件语法
|
// 支持 dotenv 文件语法
|
||||||
"mikestead.dotenv",
|
"mikestead.dotenv",
|
||||||
// 获取每个 CSS 属性的初始值。
|
|
||||||
"dzhavat.css-initial-value",
|
|
||||||
// 使 VSCode 中的 TypeScript 错误更漂亮、更易于理解
|
|
||||||
"yoavbls.pretty-ts-errors",
|
|
||||||
// 源代码的拼写检查器
|
// 源代码的拼写检查器
|
||||||
"streetsidesoftware.code-spell-checker",
|
"streetsidesoftware.code-spell-checker",
|
||||||
// Tailwind CSS 的官方 VS Code 插件
|
// Tailwind CSS 的官方 VS Code 插件
|
||||||
|
|
|
@ -4,18 +4,27 @@
|
||||||
"configurations": [
|
"configurations": [
|
||||||
{
|
{
|
||||||
"type": "chrome",
|
"type": "chrome",
|
||||||
"name": "vben admin antd dev",
|
"name": "vben admin playground dev",
|
||||||
"request": "launch",
|
"request": "launch",
|
||||||
"url": "http://localhost:5555",
|
"url": "http://localhost:5555",
|
||||||
"env": { "NODE_ENV": "development" },
|
"env": { "NODE_ENV": "development" },
|
||||||
"sourceMaps": true,
|
"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"
|
"webRoot": "${workspaceFolder}/apps/web-antd/src"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"type": "chrome",
|
"type": "chrome",
|
||||||
"name": "vben admin ele dev",
|
"name": "vben admin ele dev",
|
||||||
"request": "launch",
|
"request": "launch",
|
||||||
"url": "http://localhost:5666",
|
"url": "http://localhost:5777",
|
||||||
"env": { "NODE_ENV": "development" },
|
"env": { "NODE_ENV": "development" },
|
||||||
"sourceMaps": true,
|
"sourceMaps": true,
|
||||||
"webRoot": "${workspaceFolder}/apps/web-ele/src"
|
"webRoot": "${workspaceFolder}/apps/web-ele/src"
|
||||||
|
@ -24,7 +33,7 @@
|
||||||
"type": "chrome",
|
"type": "chrome",
|
||||||
"name": "vben admin naive dev",
|
"name": "vben admin naive dev",
|
||||||
"request": "launch",
|
"request": "launch",
|
||||||
"url": "http://localhost:5777",
|
"url": "http://localhost:5888",
|
||||||
"env": { "NODE_ENV": "development" },
|
"env": { "NODE_ENV": "development" },
|
||||||
"sourceMaps": true,
|
"sourceMaps": true,
|
||||||
"webRoot": "${workspaceFolder}/apps/web-naive/src"
|
"webRoot": "${workspaceFolder}/apps/web-naive/src"
|
||||||
|
|
|
@ -167,6 +167,7 @@
|
||||||
|
|
||||||
"i18n-ally.localesPaths": [
|
"i18n-ally.localesPaths": [
|
||||||
"packages/locales/src/langs",
|
"packages/locales/src/langs",
|
||||||
|
"playground/src/langs",
|
||||||
"apps/*/src/locales/langs"
|
"apps/*/src/locales/langs"
|
||||||
],
|
],
|
||||||
"i18n-ally.enabledParsers": ["json", "ts", "js", "yaml"],
|
"i18n-ally.enabledParsers": ["json", "ts", "js", "yaml"],
|
||||||
|
|
|
@ -21,7 +21,7 @@ RUN echo "Builder Success 🎉"
|
||||||
FROM nginx:stable-alpine as production
|
FROM nginx:stable-alpine as production
|
||||||
|
|
||||||
RUN echo "types { application/javascript js mjs; }" > /etc/nginx/conf.d/mjs.conf
|
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
|
COPY ./nginx.conf /etc/nginx/nginx.conf
|
||||||
|
|
||||||
|
|
|
@ -134,7 +134,8 @@ pnpm build
|
||||||
## 貢献者
|
## 貢献者
|
||||||
|
|
||||||
<a href="https://github.com/vbenjs/vue-vben-admin/graphs/contributors">
|
<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>
|
</a>
|
||||||
|
|
||||||
## Discord
|
## Discord
|
||||||
|
|
|
@ -133,7 +133,8 @@ If you think this project is helpful to you, you can help the author buy a cup o
|
||||||
## Contributor
|
## Contributor
|
||||||
|
|
||||||
<a href="https://github.com/vbenjs/vue-vben-admin/graphs/contributors">
|
<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>
|
</a>
|
||||||
|
|
||||||
## Discord
|
## 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>
|
<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
|
## Discord
|
||||||
|
|
||||||
- [Github Discussions](https://github.com/anncwb/vue-vben-admin/discussions)
|
- [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等功能的前缀,确保隔离
|
# 应用命名空间,用于缓存、store等功能的前缀,确保隔离
|
||||||
VITE_APP_NAMESPACE=vben-web-antd
|
VITE_APP_NAMESPACE=vben-web-antd
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
# 端口号
|
# 端口号
|
||||||
VITE_PORT=5555
|
VITE_PORT=5666
|
||||||
|
|
||||||
VITE_BASE=/
|
VITE_BASE=/
|
||||||
|
|
||||||
|
|
|
@ -21,7 +21,7 @@
|
||||||
(function () {
|
(function () {
|
||||||
var hm = document.createElement('script');
|
var hm = document.createElement('script');
|
||||||
hm.src =
|
hm.src =
|
||||||
'https://hm.baidu.com/hm.js?d20a01273820422b6aa2ee41b6c9414d';
|
'https://hm.baidu.com/hm.js?b38e689f40558f20a9a686d7f6f33edf';
|
||||||
var s = document.getElementsByTagName('script')[0];
|
var s = document.getElementsByTagName('script')[0];
|
||||||
s.parentNode.insertBefore(hm, s);
|
s.parentNode.insertBefore(hm, s);
|
||||||
})();
|
})();
|
||||||
|
|
|
@ -1,2 +1 @@
|
||||||
export * from './core';
|
export * from './core';
|
||||||
export * from './demos';
|
|
||||||
|
|
|
@ -59,7 +59,7 @@ function createRequestClient(baseURL: string) {
|
||||||
if (status >= 200 && status < 400 && code === 0) {
|
if (status >= 200 && status < 400 && code === 0) {
|
||||||
return data;
|
return data;
|
||||||
}
|
}
|
||||||
throw new Error(msg);
|
throw new Error(`Error ${status}: ${msg}`);
|
||||||
});
|
});
|
||||||
return client;
|
return client;
|
||||||
}
|
}
|
||||||
|
|
|
@ -58,7 +58,11 @@ async function loadDayjsLocale(lang: SupportedLanguagesType) {
|
||||||
locale = await import('dayjs/locale/en');
|
locale = await import('dayjs/locale/en');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
if (locale) {
|
||||||
dayjs.locale(locale);
|
dayjs.locale(locale);
|
||||||
|
} else {
|
||||||
|
console.error(`Failed to load dayjs locale for ${lang}`);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -2,66 +2,7 @@
|
||||||
"page": {
|
"page": {
|
||||||
"demos": {
|
"demos": {
|
||||||
"title": "Demos",
|
"title": "Demos",
|
||||||
"access": {
|
"antd": "Ant Design Vue"
|
||||||
"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"
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -2,66 +2,7 @@
|
||||||
"page": {
|
"page": {
|
||||||
"demos": {
|
"demos": {
|
||||||
"title": "演示",
|
"title": "演示",
|
||||||
"access": {
|
"antd": "Ant Design Vue"
|
||||||
"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": "文本省略"
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
import type { RouteRecordRaw } from 'vue-router';
|
import type { RouteRecordRaw } from 'vue-router';
|
||||||
|
|
||||||
import { BasicLayout, IFrameView } from '#/layouts';
|
import { BasicLayout } from '#/layouts';
|
||||||
import { $t } from '#/locales';
|
import { $t } from '#/locales';
|
||||||
|
|
||||||
const routes: RouteRecordRaw[] = [
|
const routes: RouteRecordRaw[] = [
|
||||||
|
@ -15,477 +15,13 @@ const routes: RouteRecordRaw[] = [
|
||||||
name: 'Demos',
|
name: 'Demos',
|
||||||
path: '/demos',
|
path: '/demos',
|
||||||
children: [
|
children: [
|
||||||
// 权限控制
|
|
||||||
{
|
{
|
||||||
meta: {
|
meta: {
|
||||||
icon: 'mdi:shield-key-outline',
|
title: $t('page.demos.antd'),
|
||||||
title: $t('page.demos.access.frontendPermissions'),
|
|
||||||
},
|
},
|
||||||
name: 'AccessDemos',
|
name: 'AntDesignDemos',
|
||||||
path: '/demos/access',
|
path: '/demos/ant-design',
|
||||||
children: [
|
component: () => import('#/views/demos/antd/index.vue'),
|
||||||
{
|
|
||||||
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'),
|
|
||||||
},
|
|
||||||
},
|
|
||||||
],
|
|
||||||
},
|
|
||||||
],
|
|
||||||
},
|
|
||||||
],
|
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
|
|
|
@ -38,8 +38,7 @@ const routes: RouteRecordRaw[] = [
|
||||||
component: IFrameView,
|
component: IFrameView,
|
||||||
meta: {
|
meta: {
|
||||||
icon: 'lucide:book-open-text',
|
icon: 'lucide:book-open-text',
|
||||||
iframeSrc: VBEN_DOC_URL,
|
link: VBEN_DOC_URL,
|
||||||
keepAlive: true,
|
|
||||||
title: $t('page.vben.document'),
|
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=/
|
VITE_BASE=/
|
||||||
|
|
||||||
|
|
|
@ -59,7 +59,7 @@ function createRequestClient(baseURL: string) {
|
||||||
if (status >= 200 && status < 400 && code === 0) {
|
if (status >= 200 && status < 400 && code === 0) {
|
||||||
return data;
|
return data;
|
||||||
}
|
}
|
||||||
throw new Error(msg);
|
throw new Error(`Error ${status}: ${msg}`);
|
||||||
});
|
});
|
||||||
return client;
|
return client;
|
||||||
}
|
}
|
||||||
|
|
|
@ -58,7 +58,11 @@ async function loadDayjsLocale(lang: SupportedLanguagesType) {
|
||||||
locale = await import('dayjs/locale/en');
|
locale = await import('dayjs/locale/en');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
if (locale) {
|
||||||
dayjs.locale(locale);
|
dayjs.locale(locale);
|
||||||
|
} else {
|
||||||
|
console.error(`Failed to load dayjs locale for ${lang}`);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -17,7 +17,6 @@ const routes: RouteRecordRaw[] = [
|
||||||
children: [
|
children: [
|
||||||
{
|
{
|
||||||
meta: {
|
meta: {
|
||||||
icon: 'mdi:shield-key-outline',
|
|
||||||
title: $t('page.demos.element-plus'),
|
title: $t('page.demos.element-plus'),
|
||||||
},
|
},
|
||||||
name: 'NaiveDemos',
|
name: 'NaiveDemos',
|
||||||
|
|
|
@ -1,11 +1,11 @@
|
||||||
import type { RouteRecordRaw } from 'vue-router';
|
import type { RouteRecordRaw } from 'vue-router';
|
||||||
|
|
||||||
import {
|
import {
|
||||||
|
VBEN_ANT_PREVIEW_URL,
|
||||||
VBEN_DOC_URL,
|
VBEN_DOC_URL,
|
||||||
VBEN_GITHUB_URL,
|
VBEN_GITHUB_URL,
|
||||||
VBEN_LOGO_URL,
|
VBEN_LOGO_URL,
|
||||||
VBEN_NAIVE_PREVIEW_URL,
|
VBEN_NAIVE_PREVIEW_URL,
|
||||||
VBEN_PREVIEW_URL,
|
|
||||||
} from '@vben/constants';
|
} from '@vben/constants';
|
||||||
|
|
||||||
import { BasicLayout, IFrameView } from '#/layouts';
|
import { BasicLayout, IFrameView } from '#/layouts';
|
||||||
|
@ -38,8 +38,7 @@ const routes: RouteRecordRaw[] = [
|
||||||
component: IFrameView,
|
component: IFrameView,
|
||||||
meta: {
|
meta: {
|
||||||
icon: 'lucide:book-open-text',
|
icon: 'lucide:book-open-text',
|
||||||
iframeSrc: VBEN_DOC_URL,
|
link: VBEN_DOC_URL,
|
||||||
keepAlive: true,
|
|
||||||
title: $t('page.vben.document'),
|
title: $t('page.vben.document'),
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
@ -69,7 +68,7 @@ const routes: RouteRecordRaw[] = [
|
||||||
component: IFrameView,
|
component: IFrameView,
|
||||||
meta: {
|
meta: {
|
||||||
badgeType: 'dot',
|
badgeType: 'dot',
|
||||||
link: VBEN_PREVIEW_URL,
|
link: VBEN_ANT_PREVIEW_URL,
|
||||||
title: $t('page.vben.antdv'),
|
title: $t('page.vben.antdv'),
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
# 端口号
|
# 端口号
|
||||||
VITE_PORT=5777
|
VITE_PORT=5888
|
||||||
|
|
||||||
VITE_BASE=/
|
VITE_BASE=/
|
||||||
|
|
||||||
|
|
|
@ -58,7 +58,7 @@ function createRequestClient(baseURL: string) {
|
||||||
if (status >= 200 && status < 400 && code === 0) {
|
if (status >= 200 && status < 400 && code === 0) {
|
||||||
return data;
|
return data;
|
||||||
}
|
}
|
||||||
throw new Error(msg);
|
throw new Error(`Error ${status}: ${msg}`);
|
||||||
});
|
});
|
||||||
return client;
|
return client;
|
||||||
}
|
}
|
||||||
|
|
|
@ -17,7 +17,6 @@ const routes: RouteRecordRaw[] = [
|
||||||
children: [
|
children: [
|
||||||
{
|
{
|
||||||
meta: {
|
meta: {
|
||||||
icon: 'mdi:shield-key-outline',
|
|
||||||
title: $t('page.demos.naive'),
|
title: $t('page.demos.naive'),
|
||||||
},
|
},
|
||||||
name: 'NaiveDemos',
|
name: 'NaiveDemos',
|
||||||
|
|
|
@ -1,11 +1,11 @@
|
||||||
import type { RouteRecordRaw } from 'vue-router';
|
import type { RouteRecordRaw } from 'vue-router';
|
||||||
|
|
||||||
import {
|
import {
|
||||||
|
VBEN_ANT_PREVIEW_URL,
|
||||||
VBEN_DOC_URL,
|
VBEN_DOC_URL,
|
||||||
VBEN_ELE_PREVIEW_URL,
|
VBEN_ELE_PREVIEW_URL,
|
||||||
VBEN_GITHUB_URL,
|
VBEN_GITHUB_URL,
|
||||||
VBEN_LOGO_URL,
|
VBEN_LOGO_URL,
|
||||||
VBEN_PREVIEW_URL,
|
|
||||||
} from '@vben/constants';
|
} from '@vben/constants';
|
||||||
|
|
||||||
import { BasicLayout, IFrameView } from '#/layouts';
|
import { BasicLayout, IFrameView } from '#/layouts';
|
||||||
|
@ -38,8 +38,7 @@ const routes: RouteRecordRaw[] = [
|
||||||
component: IFrameView,
|
component: IFrameView,
|
||||||
meta: {
|
meta: {
|
||||||
icon: 'lucide:book-open-text',
|
icon: 'lucide:book-open-text',
|
||||||
iframeSrc: VBEN_DOC_URL,
|
link: VBEN_DOC_URL,
|
||||||
keepAlive: true,
|
|
||||||
title: $t('page.vben.document'),
|
title: $t('page.vben.document'),
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
@ -59,7 +58,7 @@ const routes: RouteRecordRaw[] = [
|
||||||
component: IFrameView,
|
component: IFrameView,
|
||||||
meta: {
|
meta: {
|
||||||
badgeType: 'dot',
|
badgeType: 'dot',
|
||||||
link: VBEN_PREVIEW_URL,
|
link: VBEN_ANT_PREVIEW_URL,
|
||||||
title: $t('page.vben.antdv'),
|
title: $t('page.vben.antdv'),
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
|
|
@ -162,7 +162,11 @@ function nav(): DefaultTheme.NavItem[] {
|
||||||
items: [
|
items: [
|
||||||
{
|
{
|
||||||
link: 'https://www.vben.pro',
|
link: 'https://www.vben.pro',
|
||||||
text: 'Ant Design Vue 版本(默认)',
|
text: '演示版本',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
link: 'https://ant.vben.pro',
|
||||||
|
text: 'Ant Design Vue 版本',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
link: 'https://naive.vben.pro',
|
link: 'https://naive.vben.pro',
|
||||||
|
@ -250,6 +254,7 @@ function sidebarGuide(): DefaultTheme.SidebarItem[] {
|
||||||
text: '为什么选择我们?',
|
text: '为什么选择我们?',
|
||||||
},
|
},
|
||||||
{ link: 'introduction/quick-start', text: '快速开始' },
|
{ link: 'introduction/quick-start', text: '快速开始' },
|
||||||
|
{ link: 'introduction/thin', text: '精简版本' },
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
|
@ -284,6 +289,7 @@ function sidebarGuide(): DefaultTheme.SidebarItem[] {
|
||||||
items: [
|
items: [
|
||||||
{ link: 'project/standard', text: '规范' },
|
{ link: 'project/standard', text: '规范' },
|
||||||
{ link: 'project/cli', text: 'CLI' },
|
{ link: 'project/cli', text: 'CLI' },
|
||||||
|
{ link: 'project/dir', text: '目录说明' },
|
||||||
{ link: 'project/test', text: '单元测试' },
|
{ link: 'project/test', text: '单元测试' },
|
||||||
{ link: 'project/tailwindcss', text: 'Tailwind CSS' },
|
{ link: 'project/tailwindcss', text: 'Tailwind CSS' },
|
||||||
{ link: 'project/changeset', text: 'Changeset' },
|
{ link: 'project/changeset', text: 'Changeset' },
|
||||||
|
|
|
@ -4,7 +4,10 @@
|
||||||
<div class="vp-doc vben-contributors">
|
<div class="vp-doc vben-contributors">
|
||||||
<p>Contributors</p>
|
<p>Contributors</p>
|
||||||
<a href="https://github.com/vbenjs/vue-vben-admin/graphs/contributors">
|
<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>
|
</a>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|
|
@ -78,6 +78,8 @@ npm 脚本是项目常见的配置,用于执行一些常见的任务,比如
|
||||||
"dev:docs": "pnpm -F @vben/docs run dev",
|
"dev:docs": "pnpm -F @vben/docs run dev",
|
||||||
// 启动web-ele应用
|
// 启动web-ele应用
|
||||||
"dev:ele": "pnpm -F @vben/web-ele run dev",
|
"dev:ele": "pnpm -F @vben/web-ele run dev",
|
||||||
|
// 启动演示应用
|
||||||
|
"dev:play": "pnpm -F @vben/playground run dev",
|
||||||
// 启动web-naive应用
|
// 启动web-naive应用
|
||||||
"dev:naive": "pnpm -F @vben/web-naive run dev",
|
"dev:naive": "pnpm -F @vben/web-naive run dev",
|
||||||
// 格式化代码
|
// 格式化代码
|
||||||
|
|
|
@ -221,7 +221,7 @@ function createRequestClient(baseURL: string) {
|
||||||
if (status >= 200 && status < 400 && code === 0) {
|
if (status >= 200 && status < 400 && code === 0) {
|
||||||
return data;
|
return data;
|
||||||
}
|
}
|
||||||
throw new Error(msg);
|
throw new Error(`Error ${status}: ${msg}`);
|
||||||
});
|
});
|
||||||
return client;
|
return client;
|
||||||
}
|
}
|
||||||
|
|
|
@ -198,7 +198,11 @@ async function loadDayjsLocale(lang: SupportedLanguagesType) {
|
||||||
locale = await import('dayjs/locale/en');
|
locale = await import('dayjs/locale/en');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
if (locale) {
|
||||||
dayjs.locale(locale);
|
dayjs.locale(locale);
|
||||||
|
} else {
|
||||||
|
console.error(`Failed to load dayjs locale for ${lang}`);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
|
|
|
@ -72,11 +72,24 @@ pnpm install
|
||||||
|
|
||||||
### 运行项目
|
### 运行项目
|
||||||
|
|
||||||
执行以下命令即可运行项目:
|
执行以下命运行项目:
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
# 启动项目
|
# 启动项目
|
||||||
pnpm dev
|
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` 查看项目。
|
现在,你可以在浏览器访问 `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",
|
"moduleResolution": "node",
|
||||||
"resolveJsonModule": true,
|
"resolveJsonModule": true,
|
||||||
"allowImportingTsExtensions": true,
|
|
||||||
|
|
||||||
"strict": true,
|
"strict": true,
|
||||||
"strictNullChecks": true,
|
"strictNullChecks": true,
|
||||||
|
|
|
@ -41,6 +41,7 @@
|
||||||
"dev:antd": "pnpm -F @vben/web-antd run dev",
|
"dev:antd": "pnpm -F @vben/web-antd run dev",
|
||||||
"dev:docs": "pnpm -F @vben/docs run dev",
|
"dev:docs": "pnpm -F @vben/docs run dev",
|
||||||
"dev:ele": "pnpm -F @vben/web-ele 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",
|
"dev:naive": "pnpm -F @vben/web-naive run dev",
|
||||||
"format": "vsh lint --format",
|
"format": "vsh lint --format",
|
||||||
"lint": "vsh lint",
|
"lint": "vsh lint",
|
||||||
|
|
|
@ -35,7 +35,9 @@
|
||||||
min-height: 100vh;
|
min-height: 100vh;
|
||||||
|
|
||||||
/* overflow: overlay; */
|
/* overflow: overlay; */
|
||||||
|
|
||||||
-webkit-font-smoothing: antialiased;
|
-webkit-font-smoothing: antialiased;
|
||||||
|
-moz-osx-font-smoothing: grayscale;
|
||||||
}
|
}
|
||||||
|
|
||||||
a,
|
a,
|
||||||
|
|
|
@ -8,7 +8,7 @@
|
||||||
|
|
||||||
/* 主体区域背景色 */
|
/* 主体区域背景色 */
|
||||||
--background-deep: 210 11.11% 96.47%;
|
--background-deep: 210 11.11% 96.47%;
|
||||||
--foreground: 210 6% 21%;
|
--foreground: 222 84% 5%;
|
||||||
|
|
||||||
/* Background color for <Card /> */
|
/* Background color for <Card /> */
|
||||||
--card: 0 0% 100%;
|
--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_ELE_PREVIEW_URL = 'https://ele.vben.pro';
|
||||||
|
|
||||||
export const VBEN_NAIVE_PREVIEW_URL = 'https://naive.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">
|
<h3 class="text-foreground text-2xl font-semibold leading-7">
|
||||||
{{ title }}
|
{{ title }}
|
||||||
</h3>
|
</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">
|
<VbenLink :href="VBEN_GITHUB_URL">
|
||||||
{{ name }}
|
{{ name }}
|
||||||
</VbenLink>
|
</VbenLink>
|
||||||
|
@ -139,7 +139,7 @@ const devDependenciesItems = Object.keys(devDependencies).map((key) => ({
|
||||||
<dt class="text-foreground text-sm font-medium leading-6">
|
<dt class="text-foreground text-sm font-medium leading-6">
|
||||||
{{ item.title }}
|
{{ item.title }}
|
||||||
</dt>
|
</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" />
|
<VbenRenderContent :content="item.content" />
|
||||||
</dd>
|
</dd>
|
||||||
</div>
|
</div>
|
||||||
|
@ -159,7 +159,7 @@ const devDependenciesItems = Object.keys(devDependencies).map((key) => ({
|
||||||
<dt class="text-foreground text-sm">
|
<dt class="text-foreground text-sm">
|
||||||
{{ item.title }}
|
{{ item.title }}
|
||||||
</dt>
|
</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" />
|
<VbenRenderContent :content="item.content" />
|
||||||
</dd>
|
</dd>
|
||||||
</div>
|
</div>
|
||||||
|
@ -178,7 +178,7 @@ const devDependenciesItems = Object.keys(devDependencies).map((key) => ({
|
||||||
<dt class="text-foreground text-sm">
|
<dt class="text-foreground text-sm">
|
||||||
{{ item.title }}
|
{{ item.title }}
|
||||||
</dt>
|
</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" />
|
<VbenRenderContent :content="item.content" />
|
||||||
</dd>
|
</dd>
|
||||||
</div>
|
</div>
|
||||||
|
|
|
@ -100,15 +100,11 @@ export function useTabbar() {
|
||||||
watch(
|
watch(
|
||||||
() => route.path,
|
() => route.path,
|
||||||
() => {
|
() => {
|
||||||
// 这里不能用route,用route时,vue-router会自动将父级meta进行合并
|
const meta = route.matched?.[route.matched.length - 1]?.meta;
|
||||||
const routes = router.getRoutes();
|
|
||||||
const currentRoute = routes.find((item) => item.path === route.path);
|
|
||||||
if (currentRoute) {
|
|
||||||
tabbarStore.addTab({
|
tabbarStore.addTab({
|
||||||
...route,
|
...route,
|
||||||
meta: currentRoute.meta,
|
meta: meta || route.meta,
|
||||||
} as unknown as RouteLocationNormalizedGeneric);
|
});
|
||||||
}
|
|
||||||
},
|
},
|
||||||
{ immediate: true },
|
{ 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