Revert "chore: delete useless workflows"

This commit is contained in:
傲慢或香橙 2024-02-20 00:27:09 +08:00 committed by GitHub
parent cc52955c6c
commit 2d44cddc50
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
120 changed files with 4540 additions and 7003 deletions

16
.eslintignore Normal file
View File

@ -0,0 +1,16 @@
*.sh
node_modules
*.md
*.woff
*.ttf
.vscode
.idea
dist
/public
/docs
.husky
.local
/bin
Dockerfile
components.d.ts
components.d.ts

78
.eslintrc.js Normal file
View File

@ -0,0 +1,78 @@
// @ts-check
const { defineConfig } = require('eslint-define-config');
module.exports = defineConfig({
root: true,
env: {
browser: true,
node: true,
es6: true,
},
parser: 'vue-eslint-parser',
parserOptions: {
parser: '@typescript-eslint/parser',
ecmaVersion: 2020,
sourceType: 'module',
jsxPragma: 'React',
ecmaFeatures: {
jsx: true,
},
},
extends: [
'plugin:vue/vue3-recommended',
'plugin:@typescript-eslint/recommended',
'prettier',
'plugin:prettier/recommended',
],
rules: {
'vue/script-setup-uses-vars': 'error',
'@typescript-eslint/ban-ts-ignore': 'off',
'@typescript-eslint/explicit-function-return-type': 'off',
'@typescript-eslint/no-explicit-any': 'off',
'@typescript-eslint/no-var-requires': 'off',
'@typescript-eslint/no-empty-function': 'off',
'vue/custom-event-name-casing': 'off',
'no-use-before-define': 'off',
'@typescript-eslint/no-use-before-define': 'off',
'@typescript-eslint/ban-ts-comment': 'off',
'@typescript-eslint/ban-types': 'off',
'@typescript-eslint/no-non-null-assertion': 'off',
'@typescript-eslint/explicit-module-boundary-types': 'off',
'@typescript-eslint/no-unused-vars': [
'error',
{
argsIgnorePattern: '^_',
varsIgnorePattern: '^_',
},
],
'no-unused-vars': [
'error',
{
argsIgnorePattern: '^_',
varsIgnorePattern: '^_',
},
],
'space-before-function-paren': 'off',
'vue/attributes-order': 'off',
'vue/one-component-per-file': 'off',
'vue/html-closing-bracket-newline': 'off',
'vue/max-attributes-per-line': 'off',
'vue/multiline-html-element-content-newline': 'off',
'vue/singleline-html-element-content-newline': 'off',
'vue/attribute-hyphenation': 'off',
'vue/require-default-prop': 'off',
'vue/multi-word-component-names': 'off',
'vue/html-self-closing': [
'error',
{
html: {
void: 'always',
normal: 'never',
component: 'always',
},
svg: 'always',
math: 'always',
},
],
},
});

31
.github/workflows/node.js.yml vendored Normal file
View File

@ -0,0 +1,31 @@
# This workflow will do a clean installation of node dependencies, cache/restore them, build the source code and run tests across different versions of node
# For more information see: https://docs.github.com/en/actions/automating-builds-and-tests/building-and-testing-nodejs
name: Node.js CI
on:
push:
branches: [ "main" ]
pull_request:
branches: [ "main" ]
jobs:
build:
runs-on: ubuntu-latest
strategy:
matrix:
node-version: [14.x, 16.x, 18.x]
# See supported Node.js release schedule at https://nodejs.org/en/about/releases/
steps:
- uses: actions/checkout@v3
- name: Use Node.js ${{ matrix.node-version }}
uses: actions/setup-node@v3
with:
node-version: ${{ matrix.node-version }}
cache: 'npm'
- run: npm ci
- run: npm run build --if-present
- run: npm test

4
.husky/commit-msg Normal file
View File

@ -0,0 +1,4 @@
#!/usr/bin/env sh
. "$(dirname -- "$0")/_/husky.sh"
npx --no-install commitlint --edit "$1"

4
.husky/pre-commit Normal file
View File

@ -0,0 +1,4 @@
#!/usr/bin/env sh
. "$(dirname -- "$0")/_/husky.sh"
npm run lint:lint-staged

9
.prettierignore Normal file
View File

@ -0,0 +1,9 @@
/dist/*
.local
.output.js
/node_modules/**
**/*.svg
**/*.sh
/public/*

3
.stylelintignore Normal file
View File

@ -0,0 +1,3 @@
/dist/*
/public/*
public/*

150
README.md
View File

@ -1,6 +1,7 @@
<div align="center"> <div align="center">
<a href="https://github.com/xiangshu233/vue3-vant4-mobile"> <a href="https://github.com/xiangshu233/vue3-vant4-mobile">
<img alt="Vue3Vant4MobileLogo" width="200" height="200" src="https://fastly.jsdelivr.net/gh/xiangshu233/blogAssets/2022/07/logo.svg"> <img alt="Vue3Vant4MobileLogo" width="200" height="200" src="https://fastly.jsdelivr.net/gh/xiangshu233/blogAssets/2022/07/logo.svg">
</a> </a>
</div><br><br> </div><br><br>
@ -16,44 +17,41 @@
## 介绍 ## 介绍
👋👋👋 Vue3 Vant4 Mobile 使用了最新的 `Vue3.2``Vite3``Vant4``Pinia2``TypeScript``UnoCSS` 等主流技术开发,集成 `Dark Mode`(暗黑)模式和系统主题色,并且持久化保存,集成 `Mock` 数据,顺便写了登录/注册/找回密码 页面(包括逻辑),只需替换你的 API 即可,另外页面均可以 `<keep-alive>`,随便写了个包含 `NavBar``TabBar` 的 Layout集成了 `Axios``useECharts``IconSvg` 👋👋👋 Vue3 Vant4 Mobile 使用了最新的 `Vue3.2``Vite3``Vant4``Pinia2``TypeScript``WindiCSS` 等主流技术开发,集成 `Dark Mode`(暗黑)模式和系统主题色,并且持久化保存,集成 `Mock` 数据,顺便写了个 登录/注册/找回密码 页面(包括逻辑),你只需要替换你的 API 即可,顺便写了个包含 `NavBar``TabBar` 的 Layout顺便集成了 `Axios``useECharts``IconSvg`,顺便集成了代码规范检查工具 `Eslint``Prettier``Stylelint`。顺便全页面均可以 `<keep-alive>`,顺便......好吧没有了。现在,你可以在此之上直接开发你的业务代码!希望你能喜欢。🥳🥳🥳
项目使用了 [antfu](https://github.com/antfu) 大佬的 [antfu/eslint-config](https://github.com/antfu/eslint-config) 作为代码规范检查工具,摆脱繁琐无聊的 Eslint 配置,配合 `cz-git``lint-staged``simple-git-hooks`可对暂存区代码提交校验,代码风格不合格可打断提交,保证多人协作开发时上游 Git 库的干净。
现在你可以在此之上直接开发你的业务代码!希望你能喜欢!
## 截图预览 ## 截图预览
<table> <table>
<tr> <tr>
<td><img src="https://fastly.jsdelivr.net/gh/xiangshu233/blogAssets/2022/10/%E5%BE%AE%E4%BF%A1%E6%88%AA%E5%9B%BE_20221022091917.png" width="400" alt="登录页面" /></td> <td><img src="https://fastly.jsdelivr.net/gh/xiangshu233/blogAssets/2022/10/%E5%BE%AE%E4%BF%A1%E6%88%AA%E5%9B%BE_20221022091917.png" width="400" alt="登录页面" /></td>
<td><img src="https://fastly.jsdelivr.net/gh/xiangshu233/blogAssets/2022/10/%E5%BE%AE%E4%BF%A1%E6%88%AA%E5%9B%BE_20221022092004.png" width="400" alt="主控台页(首页)" /></td> <td><img src="https://fastly.jsdelivr.net/gh/xiangshu233/blogAssets/2022/10/%E5%BE%AE%E4%BF%A1%E6%88%AA%E5%9B%BE_20221022092004.png" width="400" alt="主控台页(首页)" /></td>
</tr> </tr>
<tr> <tr>
<td><img src="https://fastly.jsdelivr.net/gh/xiangshu233/blogAssets/2022/10/%E5%BE%AE%E4%BF%A1%E6%88%AA%E5%9B%BE_20221022092015.png" width="400" alt="消息页(图标页)" /></td> <td><img src="https://fastly.jsdelivr.net/gh/xiangshu233/blogAssets/2022/10/%E5%BE%AE%E4%BF%A1%E6%88%AA%E5%9B%BE_20221022092015.png" width="400" alt="消息页(图标页)" /></td>
<td><img src="https://fastly.jsdelivr.net/gh/xiangshu233/blogAssets/2022/10/%E5%BE%AE%E4%BF%A1%E6%88%AA%E5%9B%BE_20221022092022.png" width="400" alt="我的(我的信息页面)" /></td> <td><img src="https://fastly.jsdelivr.net/gh/xiangshu233/blogAssets/2022/10/%E5%BE%AE%E4%BF%A1%E6%88%AA%E5%9B%BE_20221022092022.png" width="400" alt="我的(我的信息页面)" /></td>
</tr> </tr>
</table> </table>
<details> <details>
<summary>展开预览暗黑模式下的界面截图。</summary> <summary>展开预览暗黑模式下的界面截图。</summary>
<table> <table>
<tr> <tr>
<td><img src="https://fastly.jsdelivr.net/gh/xiangshu233/blogAssets/2022/10/%E5%BE%AE%E4%BF%A1%E6%88%AA%E5%9B%BE_20221022092052.png" width="400" alt="登录页面(暗黑模式)" /></td> <td><img src="https://fastly.jsdelivr.net/gh/xiangshu233/blogAssets/2022/10/%E5%BE%AE%E4%BF%A1%E6%88%AA%E5%9B%BE_20221022092052.png" width="400" alt="登录页面(暗黑模式)" /></td>
<td><img src="https://fastly.jsdelivr.net/gh/xiangshu233/blogAssets/2022/10/%E5%BE%AE%E4%BF%A1%E6%88%AA%E5%9B%BE_20221022092140.png" width="400" alt="主控台页(暗黑模式)" /></td> <td><img src="https://fastly.jsdelivr.net/gh/xiangshu233/blogAssets/2022/10/%E5%BE%AE%E4%BF%A1%E6%88%AA%E5%9B%BE_20221022092140.png" width="400" alt="主控台页(暗黑模式)" /></td>
</tr> </tr>
<tr> <tr>
<td><img src="https://fastly.jsdelivr.net/gh/xiangshu233/blogAssets/2022/10/%E5%BE%AE%E4%BF%A1%E6%88%AA%E5%9B%BE_20221022092224.png" width="400" alt="我的页面(暗黑模式)" /></td> <td><img src="https://fastly.jsdelivr.net/gh/xiangshu233/blogAssets/2022/10/%E5%BE%AE%E4%BF%A1%E6%88%AA%E5%9B%BE_20221022092224.png" width="400" alt="我的页面(暗黑模式)" /></td>
<td><img src="https://fastly.jsdelivr.net/gh/xiangshu233/blogAssets/2022/10/%E5%BE%AE%E4%BF%A1%E5%9B%BE%E7%89%87_20221023152559.png" width="400" alt="主题设置页面(暗黑模式)" /></td> <td><img src="https://fastly.jsdelivr.net/gh/xiangshu233/blogAssets/2022/10/%E5%BE%AE%E4%BF%A1%E5%9B%BE%E7%89%87_20221023152559.png" width="400" alt="主题设置页面(暗黑模式)" /></td>
</tr> </tr>
</table> </table>
</details> </details>
## 线上预览 ## 线上预览
预览链接:_[https://vvmobile.xiangshu233.cn/](https://vvmobile.xiangshu233.cn/)_ 预览链接:*[https://vvmobile.xiangshu233.cn/](https://vvmobile.xiangshu233.cn/)*
账号admin密码123456 账号admin密码123456
@ -77,32 +75,28 @@
- [xicons](https://www.xicons.org/#/) - 本项目推荐图标库,当然你也可以使用 `IconSVg` - [xicons](https://www.xicons.org/#/) - 本项目推荐图标库,当然你也可以使用 `IconSVg`
- [postcss-mobile-forever](https://github.com/wswmsword/postcss-mobile-forever) - 了解手机端 `px``viewport` 插件的作用 - [postcss-mobile-forever](https://github.com/wswmsword/postcss-mobile-forever) - 了解手机端 `px``viewport` 插件的作用
- [Lodash-es](https://www.lodashjs.com/) - `JS`高性能工具库 - [Lodash-es](https://www.lodashjs.com/) - `JS`高性能工具库
- [UnoCSS](https://unocss.dev/) - 原子化 `CSS`,熟悉 `UnoCSS` 基本使用 - [WindiCSS](https://cn.windicss.org/guide/) - 原子化 `CSS`,熟悉 `WindiCSS` 基本使用
- [Mock.js](https://github.com/nuysoft/Mock) - 了解 `Mockjs` 基本语法 - [Mock.js](https://github.com/nuysoft/Mock) - 了解 `Mockjs` 基本语法
- [ES6+](http://es6.ruanyifeng.com/) - 熟悉 `ES6` 基本语法 - [ES6+](http://es6.ruanyifeng.com/) - 熟悉 `ES6` 基本语法
## 环境准备 ## 环境准备
本地环境需要安装 [Pnpm](https://www.pnpm.cn/)、[Node.js](http://nodejs.org/) 和 [Git](https://git-scm.com/) 本地环境需要安装 [pnpm7.x](https://www.pnpm.cn/)、[Node.js](http://nodejs.org/) 和 [Git](https://git-scm.com/)
- 必须使用 [pnpm>=8.6.10](https://www.pnpm.cn/),否则依赖可能安装不上。 - 必须使用[pnpm7.x](https://www.pnpm.cn/),否则依赖可能安装不上。
- [Node.js](http://nodejs.org/) 版本要求`18.x`以上,且不能为`13.x`版本,这里推荐 ` ^20.9.0 || >=21.1.0` - [Node.js](http://nodejs.org/) 版本要求`12.x`以上,且不能为`13.x`版本,这里推荐 `15.x` 及以上
## VS Code 配套插件 ## VS Code 配套插件
如果你使用的 IDE 是 [VS Code](https://code.visualstudio.com/)(推荐)的话,可以安装以下工具来提高开发效率及代码格式化 如果你使用的 IDE 是 [VS Code](https://code.visualstudio.com/)(推荐)的话,可以安装以下工具来提高开发效率及代码格式化
- [UnoCSS](https://marketplace.visualstudio.com/items?itemName=antfu.unocss) - UnoCSS 提示插件 - [WindiCSS IntelliSense](https://marketplace.visualstudio.com/items?itemName=voorjaar.windicss-intellisense) - WindiCSS 提示插件
- [Volar](https://marketplace.visualstudio.com/items?itemName=Vue.volar) - Vue 开发必备 - [Volar](https://marketplace.visualstudio.com/items?itemName=Vue.volar) - Vue 开发必备
- [TypeScript Vue Plugin (Volar)](https://marketplace.visualstudio.com/items?itemName=Vue.vscode-typescript-vue-plugin) - 用于 TypeScript 服务器的 Vue 插件
- [ESLint](https://marketplace.visualstudio.com/items?itemName=dbaeumer.vscode-eslint) - 脚本代码检查 - [ESLint](https://marketplace.visualstudio.com/items?itemName=dbaeumer.vscode-eslint) - 脚本代码检查
- [Prettier](https://marketplace.visualstudio.com/items?itemName=esbenp.prettier-vscode) - 代码格式化
- [Stylelint](https://marketplace.visualstudio.com/items?itemName=stylelint.vscode-stylelint) - CSS 格式化
- [DotENV](https://marketplace.visualstudio.com/items?itemName=mikestead.dotenv) - `.env` 文件 高亮 - [DotENV](https://marketplace.visualstudio.com/items?itemName=mikestead.dotenv) - `.env` 文件 高亮
- [Error Lens](https://marketplace.visualstudio.com/items?itemName=usernamehw.errorlens) - 更好的错误定位 - [Error Lens](https://marketplace.visualstudio.com/items?itemName=usernamehw.errorlens) - 更好的错误定位
- [EditorConfig for VS Code](https://marketplace.visualstudio.com/items?itemName=EditorConfig.EditorConfig) - 不同 IDE 维护一致的编码样式
- [File Nesting Updater](https://marketplace.visualstudio.com/items?itemName=antfu.file-nesting) - 使用 VS Code 的文件嵌套功能使文件树更干净的配置
- [Pretty TypeScript Errors](https://marketplace.visualstudio.com/items?itemName=antfu.file-nesting) - 使 VSCode 中的 TypeScript 错误更漂亮、更易于理解
- [Todo Tree](https://marketplace.visualstudio.com/items?itemName=Gruntfuggly.todo-tree) - 在树视图中显示 TODO、FIXME 等注释标签
- [Trailing Spaces](https://marketplace.visualstudio.com/items?itemName=shardulm94.trailing-spaces) - 突出显示尾随空格并立即将其删除
## 使用 ## 使用
@ -121,82 +115,22 @@ pnpm dev
pnpm build pnpm build
``` ```
## Git 提交规范 ## Git 贡献提交规范
### 提交规范 - 参考 [vue](https://github.com/vuejs/vue/blob/dev/.github/COMMIT_CONVENTION.md) 规范 ([Angular](https://github.com/conventional-changelog/conventional-changelog/tree/master/packages/conventional-changelog-angular))
- `feat` 增加新功能
参考 [vue](https://github.com/vuejs/vue/blob/dev/.github/COMMIT_CONVENTION.md) 规范 ([Angular](https://github.com/conventional-changelog/conventional-changelog/tree/master/packages/conventional-changelog-angular)) - `fix` 修复问题/BUG
- `style` 代码风格相关无影响运行结果的
- `feat` 增加新功能 - `perf` 优化/性能提升
- `fix` 修复问题/BUG - `refactor` 重构
- `style` 代码风格相关无影响运行结果的 - `revert` 撤销修改
- `perf` 优化/性能提升 - `test` 测试相关
- `refactor` 重构 - `docs` 文档/注释
- `revert` 撤销修改 - `chore` 依赖更新/脚手架配置修改等
- `test` 测试相关 - `workflow` 工作流改进
- `docs` 文档/注释 - `ci` 持续集成
- `chore` 依赖更新/脚手架配置修改等 - `types` 类型定义文件更改
- `workflow` 工作流改进 - `wip` 开发中
- `ci` 持续集成
- `types` 类型定义文件更改
- `wip` 开发中
### 提交校验
关于前端工程化 **配置构建代码检查工作流** 不了解的可以看下面这两篇文章了解下
[前端工程化配置(上) 构建代码检查工作流](https://xiangshu233.cn/%E5%89%8D%E7%AB%AF%E5%B7%A5%E7%A8%8B%E5%8C%96%E9%85%8D%E7%BD%AE%EF%BC%88%E4%B8%8A%EF%BC%89%20%E6%9E%84%E5%BB%BA%E4%BB%A3%E7%A0%81%E6%A3%80%E6%9F%A5%E5%B7%A5%E4%BD%9C%E6%B5%81/)
[前端工程化配置(下) 规范仓库提交记录 commitlint + commitizen + cz-git + 配置](https://xiangshu233.cn/%E5%89%8D%E7%AB%AF%E5%B7%A5%E7%A8%8B%E5%8C%96%E9%85%8D%E7%BD%AE%EF%BC%88%E4%B8%8B%EF%BC%89%20%E8%A7%84%E8%8C%83%E4%BB%93%E5%BA%93%E6%8F%90%E4%BA%A4%E8%AE%B0%E5%BD%95/)
代码首次拉下来 `pnpm install` 后 需要执行以下命令来更新 `git hooks`
```shell
# Update ./git/hooks
npx simple-git-hooks
```
本项目提交规范校验使用 [simple-git-hooks](https://github.com/toplenboren/simple-git-hooks) 作为 git hooks使用 [cz-git](https://github.com/Zhengqbbb/cz-git) 作为 commitlint commitizen。
代码要想使用 commitlint 规范提交需要在更改的文件 `git add` 后,控制台执行 `cz` 命令开启 cz-git CLI
若想直接执行 `git commit` 需要满足上面提交规范才能通过校验,否则无法提交
simple-git-hooks 和 husky 都是用于管理 Git 钩子Git hooks的工具但它们有一些区别
1. simple-git-hooks:
- 简介: simple-git-hooks 是一个轻量级的工具用于管理和运行Git钩子。
- 特点:
- 提供了简单的配置方式来定义和运行 Gi 钩子。
- 适合于小型项目或对 Git 钩子需求不复杂的项目。
- 相对较少的功能和配置选项。
- 使用场景: 适用于简单的项目或对 Git 钩子管理需求不高的情况。
2. husky:
- 简介: husky 是一个功能强大的工具,用于管理 Git 钩子,并且在项目中被广泛使用。
- 特点:
- 提供了丰富的配置选项和灵活性,可以精细地控制 Git 钩子的行为。
- 支持在不同的 Git 钩子事件上运行自定义脚本。
- 可以与其他工具如linters、测试框架等集成实现更复杂的工作流。
- 使用场景: 适用于需要灵活配置和管理 Git 钩子的项目,尤其是大型或复杂的项目。
```json
// package.json
{
"simple-git-hooks": {
// 对暂存区执行 eslint --fix
"pre-commit": "pnpm lint-staged",
// 对提交信息进行校验
"commit-msg": "npx --no-install commitlint --edit $1"
},
"lint-staged": {
"*": "eslint --fix"
}
}
```
## 浏览器支持 ## 浏览器支持
@ -205,8 +139,8 @@ simple-git-hooks 和 husky 都是用于管理 Git 钩子Git hooks的工具
支持现代浏览器, 不支持 IE 支持现代浏览器, 不支持 IE
| [![ Edge](https://raw.githubusercontent.com/alrra/browser-logos/master/src/edge/edge_48x48.png)](http://godban.github.io/browsers-support-badges/) IE | [![ Edge](https://raw.githubusercontent.com/alrra/browser-logos/master/src/edge/edge_48x48.png)](http://godban.github.io/browsers-support-badges/) Edge | [![Firefox](https://raw.githubusercontent.com/alrra/browser-logos/master/src/firefox/firefox_48x48.png)](http://godban.github.io/browsers-support-badges/) Firefox | [![Chrome](https://raw.githubusercontent.com/alrra/browser-logos/master/src/chrome/chrome_48x48.png)](http://godban.github.io/browsers-support-badges/) Chrome | [![Safari](https://raw.githubusercontent.com/alrra/browser-logos/master/src/safari/safari_48x48.png)](http://godban.github.io/browsers-support-badges/) Safari | | [![ Edge](https://raw.githubusercontent.com/alrra/browser-logos/master/src/edge/edge_48x48.png)](http://godban.github.io/browsers-support-badges/) IE | [![ Edge](https://raw.githubusercontent.com/alrra/browser-logos/master/src/edge/edge_48x48.png)](http://godban.github.io/browsers-support-badges/) Edge | [![Firefox](https://raw.githubusercontent.com/alrra/browser-logos/master/src/firefox/firefox_48x48.png)](http://godban.github.io/browsers-support-badges/) Firefox | [![Chrome](https://raw.githubusercontent.com/alrra/browser-logos/master/src/chrome/chrome_48x48.png)](http://godban.github.io/browsers-support-badges/) Chrome | [![Safari](https://raw.githubusercontent.com/alrra/browser-logos/master/src/safari/safari_48x48.png)](http://godban.github.io/browsers-support-badges/) Safari |
| ----------------------------------------------------------------------------------------------------------------------------------------------------- | ------------------------------------------------------------------------------------------------------------------------------------------------------- | ------------------------------------------------------------------------------------------------------------------------------------------------------------------ | -------------------------------------------------------------------------------------------------------------------------------------------------------------- | -------------------------------------------------------------------------------------------------------------------------------------------------------------- | | --- | --- | --- | --- | --- |
| not support | last 2 versions | last 2 versions | last 2 versions | last 2 versions | | not support | last 2 versions | last 2 versions | last 2 versions | last 2 versions |
## 维护者 ## 维护者

View File

@ -1,6 +1,6 @@
/** /**
* The name of the configuration file entered in the production environment * The name of the configuration file entered in the production environment
*/ */
export const GLOB_CONFIG_FILE_NAME = 'app.config.js' export const GLOB_CONFIG_FILE_NAME = 'app.config.js';
export const OUTPUT_DIR = 'dist/vant-mobile' export const OUTPUT_DIR = 'dist/vant-mobile';

View File

@ -2,8 +2,8 @@
* Get the configuration file variable name * Get the configuration file variable name
* @param env * @param env
*/ */
export function getConfigFileName(env: Record<string, any>) { export const getConfigFileName = (env: Record<string, any>) => {
return `__PRODUCTION__${env.VITE_GLOB_APP_SHORT_NAME || '__APP'}__CONF__` return `__PRODUCTION__${env.VITE_GLOB_APP_SHORT_NAME || '__APP'}__CONF__`
.toUpperCase() .toUpperCase()
.replace(/\s/g, '') .replace(/\s/g, '');
} };

View File

@ -1,24 +1,24 @@
/** /**
* Generate additional configuration files when used for packaging. The file can be configured with some global variables, so that it can be changed directly externally without repackaging * Generate additional configuration files when used for packaging. The file can be configured with some global variables, so that it can be changed directly externally without repackaging
*/ */
import fs from 'fs-extra' import { GLOB_CONFIG_FILE_NAME, OUTPUT_DIR } from '../constant';
import colors from 'picocolors' import fs, { writeFileSync } from 'fs-extra';
import { GLOB_CONFIG_FILE_NAME, OUTPUT_DIR } from '../constant' import colors from 'picocolors';
import { getEnvConfig, getRootPath } from '../utils' import { getRootPath, getEnvConfig } from '../utils';
import { getConfigFileName } from '../getConfigFileName' import { getConfigFileName } from '../getConfigFileName';
import pkg from '../../package.json' import pkg from '../../package.json';
function createConfig( function createConfig(
{ {
configName, configName,
config, config,
configFileName = GLOB_CONFIG_FILE_NAME, configFileName = GLOB_CONFIG_FILE_NAME,
}: { configName: string, config: any, configFileName?: string } = { configName: '', config: {} }, }: { configName: string; config: any; configFileName?: string } = { configName: '', config: {} }
) { ) {
try { try {
const windowConf = `window.${configName}` const windowConf = `window.${configName}`;
// Ensure that the variable will not be modified // Ensure that the variable will not be modified
const configStr = `${windowConf}=${JSON.stringify(config)}; const configStr = `${windowConf}=${JSON.stringify(config)};
Object.freeze(${windowConf}); Object.freeze(${windowConf});
@ -26,20 +26,19 @@ function createConfig(
configurable: false, configurable: false,
writable: false, writable: false,
}); });
`.replace(/\s/g, '') `.replace(/\s/g, '');
fs.mkdirp(getRootPath(OUTPUT_DIR)) fs.mkdirp(getRootPath(OUTPUT_DIR));
fs.writeFileSync(getRootPath(`${OUTPUT_DIR}/${configFileName}`), configStr) writeFileSync(getRootPath(`${OUTPUT_DIR}/${configFileName}`), configStr);
console.log(`${colors.cyan(`✨ [${pkg.name}]`)} - configuration file is build successfully:`) console.log(colors.cyan(`✨ [${pkg.name}]`) + ` - configuration file is build successfully:`);
console.log(`${colors.gray(`${OUTPUT_DIR}/${colors.green(configFileName)}`)}\n`) console.log(colors.gray(OUTPUT_DIR + '/' + colors.green(configFileName)) + '\n');
} } catch (error) {
catch (error) { console.log(colors.red('configuration file configuration file failed to package:\n' + error));
console.log(colors.red(`configuration file configuration file failed to package:\n${error}`))
} }
} }
export function runBuildConfig() { export function runBuildConfig() {
const config = getEnvConfig() const config = getEnvConfig();
const configFileName = getConfigFileName(config) const configFileName = getConfigFileName(config);
createConfig({ config, configName: configFileName }) createConfig({ config, configName: configFileName });
} }

View File

@ -1,24 +1,23 @@
// #!/usr/bin/env node // #!/usr/bin/env node
import colors from 'picocolors' import { runBuildConfig } from './buildConf';
import colors from 'picocolors';
import pkg from '../../package.json' import pkg from '../../package.json';
import { runBuildConfig } from './buildConf'
export async function runBuild() { export const runBuild = async () => {
try { try {
const argvList = process.argv.splice(2) const argvList = process.argv.splice(2);
// Generate configuration file // Generate configuration file
if (!argvList.includes('disabled-config')) { if (!argvList.includes('disabled-config')) {
await runBuildConfig() await runBuildConfig();
} }
console.log(`${colors.cyan(`[${pkg.name}]`)} - build successfully!`) console.log(`${colors.cyan(`[${pkg.name}]`)}` + ' - build successfully!');
} catch (error) {
console.log(colors.red('vite build error:\n' + error));
process.exit(1);
} }
catch (error) { };
console.log(colors.red(`vite build error:\n${error}`)) runBuild();
process.exit(1)
}
}
runBuild()

View File

@ -1,45 +1,44 @@
import fs from 'node:fs' import fs from 'fs';
import path from 'node:path' import path from 'path';
import dotenv from 'dotenv' import dotenv from 'dotenv';
export function isDevFn(mode: string): boolean { export function isDevFn(mode: string): boolean {
return mode === 'development' return mode === 'development';
} }
export function isProdFn(mode: string): boolean { export function isProdFn(mode: string): boolean {
return mode === 'production' return mode === 'production';
} }
/** /**
* Whether to generate package preview * Whether to generate package preview
*/ */
export function isReportMode(): boolean { export function isReportMode(): boolean {
return process.env.REPORT === 'true' return process.env.REPORT === 'true';
} }
// Read all environment variable configuration files to process.env // Read all environment variable configuration files to process.env
// 读取并处理所有环境变量配置文件 .env // 读取并处理所有环境变量配置文件 .env
export function wrapperEnv(envConf: Recordable): ViteEnv { export function wrapperEnv(envConf: Recordable): ViteEnv {
const ret: any = {} const ret: any = {};
for (const envName of Object.keys(envConf)) { for (const envName of Object.keys(envConf)) {
// 去除空格 // 去除空格
let realName = envConf[envName].replace(/\\n/g, '\n') let realName = envConf[envName].replace(/\\n/g, '\n');
realName = realName === 'true' ? true : realName === 'false' ? false : realName realName = realName === 'true' ? true : realName === 'false' ? false : realName;
if (envName === 'VITE_PORT') { if (envName === 'VITE_PORT') {
realName = Number(realName) realName = Number(realName);
} }
if (envName === 'VITE_PROXY') { if (envName === 'VITE_PROXY') {
try { try {
realName = JSON.parse(realName) realName = JSON.parse(realName);
} } catch (error) {}
catch (error) {}
} }
ret[envName] = realName ret[envName] = realName;
process.env[envName] = realName process.env[envName] = realName;
} }
return ret return ret;
} }
/** /**
@ -48,22 +47,21 @@ export function wrapperEnv(envConf: Recordable): ViteEnv {
* @param confFiles ext * @param confFiles ext
*/ */
export function getEnvConfig(match = 'VITE_GLOB_', confFiles = ['.env', '.env.production']) { export function getEnvConfig(match = 'VITE_GLOB_', confFiles = ['.env', '.env.production']) {
let envConfig = {} let envConfig = {};
confFiles.forEach((item) => { confFiles.forEach((item) => {
try { try {
const env = dotenv.parse(fs.readFileSync(path.resolve(process.cwd(), item))) const env = dotenv.parse(fs.readFileSync(path.resolve(process.cwd(), item)));
envConfig = { ...envConfig, ...env } envConfig = { ...envConfig, ...env };
} } catch (error) {}
catch (error) {} });
})
Object.keys(envConfig).forEach((key) => { Object.keys(envConfig).forEach((key) => {
const reg = new RegExp(`^(${match})`) const reg = new RegExp(`^(${match})`);
if (!reg.test(key)) { if (!reg.test(key)) {
Reflect.deleteProperty(envConfig, key) Reflect.deleteProperty(envConfig, key);
} }
}) });
return envConfig return envConfig;
} }
/** /**
@ -71,5 +69,5 @@ export function getEnvConfig(match = 'VITE_GLOB_', confFiles = ['.env', '.env.pr
* @param dir file path * @param dir file path
*/ */
export function getRootPath(...dir: string[]) { export function getRootPath(...dir: string[]) {
return path.resolve(process.cwd(), ...dir) return path.resolve(process.cwd(), ...dir);
} }

View File

@ -2,25 +2,25 @@
* Used to package and output gzip. Note that this does not work properly in Vite, the specific reason is still being investigated * Used to package and output gzip. Note that this does not work properly in Vite, the specific reason is still being investigated
* https://github.com/anncwb/vite-plugin-compression * https://github.com/anncwb/vite-plugin-compression
*/ */
import type { PluginOption } from 'vite' import type { PluginOption } from 'vite';
import compressPlugin from 'vite-plugin-compression' import compressPlugin from 'vite-plugin-compression';
export function configCompressPlugin( export function configCompressPlugin(
compress: 'gzip' | 'brotli' | 'none', compress: 'gzip' | 'brotli' | 'none',
deleteOriginFile = false, deleteOriginFile = false
): PluginOption | PluginOption[] { ): PluginOption | PluginOption[] {
const compressList = compress.split(',') const compressList = compress.split(',');
const plugins: PluginOption[] = [] const plugins: PluginOption[] = [];
if (compressList.includes('gzip')) { if (compressList.includes('gzip')) {
plugins.push( plugins.push(
compressPlugin({ compressPlugin({
ext: '.gz', ext: '.gz',
deleteOriginFile, deleteOriginFile,
}), })
) );
} }
if (compressList.includes('brotli')) { if (compressList.includes('brotli')) {
plugins.push( plugins.push(
@ -28,8 +28,8 @@ export function configCompressPlugin(
ext: '.br', ext: '.br',
algorithm: 'brotliCompress', algorithm: 'brotliCompress',
deleteOriginFile, deleteOriginFile,
}), })
) );
} }
return plugins return plugins;
} }

View File

@ -2,19 +2,19 @@
* Plugin to minimize and use ejs template syntax in index.html. * Plugin to minimize and use ejs template syntax in index.html.
* https://github.com/anncwb/vite-plugin-html * https://github.com/anncwb/vite-plugin-html
*/ */
import type { PluginOption } from 'vite' import type { PluginOption } from 'vite';
import { createHtmlPlugin } from 'vite-plugin-html' import { createHtmlPlugin } from 'vite-plugin-html';
import pkg from '../../../package.json' import pkg from '../../../package.json';
import { GLOB_CONFIG_FILE_NAME } from '../../constant' import { GLOB_CONFIG_FILE_NAME } from '../../constant';
export function configHtmlPlugin(env: ViteEnv, isBuild: boolean) { export function configHtmlPlugin(env: ViteEnv, isBuild: boolean) {
const { VITE_GLOB_APP_TITLE, VITE_PUBLIC_PATH } = env const { VITE_GLOB_APP_TITLE, VITE_PUBLIC_PATH } = env;
const path = VITE_PUBLIC_PATH.endsWith('/') ? VITE_PUBLIC_PATH : `${VITE_PUBLIC_PATH}/` const path = VITE_PUBLIC_PATH.endsWith('/') ? VITE_PUBLIC_PATH : `${VITE_PUBLIC_PATH}/`;
const getAppConfigSrc = () => { const getAppConfigSrc = () => {
return `${path || '/'}${GLOB_CONFIG_FILE_NAME}?v=${pkg.version}-${new Date().getTime()}` return `${path || '/'}${GLOB_CONFIG_FILE_NAME}?v=${pkg.version}-${new Date().getTime()}`;
} };
// 当执行 yarn build 构建项目之后,会自动生成 _app.config.js 文件并插入 index.html // 当执行 yarn build 构建项目之后,会自动生成 _app.config.js 文件并插入 index.html
// _app.config.js 用于项目在打包后,需要动态修改配置的需求,如接口地址 // _app.config.js 用于项目在打包后,需要动态修改配置的需求,如接口地址
@ -41,6 +41,6 @@ export function configHtmlPlugin(env: ViteEnv, isBuild: boolean) {
] ]
: [], : [],
}, },
}) });
return htmlPlugin return htmlPlugin;
} }

View File

@ -1,14 +1,16 @@
import type { PluginOption } from 'vite' import type { PluginOption } from 'vite';
import Components from 'unplugin-vue-components/vite' import Components from 'unplugin-vue-components/vite';
import { VantResolver } from 'unplugin-vue-components/resolvers' import { VantResolver } from 'unplugin-vue-components/resolvers';
import vue from '@vitejs/plugin-vue'
import UnoCSS from 'unocss/vite'
import { configHtmlPlugin } from './html' import vue from '@vitejs/plugin-vue';
import { configMockPlugin } from './mock' import vueSetupExtend from 'vite-plugin-vue-setup-extend';
import { configCompressPlugin } from './compress' import WindiCSS from 'vite-plugin-windicss';
import { configVisualizerConfig } from './visualizer'
import { configSvgIconsPlugin } from './svgSprite' import { configHtmlPlugin } from './html';
import { configMockPlugin } from './mock';
import { configCompressPlugin } from './compress';
import { configVisualizerConfig } from './visualizer';
import { configSvgIconsPlugin } from './svgSprite';
/** /**
* vite * vite
@ -22,41 +24,43 @@ export function createVitePlugins(viteEnv: ViteEnv, isBuild: boolean, prodMock:
// 如果你需要多种形式,你可以用','来分隔 // 如果你需要多种形式,你可以用','来分隔
// VITE_BUILD_COMPRESS_DELETE_ORIGIN_FILE 打包使用压缩时是否删除原始文件,默认为 false // VITE_BUILD_COMPRESS_DELETE_ORIGIN_FILE 打包使用压缩时是否删除原始文件,默认为 false
const { VITE_USE_MOCK, VITE_BUILD_COMPRESS, VITE_BUILD_COMPRESS_DELETE_ORIGIN_FILE } = viteEnv const { VITE_USE_MOCK, VITE_BUILD_COMPRESS, VITE_BUILD_COMPRESS_DELETE_ORIGIN_FILE } = viteEnv;
const vitePlugins: (PluginOption | PluginOption[])[] = [ const vitePlugins: (PluginOption | PluginOption[])[] = [
// have to // have to
vue(), vue(),
// support name https://github.com/vbenjs/vite-plugin-vue-setup-extend
vueSetupExtend(),
// 按需引入VantUi且自动创建组件声明 // 按需引入VantUi且自动创建组件声明
Components({ Components({
dts: true, dts: true,
resolvers: [VantResolver()], resolvers: [VantResolver()],
types: [], types: [],
}), }),
] ];
// UnoCSS // vite-plugin-windicss
vitePlugins.push(UnoCSS()) vitePlugins.push(WindiCSS());
// 加载 html 插件 vite-plugin-html // 加载 html 插件 vite-plugin-html
vitePlugins.push(configHtmlPlugin(viteEnv, isBuild)) vitePlugins.push(configHtmlPlugin(viteEnv, isBuild));
// rollup-plugin-visualizer // rollup-plugin-visualizer
vitePlugins.push(configVisualizerConfig()) vitePlugins.push(configVisualizerConfig());
// vite-plugin-mock // vite-plugin-mock
VITE_USE_MOCK && vitePlugins.push(configMockPlugin(isBuild, prodMock)) VITE_USE_MOCK && vitePlugins.push(configMockPlugin(isBuild, prodMock));
// vite-plugin-svg-icons // vite-plugin-svg-icons
vitePlugins.push(configSvgIconsPlugin(isBuild)) vitePlugins.push(configSvgIconsPlugin(isBuild));
if (isBuild) { if (isBuild) {
// rollup-plugin-gzip // rollup-plugin-gzip
// 加载 gzip 打包 // 加载 gzip 打包
vitePlugins.push( vitePlugins.push(
configCompressPlugin(VITE_BUILD_COMPRESS, VITE_BUILD_COMPRESS_DELETE_ORIGIN_FILE), configCompressPlugin(VITE_BUILD_COMPRESS, VITE_BUILD_COMPRESS_DELETE_ORIGIN_FILE)
) );
} }
return vitePlugins return vitePlugins;
} }

View File

@ -2,7 +2,7 @@
* Mock plugin for development and production. * Mock plugin for development and production.
* https://github.com/anncwb/vite-plugin-mock * https://github.com/anncwb/vite-plugin-mock
*/ */
import { viteMockServe } from 'vite-plugin-mock' import { viteMockServe } from 'vite-plugin-mock';
export function configMockPlugin(isBuild: boolean, prodMock: boolean) { export function configMockPlugin(isBuild: boolean, prodMock: boolean) {
return viteMockServe({ return viteMockServe({
@ -15,5 +15,5 @@ export function configMockPlugin(isBuild: boolean, prodMock: boolean) {
setupProdMockServer(); setupProdMockServer();
`, `,
}) });
} }

View File

@ -3,8 +3,8 @@
* https://github.com/anncwb/vite-plugin-svg-icons * https://github.com/anncwb/vite-plugin-svg-icons
*/ */
import path from 'node:path' import { createSvgIconsPlugin } from 'vite-plugin-svg-icons';
import { createSvgIconsPlugin } from 'vite-plugin-svg-icons' import path from 'path';
export function configSvgIconsPlugin(isBuild: boolean) { export function configSvgIconsPlugin(isBuild: boolean) {
// 指定需要缓存的图标文件夹 // 指定需要缓存的图标文件夹
@ -14,6 +14,6 @@ export function configSvgIconsPlugin(isBuild: boolean) {
svgoOptions: isBuild, svgoOptions: isBuild,
// 指定symbolId格式 // 指定symbolId格式
symbolId: 'icon-[dir]-[name]', symbolId: 'icon-[dir]-[name]',
}) });
return svgIconsPlugin return svgIconsPlugin;
} }

View File

@ -1,9 +1,9 @@
/** /**
* Package file volume analysis * Package file volume analysis
*/ */
import visualizer from 'rollup-plugin-visualizer' import visualizer from 'rollup-plugin-visualizer';
import type { PluginOption } from 'vite' import type { PluginOption } from 'vite';
import { isReportMode } from '../../utils' import { isReportMode } from '../../utils';
export function configVisualizerConfig() { export function configVisualizerConfig() {
if (isReportMode()) { if (isReportMode()) {
@ -12,7 +12,7 @@ export function configVisualizerConfig() {
open: true, open: true,
gzipSize: true, gzipSize: true,
brotliSize: true, brotliSize: true,
}) as PluginOption }) as PluginOption;
} }
return [] return [];
} }

View File

@ -1,38 +1,38 @@
/** /**
* Used to parse the .env.development proxy configuration * Used to parse the .env.development proxy configuration
*/ */
import type { ProxyOptions } from 'vite' import type { ProxyOptions } from 'vite';
type ProxyItem = [string, string] type ProxyItem = [string, string];
type ProxyList = ProxyItem[] type ProxyList = ProxyItem[];
type ProxyTargetList = Record<string, ProxyOptions & { rewrite: (path: string) => string }> type ProxyTargetList = Record<string, ProxyOptions & { rewrite: (path: string) => string }>;
const httpsRE = /^https:\/\// const httpsRE = /^https:\/\//;
/** /**
* Generate proxy * Generate proxy
* @param list * @param list
*/ */
export function createProxy(list: ProxyList = []) { export function createProxy(list: ProxyList = []) {
const ret: ProxyTargetList = {} const ret: ProxyTargetList = {};
for (const [prefix, target] of list) { for (const [prefix, target] of list) {
const isHttps = httpsRE.test(target) const isHttps = httpsRE.test(target);
// https://github.com/http-party/node-http-proxy#options // https://github.com/http-party/node-http-proxy#options
ret[prefix] = { ret[prefix] = {
target, target: target,
changeOrigin: true, changeOrigin: true,
ws: true, ws: true,
rewrite: path => path.replace(new RegExp(`^${prefix}`), ''), rewrite: (path) => path.replace(new RegExp(`^${prefix}`), ''),
// https is require secure=false // https is require secure=false
// 如果您secure="true"只允许来自 HTTPS 的请求则secure="false"意味着允许来自 HTTP 和 HTTPS 的请求。 // 如果您secure="true"只允许来自 HTTPS 的请求则secure="false"意味着允许来自 HTTP 和 HTTPS 的请求。
...(isHttps ? { secure: false } : {}), ...(isHttps ? { secure: false } : {}),
} };
} }
return ret return ret;
// ret // ret
// { // {

View File

@ -1,26 +1,26 @@
// commitlint.config.js // commitlint.config.js
const fs = require('node:fs') const fs = require('fs');
const path = require('node:path') const path = require('path');
const { execSync } = require('node:child_process') const { execSync } = require('child_process');
const scopes = fs const scopes = fs
.readdirSync(path.resolve(__dirname, 'src'), { withFileTypes: true }) .readdirSync(path.resolve(__dirname, 'src'), { withFileTypes: true })
.filter(dirent => dirent.isDirectory()) .filter((dirent) => dirent.isDirectory())
.map(dirent => dirent.name.replace(/s$/, '')) .map((dirent) => dirent.name.replace(/s$/, ''));
// precomputed scope // precomputed scope
const scopeComplete = execSync('git status --porcelain || true') const scopeComplete = execSync('git status --porcelain || true')
.toString() .toString()
.trim() .trim()
.split('\n') .split('\n')
.find(r => ~r.indexOf('M src')) .find((r) => ~r.indexOf('M src'))
?.replace(/(\/)/g, '%%') ?.replace(/(\/)/g, '%%')
?.match(/src%%((\w|-)*)/)?.[1] ?.match(/src%%((\w|-)*)/)?.[1]
?.replace(/s$/, '') ?.replace(/s$/, '');
/** @type {import('cz-git').UserConfig} */ /** @type {import('cz-git').UserConfig} */
module.exports = { module.exports = {
ignores: [commit => commit.includes('init')], ignores: [(commit) => commit.includes('init')],
extends: ['@commitlint/config-conventional'], extends: ['@commitlint/config-conventional'],
rules: { rules: {
'body-leading-blank': [2, 'always'], 'body-leading-blank': [2, 'always'],
@ -144,12 +144,12 @@ module.exports = {
emptyIssuePrefixsAlias: 'skip', emptyIssuePrefixsAlias: 'skip',
customIssuePrefixsAlias: 'custom', customIssuePrefixsAlias: 'custom',
confirmColorize: true, confirmColorize: true,
maxHeaderLength: Number.POSITIVE_INFINITY, maxHeaderLength: Infinity,
maxSubjectLength: Number.POSITIVE_INFINITY, maxSubjectLength: Infinity,
minSubjectLength: 0, minSubjectLength: 0,
scopeOverrides: undefined, scopeOverrides: undefined,
defaultBody: '', defaultBody: '',
defaultIssues: '', defaultIssues: '',
defaultSubject: '', defaultSubject: '',
}, },
} };

View File

@ -1,61 +0,0 @@
// eslint.config.js
import antfu from '@antfu/eslint-config'
export default antfu({
unocss: true,
stylistic: {
indent: 2, // 4, or 'tab'
quotes: 'single', // or 'double'
},
// 使用外部格式化程序来格式化 ESLint 无法处理的文件( .css 、 .html 等)
formatters: {
css: true,
html: true,
markdown: 'prettier',
},
// https://alloyteam.github.io/eslint-config-alloy/?language=zh-CN&rule=base
// https://eslint.vuejs.org/rules/
rules: {
'no-console': 'off',
// 强制组件顶级元素的顺序
'vue/block-order': [
'error',
{
order: ['template', 'script', 'style'],
},
],
'max-params': ['error', 4],
// 代码块嵌套的深度禁止超过 4 层
'max-depth': ['error', 4],
// 回调函数嵌套禁止超过 3 层,多了请用 async await 替代
'max-nested-callbacks': ['error', 4],
// 禁止使用 Array 构造函数时传入的参数超过一个
// 参数为多个时表示创建一个指定内容的数组,此时可以用数组字面量实现,不必使用构造函数
'no-array-constructor': 'error',
// 禁止 if else 的条件判断中出现重复的条件
'no-dupe-else-if': 'error',
// 禁止出现空代码块,允许 catch 为空代码块
'no-empty': [
'error',
{
allowEmptyCatch: true,
},
],
// 禁止出现没必要的字符串连接
'no-useless-concat': 'error',
// 禁止使用 var
'no-var': 'error',
// 禁止变量申明时用逗号一次申明多个
'one-var': ['error', 'never'],
// 必须使用 ... 而不是 Object.assign除非 Object.assign 的第一个参数是一个变量
'prefer-object-spread': 'error',
// 回调函数必须使用箭头函数
'prefer-arrow-callback': 'error',
// "stroustrup":强制一致的大括号风格,左括号必须与控制语句在同一行开始,右括号必须独占一行。
'brace-style': ['error', 'stroustrup'],
// 强制使用 node 全局变量 process 而不是 require("process") 。
'node/prefer-global/process': 'off',
// 对所有控制语句强制执行一致的大括号样式只有一行的时候eslint默认是不需要大括号的这样会降低代码清晰度
'curly': ['error', 'all'],
},
})

View File

@ -1,26 +1,27 @@
<!doctype html> <!DOCTYPE html>
<html lang="zh-cmn-Hans" id="htmlRoot" class> <html lang="zh-cmn-Hans" id="htmlRoot" class>
<head> <head>
<meta charset="UTF-8" /> <meta charset="UTF-8" />
<link rel="icon" type="image/svg+xml" href="/logo.svg" /> <link rel="icon" type="image/svg+xml" href="/logo.svg" />
<meta name="viewport" content="width=device-width, initial-scale=1.0" /> <meta name="viewport" content="width=device-width, initial-scale=1.0" />
<title><%= title %></title> <title>
<%= title %>
</title>
</head> </head>
<body> <body>
<div id="app"> <div id="app">
<script> <script>
;(() => { (() => {
let htmlRoot = document.getElementById('htmlRoot') let htmlRoot = document.getElementById('htmlRoot');
const appDesignSetting = window.localStorage.getItem('DESIGN-SETTING') const appDesignSetting = window.localStorage.getItem('DESIGN-SETTING')
let darkMode = let darkMode = appDesignSetting && JSON.parse(appDesignSetting).darkMode
appDesignSetting && JSON.parse(appDesignSetting).darkMode
if (htmlRoot && darkMode) { if (htmlRoot && darkMode) {
htmlRoot.setAttribute('data-theme', darkMode) htmlRoot.setAttribute('data-theme', darkMode);
darkMode = htmlRoot = null darkMode = htmlRoot = null;
} else { } else {
htmlRoot.setAttribute('data-theme', 'light') htmlRoot.setAttribute('data-theme', 'light');
} }
})() })();
</script> </script>
<style> <style>
body { body {
@ -37,14 +38,14 @@
align-items: center; align-items: center;
flex-direction: column; flex-direction: column;
} }
.first-loading-wrap > h1 { .first-loading-wrap>h1 {
font-size: 128px; font-size: 128px
} }
.first-loading-wrap .loading-wrap { .first-loading-wrap .loading-wrap {
padding: 98px; padding: 98px;
display: flex; display: flex;
justify-content: center; justify-content: center;
align-items: center; align-items: center
} }
.dot { .dot {
animation: antRotate 1.2s infinite linear; animation: antRotate 1.2s infinite linear;
@ -54,7 +55,7 @@
font-size: 52px; font-size: 52px;
width: 52px; width: 52px;
height: 52px; height: 52px;
box-sizing: border-box; box-sizing: border-box
} }
.dot i { .dot i {
width: 24px; width: 24px;
@ -63,53 +64,53 @@
display: block; display: block;
background-color: #1890ff; background-color: #1890ff;
border-radius: 100%; border-radius: 100%;
transform: scale(0.75); transform: scale(.75);
transform-origin: 50% 50%; transform-origin: 50% 50%;
opacity: 0.3; opacity: .3;
animation: antSpinMove 1s infinite linear alternate; animation: antSpinMove 1s infinite linear alternate
} }
.dot i:nth-child(1) { .dot i:nth-child(1) {
top: 0; top: 0;
left: 0; left: 0
} }
.dot i:nth-child(2) { .dot i:nth-child(2) {
top: 0; top: 0;
right: 0; right: 0;
-webkit-animation-delay: 0.4s; -webkit-animation-delay: .4s;
animation-delay: 0.4s; animation-delay: .4s
} }
.dot i:nth-child(3) { .dot i:nth-child(3) {
right: 0; right: 0;
bottom: 0; bottom: 0;
-webkit-animation-delay: 0.8s; -webkit-animation-delay: .8s;
animation-delay: 0.8s; animation-delay: .8s
} }
.dot i:nth-child(4) { .dot i:nth-child(4) {
bottom: 0; bottom: 0;
left: 0; left: 0;
-webkit-animation-delay: 1.2s; -webkit-animation-delay: 1.2s;
animation-delay: 1.2s; animation-delay: 1.2s
} }
@keyframes antRotate { @keyframes antRotate {
to { to {
-webkit-transform: rotate(405deg); -webkit-transform: rotate(405deg);
transform: rotate(405deg); transform: rotate(405deg)
} }
} }
@-webkit-keyframes antRotate { @-webkit-keyframes antRotate {
to { to {
-webkit-transform: rotate(405deg); -webkit-transform: rotate(405deg);
transform: rotate(405deg); transform: rotate(405deg)
} }
} }
@keyframes antSpinMove { @keyframes antSpinMove {
to { to {
opacity: 1; opacity: 1
} }
} }
@-webkit-keyframes antSpinMove { @-webkit-keyframes antSpinMove {
to { to {
opacity: 1; opacity: 1
} }
} }
</style> </style>

View File

@ -1,18 +1,18 @@
import { createProdMockServer } from 'vite-plugin-mock/es/createProdMockServer' import { createProdMockServer } from 'vite-plugin-mock/es/createProdMockServer';
const modules = import.meta.glob('./**/*.ts', { eager: true }) as any const modules = import.meta.glob('./**/*.ts', { eager: true }) as any;
const mockModules: any[] = [] const mockModules: any[] = [];
Object.keys(modules).forEach((key) => { Object.keys(modules).forEach((key) => {
if (key.includes('/_')) { if (key.includes('/_')) {
return return;
} }
mockModules.push(...modules[key].default) mockModules.push(...modules[key].default);
}) });
/** /**
* Used in a production environment. Need to manually import all modules * Used in a production environment. Need to manually import all modules
*/ */
export function setupProdMockServer() { export function setupProdMockServer() {
createProdMockServer(mockModules) createProdMockServer(mockModules);
} }

View File

@ -1,5 +1,5 @@
import Mock from 'mockjs' import Mock from 'mockjs';
import { ResultEnum } from '@/enums/httpEnum' import { ResultEnum } from '@/enums/httpEnum';
export function resultSuccess<T = Recordable>(result: T, { message = 'ok' } = {}) { export function resultSuccess<T = Recordable>(result: T, { message = 'ok' } = {}) {
return Mock.mock({ return Mock.mock({
@ -7,16 +7,16 @@ export function resultSuccess<T = Recordable>(result: T, { message = 'ok' } = {}
result, result,
message, message,
type: 'success', type: 'success',
}) });
} }
export function resultPageSuccess<T = any>( export function resultPageSuccess<T = any>(
page: number, page: number,
pageSize: number, pageSize: number,
list: T[], list: T[],
{ message = 'ok' } = {}, { message = 'ok' } = {}
) { ) {
const pageData = pagination(page, pageSize, list) const pageData = pagination(page, pageSize, list);
return { return {
...resultSuccess({ ...resultSuccess({
@ -26,46 +26,46 @@ export function resultPageSuccess<T = any>(
list: pageData, list: pageData,
}), }),
message, message,
} };
} }
export function resultError( export function resultError(
message = 'Request failed', message = 'Request failed',
{ code = ResultEnum.ERROR, result = null } = {}, { code = ResultEnum.ERROR, result = null } = {}
) { ) {
return { return {
code, code,
result, result,
message, message,
type: 'error', type: 'error',
} };
} }
export function pagination<T = any>(pageNo: number, pageSize: number, array: T[]): T[] { export function pagination<T = any>(pageNo: number, pageSize: number, array: T[]): T[] {
const offset = (pageNo - 1) * Number(pageSize) const offset = (pageNo - 1) * Number(pageSize);
const ret const ret =
= offset + Number(pageSize) >= array.length offset + Number(pageSize) >= array.length
? array.slice(offset, array.length) ? array.slice(offset, array.length)
: array.slice(offset, offset + Number(pageSize)) : array.slice(offset, offset + Number(pageSize));
return ret return ret;
} }
/** /**
* @param {number} times * @param {Number} times
* @param {Function} callback * @param {Function} callback
*/ */
export function doCustomTimes(times: number, callback: any) { export function doCustomTimes(times: number, callback: any) {
let i = -1 let i = -1;
while (++i < times) { while (++i < times) {
callback(i) callback(i);
} }
} }
export interface requestParams { export interface requestParams {
method: string method: string;
body: any body: any;
headers?: { authorization?: string } headers?: { authorization?: string };
query: any query: any;
} }
/** /**
@ -73,5 +73,5 @@ export interface requestParams {
* *
*/ */
export function getRequestToken({ headers }: requestParams): string | undefined { export function getRequestToken({ headers }: requestParams): string | undefined {
return headers?.authorization return headers?.authorization;
} }

View File

@ -1,7 +1,6 @@
import type { MockMethod } from 'vite-plugin-mock' import { MockMethod } from 'vite-plugin-mock';
import type { requestParams } from '../_util' import { getRequestToken, requestParams, resultError, resultSuccess } from '../_util';
import { getRequestToken, resultError, resultSuccess } from '../_util' import { ResultEnum } from '@/enums/httpEnum';
import { ResultEnum } from '@/enums/httpEnum'
const fakeUserList = [ const fakeUserList = [
{ {
@ -33,7 +32,7 @@ const fakeUserList = [
phone: '18822137893', phone: '18822137893',
token: 'fakeToken2', token: 'fakeToken2',
}, },
] ];
export default [ export default [
{ {
@ -41,21 +40,21 @@ export default [
timeout: 1000, timeout: 1000,
method: 'post', method: 'post',
response: ({ body }) => { response: ({ body }) => {
const { username, password } = body const { username, password } = body;
const checkUser = fakeUserList.find( const checkUser = fakeUserList.find(
item => item.username === username && password === item.password, (item) => item.username === username && password === item.password
) );
if (!checkUser) { if (!checkUser) {
return resultError('帐号或密码不正确') return resultError('帐号或密码不正确');
} }
const { userId, username: _username, token, realname, sign } = checkUser const { userId, username: _username, token, realname, sign } = checkUser;
return resultSuccess({ return resultSuccess({
userId, userId,
username: _username, username: _username,
token, token,
realname, realname,
sign, sign,
}) });
}, },
}, },
{ {
@ -63,17 +62,15 @@ export default [
timeout: 1000, timeout: 1000,
method: 'get', method: 'get',
response: (request: requestParams) => { response: (request: requestParams) => {
const token = getRequestToken(request) const token = getRequestToken(request);
if (!token) { if (!token) return resultError('无效令牌');
return resultError('无效令牌') const checkUser = fakeUserList.find((item) => item.token === token);
}
const checkUser = fakeUserList.find(item => item.token === token)
if (!checkUser) { if (!checkUser) {
return resultError('没有获取到对应的用户信息', { return resultError('没有获取到对应的用户信息', {
code: ResultEnum.TOKEN_EXPIRED, code: ResultEnum.TOKEN_EXPIRED,
}) });
} }
return resultSuccess(checkUser) return resultSuccess(checkUser);
}, },
}, },
{ {
@ -81,15 +78,13 @@ export default [
timeout: 1000, timeout: 1000,
method: 'post', method: 'post',
response: (request: requestParams) => { response: (request: requestParams) => {
const token = getRequestToken(request) const token = getRequestToken(request);
if (!token) { if (!token) return resultError('无效令牌');
return resultError('无效令牌') const checkUser = fakeUserList.find((item) => item.token === token);
}
const checkUser = fakeUserList.find(item => item.token === token)
if (!checkUser) { if (!checkUser) {
return resultError('无效令牌') return resultError('无效令牌');
} }
return resultSuccess(undefined, { message: '令牌已被销毁' }) return resultSuccess(undefined, { message: '令牌已被销毁' });
}, },
}, },
] as MockMethod[] ] as MockMethod[];

View File

@ -1,18 +1,12 @@
{ {
"name": "vue3-vant4-mobile", "name": "vue3-vant4-mobile",
"type": "module",
"version": "1.0.0",
"private": true, "private": true,
"packageManager": "pnpm@8.6.10", "version": "0.0.1",
"author": { "author": {
"name": "xiangshu233", "name": "xiangshu233",
"email": "xiangshu233@outlook.com", "email": "xiangshu233@outlook.com",
"url": "https://github.com/xiangshu233" "url": "https://github.com/xiangshu233"
}, },
"engines": {
"node": "^18.18.0 || ^20.9.0 || >=21.1.0",
"pnpm": ">=8.6.10"
},
"scripts": { "scripts": {
"preinstall": "npx only-allow pnpm", "preinstall": "npx only-allow pnpm",
"bootstrap": "pnpm install", "bootstrap": "pnpm install",
@ -25,87 +19,112 @@
"preview": "vite preview", "preview": "vite preview",
"clean:cache": "rimraf node_modules/.cache/ && rimraf node_modules/.vite", "clean:cache": "rimraf node_modules/.cache/ && rimraf node_modules/.vite",
"clean:lib": "rimraf node_modules", "clean:lib": "rimraf node_modules",
"lint": "eslint .", "lint:lint-staged": "lint-staged",
"lint:fix": "eslint . --fix", "lint:eslint": "eslint \"{src}/**/*.{vue,ts,tsx}\" --fix",
"lint:lint-staged": "lint-staged" "lint:prettier": "prettier --write --loglevel warn \"src/**/*.{js,json,tsx,css,less,scss,vue,html,md}\"",
"lint:stylelint": "stylelint --fix \"**/*.{vue,less,postcss,css,scss}\" --cache --cache-location node_modules/.cache/stylelint/",
"prepare": "husky install"
}, },
"dependencies": { "dependencies": {
"@types/lodash-es": "^4.17.12", "@types/lodash-es": "^4.17.6",
"@unocss/reset": "^0.58.5",
"@vicons/antd": "^0.12.0", "@vicons/antd": "^0.12.0",
"@vicons/ionicons5": "^0.12.0", "@vicons/ionicons5": "^0.12.0",
"@vicons/utils": "^0.1.4", "@vicons/utils": "^0.1.4",
"@vueuse/core": "^10.7.0", "@vueuse/core": "^9.2.0",
"axios": "^1.4.0", "axios": "^0.26.1",
"date-fns": "^3.0.6", "date-fns": "^2.29.2",
"echarts": "^5.4.3", "echarts": "^5.3.3",
"lodash-es": "^4.17.21", "lodash-es": "^4.17.21",
"mockjs": "^1.1.0", "mockjs": "^1.1.0",
"pinia": "^2.1.7", "pinia": "^2.0.19",
"pinia-plugin-persist": "^1.0.0", "pinia-plugin-persist": "^1.0.0",
"qs": "^6.11.2", "qs": "^6.11.0",
"vant": "^4.8.1", "vant": "4.0.0-beta.0",
"vue": "^3.3.13", "vue": "^3.2.37",
"vue-router": "4.2.5" "vue-router": "4.1.5",
"vue-types": "^4.2.1"
}, },
"devDependencies": { "devDependencies": {
"@antfu/eslint-config": "^2.6.3", "@commitlint/cli": "^17.1.2",
"@commitlint/cli": "^18.4.3", "@commitlint/config-conventional": "^17.1.0",
"@commitlint/config-conventional": "^18.4.3", "@types/fs-extra": "^9.0.13",
"@types/fs-extra": "^11.0.4", "@types/mockjs": "^1.0.6",
"@types/mockjs": "^1.0.10", "@types/node": "^18.7.1",
"@types/node": "^20.10.5", "@types/qs": "^6.9.7",
"@types/qs": "^6.9.11", "@typescript-eslint/eslint-plugin": "^5.33.1",
"@unocss/eslint-plugin": "^0.58.4", "@typescript-eslint/parser": "^5.33.1",
"@unocss/transformer-directives": "^0.58.4", "@vitejs/plugin-vue": "^3.0.3",
"@unocss/transformer-variant-group": "^0.58.4", "@vue/compiler-sfc": "^3.2.37",
"@vitejs/plugin-vue": "^5.0.0", "@vue/eslint-config-typescript": "^11.0.0",
"autoprefixer": "^10.4.16", "autoprefixer": "^10.4.8",
"cross-env": "^7.0.3", "cross-env": "^7.0.3",
"cz-git": "^1.8.0", "cz-git": "^1.3.12",
"dotenv": "^16.3.1", "dotenv": "^16.0.1",
"eslint": "^8.56.0", "eslint": "^8.22.0",
"eslint-plugin-format": "^0.1.0", "eslint-config-prettier": "^8.5.0",
"eslint-define-config": "^1.6.0",
"eslint-plugin-prettier": "^4.2.1",
"eslint-plugin-vue": "^9.3.0",
"esno": "^0.16.3", "esno": "^0.16.3",
"fs-extra": "^11.2.0", "fs-extra": "^10.1.0",
"less": "^4.2.0", "husky": "^8.0.1",
"lint-staged": "^15.2.0", "less": "^4.1.3",
"only-allow": "^1.2.1", "lint-staged": "^13.0.3",
"only-allow": "^1.1.1",
"picocolors": "^1.0.0", "picocolors": "^1.0.0",
"postcss": "^8.4.32", "postcss": "^8.4.16",
"postcss-html": "^1.5.0", "postcss-html": "^1.0.0",
"postcss-less": "^6.0.0", "postcss-less": "^6.0.0",
"postcss-mobile-forever": "^4.0.0", "postcss-mobile-forever": "^4.0.0",
"prettier": "^3.2.4", "prettier": "^2.7.1",
"rimraf": "^3.0.2", "rimraf": "^3.0.2",
"rollup": "^4.9.1", "rollup": "^2.79.0",
"rollup-plugin-visualizer": "^5.11.0", "rollup-plugin-visualizer": "^5.8.1",
"simple-git-hooks": "^2.9.0", "stylelint": "^14.10.0",
"stylelint": "^16.2.0", "stylelint-config-prettier": "^9.0.3",
"stylelint-config-recommended": "^14.0.0", "stylelint-config-recommended": "^9.0.0",
"stylelint-config-recommended-vue": "^1.5.0", "stylelint-config-recommended-vue": "^1.4.0",
"stylelint-config-standard": "^36.0.0", "stylelint-config-standard": "^27.0.0",
"stylelint-config-standard-less": "^3.0.1", "stylelint-order": "^5.0.0",
"stylelint-order": "^6.0.4", "typescript": "^4.6.4",
"stylelint-prettier": "^5.0.0", "unplugin-vue-components": "^0.22.4",
"typescript": "^5.3.3", "vite": "^3.0.0",
"unocss": "^0.58.5",
"unplugin-vue-components": "^0.26.0",
"vite": "^5.0.10",
"vite-plugin-compression": "^0.5.1", "vite-plugin-compression": "^0.5.1",
"vite-plugin-html": "^3.2.1", "vite-plugin-html": "^3.2.0",
"vite-plugin-mock": "^2.9.8", "vite-plugin-mock": "^2.9.6",
"vite-plugin-svg-icons": "^2.0.1", "vite-plugin-svg-icons": "^2.0.1",
"vue-tsc": "^1.8.27" "vite-plugin-vue-setup-extend": "^0.4.0",
"vite-plugin-windicss": "^1.8.7",
"vue-eslint-parser": "^9.0.0",
"vue-tsc": "^1.0.5"
}, },
"engines": {
"simple-git-hooks": { "node": ">=15",
"pre-commit": "pnpm lint-staged", "pnpm": ">=7"
"commit-msg": "npx --no-install commitlint --edit $1"
}, },
"lint-staged": { "lint-staged": {
"*": "eslint --fix" "*.{js,jsx,ts,tsx}": [
"eslint --fix",
"prettier --write"
],
"{!(package)*.json,*.code-snippets,.!(browserslist)*rc}": [
"prettier --write--parser json"
],
"package.json": [
"prettier --write"
],
"*.vue": [
"eslint --fix",
"prettier --write",
"stylelint --fix"
],
"*.{scss,less,styl,html}": [
"stylelint --fix",
"prettier --write"
],
"*.md": [
"prettier --write"
]
}, },
"config": { "config": {
"commitizen": { "commitizen": {

5622
pnpm-lock.yaml generated

File diff suppressed because it is too large Load Diff

View File

@ -10,13 +10,8 @@
* 改用postcss-px-to-viewport-8-plugin替代 * 改用postcss-px-to-viewport-8-plugin替代
*/ */
// FIXME: 升级 vite5 后控制台警告The CJS build of Vite's Node API is deprecated. const autoprefixer = require('autoprefixer');
// 将 "type": "module" 添加到 package.json 后, const viewport = require('postcss-mobile-forever');
// 所有*.js文件现在都解释为 ESM并且需要使用 ESM 语法。您可以使用扩展名重命名文件.cjs来继续使用 CJS。
// require 是cjs 语法
import autoprefixer from 'autoprefixer'
import viewport from 'postcss-mobile-forever'
const baseViewportOpts = { const baseViewportOpts = {
appSelector: '#app', // 根元素选择器,用于设置桌面端和横屏时的居中样式 appSelector: '#app', // 根元素选择器,用于设置桌面端和横屏时的居中样式
@ -26,8 +21,7 @@ const baseViewportOpts = {
propList: [ propList: [
'*', '*',
// '!font-size' // '!font-size'
], ], // 能转化为vw的属性列表!font-size表示font-size后面的单位不会被转换
// 能转化为vw的属性列表!font-size表示font-size后面的单位不会被转换
// 指定不转换为视口单位的类,可以自定义,可以无限添加,建议定义一至两个通用的类名 // 指定不转换为视口单位的类,可以自定义,可以无限添加,建议定义一至两个通用的类名
// 需要忽略的CSS选择器不会转为视口单位使用原有的px等单位。 // 需要忽略的CSS选择器不会转为视口单位使用原有的px等单位。
// 下面配置表示类名中含有'keep-px'以及'.ignore'类都不会被转换 // 下面配置表示类名中含有'keep-px'以及'.ignore'类都不会被转换
@ -37,16 +31,16 @@ const baseViewportOpts = {
// exclude: [/node_modules/], // 忽略某些文件夹下的文件或特定文件 // exclude: [/node_modules/], // 忽略某些文件夹下的文件或特定文件
// include: [/src/], // 如果设置了include那将只有匹配到的文件才会被转换 // include: [/src/], // 如果设置了include那将只有匹配到的文件才会被转换
mobileUnit: 'vw', // 指定需要转换成的视口单位,建议使用 vw mobileUnit: 'vw', // 指定需要转换成的视口单位,建议使用 vw
rootContainingBlockSelectorList: ['van-popup--bottom'], // 指定包含块是根包含块的选择器,这种选择器的定位通常是 `fixed`,但是选择器内没有 `position: fixed` rootContainingBlockSelectorList: ["van-popup--bottom"], // 指定包含块是根包含块的选择器,这种选择器的定位通常是 `fixed`,但是选择器内没有 `position: fixed`
} };
export default { module.exports = {
plugins: [ plugins: [
autoprefixer(), autoprefixer(),
viewport({ viewport({
...baseViewportOpts, ...baseViewportOpts,
// 只将 vant 转为 350 设计稿的 viewport其它样式的视图宽度为 750 // 只将 vant 转为 350 设计稿的 viewport其它样式的视图宽度为 750
viewportWidth: file => (file.includes('node_modules/vant/') ? 375 : 750), viewportWidth: (file) => (file.includes('node_modules/vant/') ? 375 : 750),
}), }),
], ],
} };

20
prettier.config.js Normal file
View File

@ -0,0 +1,20 @@
module.exports = {
printWidth: 100,
tabWidth: 2,
useTabs: false,
semi: true,
vueIndentScriptAndStyle: true,
singleQuote: true,
quoteProps: 'as-needed',
bracketSpacing: true,
trailingComma: 'es5',
jsxBracketSameLine: false,
jsxSingleQuote: false,
arrowParens: 'always',
insertPragma: false,
requirePragma: false,
proseWrap: 'never',
htmlWhitespaceSensitivity: 'strict',
endOfLine: 'auto',
rangeStart: 0,
};

View File

@ -1,7 +1,7 @@
<template> <template>
<vanConfigProvider :theme="getDarkMode" :theme-vars="getThemeVars()"> <vanConfigProvider :theme="getDarkMode" :theme-vars="getThemeVars()">
<routerView v-slot="{ Component }"> <routerView v-slot="{ Component }">
<div class="absolute bottom-0 top-0 w-full overflow-hidden"> <div class="absolute top-0 bottom-0 w-full overflow-hidden">
<transition :name="getTransitionName" mode="out-in" appear> <transition :name="getTransitionName" mode="out-in" appear>
<keep-alive v-if="keepAliveComponents" :include="keepAliveComponents"> <keep-alive v-if="keepAliveComponents" :include="keepAliveComponents">
<component :is="Component" /> <component :is="Component" />
@ -13,64 +13,64 @@
</template> </template>
<script setup lang="ts"> <script setup lang="ts">
import { computed, unref } from 'vue' import { computed, unref } from 'vue';
import { darken, lighten } from '@/utils/index' import { darken, lighten } from '@/utils/index';
import { useRouteStore } from '@/store/modules/route' import { useRouteStore } from '@/store/modules/route';
import { useDesignSetting } from '@/hooks/setting/useDesignSetting' import { useDesignSetting } from '@/hooks/setting/useDesignSetting';
const routeStore = useRouteStore() const routeStore = useRouteStore();
const { getDarkMode, getAppTheme, getIsPageAnimate, getPageAnimateType } = useDesignSetting() const { getDarkMode, getAppTheme, getIsPageAnimate, getPageAnimateType } = useDesignSetting();
// //
const keepAliveComponents = computed(() => routeStore.keepAliveComponents) const keepAliveComponents = computed(() => routeStore.keepAliveComponents);
function getThemeVars() { const getThemeVars = () => {
const appTheme = unref(getAppTheme) const appTheme = unref(getAppTheme);
const darkenStr = darken(appTheme, 25) const darkenStr = darken(appTheme, 25);
const lightenStr = lighten(appTheme, 10) const lightenStr = lighten(appTheme, 10);
return { return {
actionSheetCancelTextColor: appTheme, actionSheetCancelTextColor: appTheme,
buttonPrimaryBackground: appTheme, buttonPrimaryBackground: appTheme,
buttonPrimaryBorderColor: appTheme, buttonPrimaryBorderColor: appTheme,
radioCheckedIconColor: appTheme, radioCheckedIconColor: appTheme,
sliderActiveBackground: appTheme, sliderActiveBackground: appTheme,
cascaderActiveColor: appTheme, cascaderActiveColor: appTheme,
checkboxCheckedIconColor: appTheme, checkboxCheckedIconColor: appTheme,
numberKeyboardButtonBackground: appTheme, numberKeyboardButtonBackground: appTheme,
pickerLoadingIconColor: appTheme, pickerLoadingIconColor: appTheme,
calendarRangeEdgeBackground: appTheme, calendarRangeEdgeBackground: appTheme,
calendarRangeMiddleColor: appTheme, calendarRangeMiddleColor: appTheme,
calendarSelectedDayBackground: appTheme, calendarSelectedDayBackground: appTheme,
stepperButtonRoundThemeColor: appTheme, stepperButtonRoundThemeColor: appTheme,
switchOnBackground: appTheme, switchOnBackground: appTheme,
dialogConfirmButtonTextColor: appTheme, dialogConfirmButtonTextColor: appTheme,
dropdownMenuOptionActiveColor: appTheme, dropdownMenuOptionActiveColor: appTheme,
dropdownMenuTitleActiveTextColor: appTheme, dropdownMenuTitleActiveTextColor: appTheme,
notifyPrimaryBackground: appTheme, notifyPrimaryBackground: appTheme,
circleColor: appTheme, circleColor: appTheme,
noticeBarBackground: lightenStr, noticeBarBackground: lightenStr,
noticeBarTextColor: darkenStr, noticeBarTextColor: darkenStr,
progressColor: appTheme, progressColor: appTheme,
progressPivotBackground: appTheme, progressPivotBackground: appTheme,
stepActiveColor: appTheme, stepActiveColor: appTheme,
stepFinishLineColor: appTheme, stepFinishLineColor: appTheme,
swipeIndicatorActiveBackground: appTheme, swipeIndicatorActiveBackground: appTheme,
tagPrimaryColor: appTheme, tagPrimaryColor: appTheme,
navBarIconColor: appTheme, navBarIconColor: appTheme,
navBarTextColor: appTheme, navBarTextColor: appTheme,
paginationItemDefaultColor: appTheme, paginationItemDefaultColor: appTheme,
sidebarSelectedBorderColor: appTheme, sidebarSelectedBorderColor: appTheme,
tabsDefaultColor: appTheme, tabsDefaultColor: appTheme,
tabsBottomBarColor: appTheme, tabsBottomBarColor: appTheme,
tabbarItemActiveColor: appTheme, tabbarItemActiveColor: appTheme,
treeSelectItemActiveColor: appTheme, treeSelectItemActiveColor: appTheme,
} };
} };
const getTransitionName = computed(() => { const getTransitionName = computed(() => {
return unref(getIsPageAnimate) ? unref(getPageAnimateType) : undefined return unref(getIsPageAnimate) ? unref(getPageAnimateType) : undefined;
}) });
</script> </script>
<style lang="less"> <style lang="less">

View File

@ -1,9 +1,9 @@
import { http } from '@/utils/http/axios' import { http } from '@/utils/http/axios';
export interface BasicResponseModel<T = any> { export interface BasicResponseModel<T = any> {
code: number code: number;
message: string message: string;
result: T result: T;
} }
/** /**
@ -18,8 +18,8 @@ export function login(params: any) {
}, },
{ {
isTransformResponse: false, isTransformResponse: false,
}, }
) );
} }
/** /**
@ -29,7 +29,7 @@ export function getUserInfo() {
return http.request({ return http.request({
url: '/getUserInfo', url: '/getUserInfo',
method: 'get', method: 'get',
}) });
} }
/** /**
@ -39,7 +39,7 @@ export function doLogout() {
return http.request({ return http.request({
url: '/logout', url: '/logout',
method: 'POST', method: 'POST',
}) });
} }
/** /**
@ -54,6 +54,6 @@ export function changePassword(params: any, uid: any) {
}, },
{ {
isTransformResponse: false, isTransformResponse: false,
}, }
) );
} }

View File

@ -5,45 +5,44 @@
</template> </template>
<script lang="ts"> <script lang="ts">
import type { CSSProperties } from 'vue' import type { CSSProperties } from 'vue';
import { computed, defineComponent } from 'vue' import { defineComponent, computed } from 'vue';
export default defineComponent({ export default defineComponent({
name: 'SvgIcon', name: 'SvgIcon',
props: { props: {
prefix: { prefix: {
type: String, type: String,
default: 'icon', default: 'icon',
},
name: {
type: String,
required: true,
},
size: {
type: [Number, String],
default: 16,
},
color: {
type: String,
default: '#333',
},
}, },
name: { setup(props) {
type: String, const symbolId = computed(() => `#${props.prefix}-${props.name}`);
required: true,
},
size: {
type: [Number, String],
default: 16,
},
color: {
type: String,
default: '#333',
},
},
setup(props) {
const symbolId = computed(() => `#${props.prefix}-${props.name}`)
const getStyle = computed((): CSSProperties => { const getStyle = computed((): CSSProperties => {
const { size } = props const { size } = props;
let s = `${size}` let s = `${size}`;
s = `${s.replace('px', '')}px` s = `${s.replace('px', '')}px`;
return { return {
width: s, width: s,
height: s, height: s,
} };
}) });
return { symbolId, getStyle } return { symbolId, getStyle };
}, },
}) });
</script> </script>
<style scoped lang="less"></style> <style scoped lang="less"></style>

View File

@ -16,13 +16,13 @@ export enum screenEnum {
XXL = 1600, XXL = 1600,
} }
const screenMap = new Map<sizeEnum, number>() const screenMap = new Map<sizeEnum, number>();
screenMap.set(sizeEnum.XS, screenEnum.XS) screenMap.set(sizeEnum.XS, screenEnum.XS);
screenMap.set(sizeEnum.SM, screenEnum.SM) screenMap.set(sizeEnum.SM, screenEnum.SM);
screenMap.set(sizeEnum.MD, screenEnum.MD) screenMap.set(sizeEnum.MD, screenEnum.MD);
screenMap.set(sizeEnum.LG, screenEnum.LG) screenMap.set(sizeEnum.LG, screenEnum.LG);
screenMap.set(sizeEnum.XL, screenEnum.XL) screenMap.set(sizeEnum.XL, screenEnum.XL);
screenMap.set(sizeEnum.XXL, screenEnum.XXL) screenMap.set(sizeEnum.XXL, screenEnum.XXL);
export { screenMap } export { screenMap };

View File

@ -1,14 +1,14 @@
// token key // token key
export const TOKEN_KEY = 'TOKEN' export const TOKEN_KEY = 'TOKEN';
// user info key // user info key
export const USER_INFO_KEY = 'USER__INFO__' export const USER_INFO_KEY = 'USER__INFO__';
// role info key // role info key
export const ROLES_KEY = 'ROLES__KEY__' export const ROLES_KEY = 'ROLES__KEY__';
// base global local key // base global local key
export const BASE_LOCAL_CACHE_KEY = 'LOCAL__CACHE__KEY__' export const BASE_LOCAL_CACHE_KEY = 'LOCAL__CACHE__KEY__';
// base global session key // base global session key
export const BASE_SESSION_CACHE_KEY = 'SESSION__CACHE__KEY__' export const BASE_SESSION_CACHE_KEY = 'SESSION__CACHE__KEY__';

View File

@ -1,4 +1,3 @@
/* eslint-disable ts/no-duplicate-enum-values */
export enum PageEnum { export enum PageEnum {
// 登录 // 登录
BASE_LOGIN = '/login', BASE_LOGIN = '/login',

View File

@ -1,48 +1,47 @@
import { ref, watch } from 'vue' import { ref, watch } from 'vue';
import { tryOnUnmounted } from '@vueuse/core' import { tryOnUnmounted } from '@vueuse/core';
import { isFunction } from '@/utils/is' import { isFunction } from '@/utils/is';
export function useTimeoutFn(handle: Fn<any>, wait: number, native = false) { export function useTimeoutFn(handle: Fn<any>, wait: number, native = false) {
if (!isFunction(handle)) { if (!isFunction(handle)) {
throw new Error('handle is not Function!') throw new Error('handle is not Function!');
} }
const { readyRef, stop, start } = useTimeoutRef(wait) const { readyRef, stop, start } = useTimeoutRef(wait);
if (native) { if (native) {
handle() handle();
} } else {
else {
watch( watch(
readyRef, readyRef,
(maturity) => { (maturity) => {
maturity && handle() maturity && handle();
}, },
{ immediate: false }, { immediate: false }
) );
} }
return { readyRef, stop, start } return { readyRef, stop, start };
} }
export function useTimeoutRef(wait: number) { export function useTimeoutRef(wait: number) {
const readyRef = ref(false) const readyRef = ref(false);
let timer: TimeoutHandle let timer: TimeoutHandle;
function stop(): void { function stop(): void {
readyRef.value = false readyRef.value = false;
timer && window.clearTimeout(timer) timer && window.clearTimeout(timer);
} }
function start(): void { function start(): void {
stop() stop();
timer = setTimeout(() => { timer = setTimeout(() => {
readyRef.value = true readyRef.value = true;
}, wait) }, wait);
} }
start() start();
tryOnUnmounted(stop) tryOnUnmounted(stop);
return { readyRef, stop, start } return { readyRef, stop, start };
} }

View File

@ -1,19 +1,18 @@
import type { ComputedRef } from 'vue' import { ref, computed, ComputedRef, unref } from 'vue';
import { computed, ref, unref } from 'vue' import { useEventListener } from '@/hooks/event/useEventListener';
import { useEventListener } from '@/hooks/event/useEventListener' import { screenMap, sizeEnum, screenEnum } from '@/enums/breakpointEnum';
import { screenEnum, screenMap, sizeEnum } from '@/enums/breakpointEnum'
let globalScreenRef: ComputedRef<sizeEnum | undefined> let globalScreenRef: ComputedRef<sizeEnum | undefined>;
let globalWidthRef: ComputedRef<number> let globalWidthRef: ComputedRef<number>;
let globalRealWidthRef: ComputedRef<number> let globalRealWidthRef: ComputedRef<number>;
export interface CreateCallbackParams { export interface CreateCallbackParams {
screen: ComputedRef<sizeEnum | undefined> screen: ComputedRef<sizeEnum | undefined>;
width: ComputedRef<number> width: ComputedRef<number>;
realWidth: ComputedRef<number> realWidth: ComputedRef<number>;
screenEnum: typeof screenEnum screenEnum: typeof screenEnum;
screenMap: Map<sizeEnum, number> screenMap: Map<sizeEnum, number>;
sizeEnum: typeof sizeEnum sizeEnum: typeof sizeEnum;
} }
export function useBreakpoint() { export function useBreakpoint() {
@ -22,40 +21,35 @@ export function useBreakpoint() {
widthRef: globalWidthRef, widthRef: globalWidthRef,
screenEnum, screenEnum,
realWidthRef: globalRealWidthRef, realWidthRef: globalRealWidthRef,
} };
} }
// Just call it once // Just call it once
export function createBreakpointListen(fn?: (opt: CreateCallbackParams) => void) { export function createBreakpointListen(fn?: (opt: CreateCallbackParams) => void) {
const screenRef = ref<sizeEnum>(sizeEnum.XL) const screenRef = ref<sizeEnum>(sizeEnum.XL);
const realWidthRef = ref(window.innerWidth) const realWidthRef = ref(window.innerWidth);
function getWindowWidth() { function getWindowWidth() {
const width = document.body.clientWidth const width = document.body.clientWidth;
const xs = screenMap.get(sizeEnum.XS)! const xs = screenMap.get(sizeEnum.XS)!;
const sm = screenMap.get(sizeEnum.SM)! const sm = screenMap.get(sizeEnum.SM)!;
const md = screenMap.get(sizeEnum.MD)! const md = screenMap.get(sizeEnum.MD)!;
const lg = screenMap.get(sizeEnum.LG)! const lg = screenMap.get(sizeEnum.LG)!;
const xl = screenMap.get(sizeEnum.XL)! const xl = screenMap.get(sizeEnum.XL)!;
if (width < xs) { if (width < xs) {
screenRef.value = sizeEnum.XS screenRef.value = sizeEnum.XS;
} else if (width < sm) {
screenRef.value = sizeEnum.SM;
} else if (width < md) {
screenRef.value = sizeEnum.MD;
} else if (width < lg) {
screenRef.value = sizeEnum.LG;
} else if (width < xl) {
screenRef.value = sizeEnum.XL;
} else {
screenRef.value = sizeEnum.XXL;
} }
else if (width < sm) { realWidthRef.value = width;
screenRef.value = sizeEnum.SM
}
else if (width < md) {
screenRef.value = sizeEnum.MD
}
else if (width < lg) {
screenRef.value = sizeEnum.LG
}
else if (width < xl) {
screenRef.value = sizeEnum.XL
}
else {
screenRef.value = sizeEnum.XXL
}
realWidthRef.value = width
} }
useEventListener({ useEventListener({
@ -63,16 +57,16 @@ export function createBreakpointListen(fn?: (opt: CreateCallbackParams) => void)
name: 'resize', name: 'resize',
listener: () => { listener: () => {
getWindowWidth() getWindowWidth();
resizeFn() resizeFn();
}, },
// wait: 100, // wait: 100,
}) });
getWindowWidth() getWindowWidth();
globalScreenRef = computed(() => unref(screenRef)) globalScreenRef = computed(() => unref(screenRef));
globalWidthRef = computed((): number => screenMap.get(unref(screenRef)!)!) globalWidthRef = computed((): number => screenMap.get(unref(screenRef)!)!);
globalRealWidthRef = computed((): number => unref(realWidthRef)) globalRealWidthRef = computed((): number => unref(realWidthRef));
function resizeFn() { function resizeFn() {
fn?.({ fn?.({
@ -82,14 +76,14 @@ export function createBreakpointListen(fn?: (opt: CreateCallbackParams) => void)
screenEnum, screenEnum,
screenMap, screenMap,
sizeEnum, sizeEnum,
}) });
} }
resizeFn() resizeFn();
return { return {
screenRef: globalScreenRef, screenRef: globalScreenRef,
screenEnum, screenEnum,
widthRef: globalWidthRef, widthRef: globalWidthRef,
realWidthRef: globalRealWidthRef, realWidthRef: globalRealWidthRef,
} };
} }

View File

@ -1,18 +1,18 @@
import type { Ref } from 'vue' import type { Ref } from 'vue';
import { ref, unref, watch } from 'vue' import { ref, watch, unref } from 'vue';
import { useDebounceFn, useThrottleFn } from '@vueuse/core' import { useThrottleFn, useDebounceFn } from '@vueuse/core';
export type RemoveEventFn = () => void export type RemoveEventFn = () => void;
export interface UseEventParams { export interface UseEventParams {
el?: Element | Ref<Element | undefined> | Window | any el?: Element | Ref<Element | undefined> | Window | any;
name: string name: string;
listener: EventListener listener: EventListener;
options?: boolean | AddEventListenerOptions options?: boolean | AddEventListenerOptions;
autoRemove?: boolean autoRemove?: boolean;
isDebounce?: boolean isDebounce?: boolean;
wait?: number wait?: number;
} }
export function useEventListener({ export function useEventListener({
@ -24,38 +24,39 @@ export function useEventListener({
isDebounce = true, isDebounce = true,
wait = 80, wait = 80,
}: UseEventParams): { removeEvent: RemoveEventFn } { }: UseEventParams): { removeEvent: RemoveEventFn } {
let remove: RemoveEventFn = () => { /* eslint-disable-next-line */
} let remove: RemoveEventFn = () => {
const isAddRef = ref(false) };
const isAddRef = ref(false);
if (el) { if (el) {
const element: Ref<Element> = ref(el as Element) const element: Ref<Element> = ref(el as Element);
const handler = isDebounce ? useDebounceFn(listener, wait) : useThrottleFn(listener, wait) const handler = isDebounce ? useDebounceFn(listener, wait) : useThrottleFn(listener, wait);
const realHandler = wait ? handler : listener const realHandler = wait ? handler : listener;
const removeEventListener = (e: Element) => { const removeEventListener = (e: Element) => {
isAddRef.value = true isAddRef.value = true;
e.removeEventListener(name, realHandler, options) e.removeEventListener(name, realHandler, options);
} };
const addEventListener = (e: Element) => e.addEventListener(name, realHandler, options) const addEventListener = (e: Element) => e.addEventListener(name, realHandler, options);
const removeWatch = watch( const removeWatch = watch(
element, element,
(v, _ov, cleanUp) => { (v, _ov, cleanUp) => {
if (v) { if (v) {
!unref(isAddRef) && addEventListener(v) !unref(isAddRef) && addEventListener(v);
cleanUp(() => { cleanUp(() => {
autoRemove && removeEventListener(v) autoRemove && removeEventListener(v);
}) });
} }
}, },
{ immediate: true }, { immediate: true }
) );
remove = () => { remove = () => {
removeEventListener(element.value) removeEventListener(element.value);
removeWatch() removeWatch();
} };
} }
return { removeEvent: remove } return { removeEvent: remove };
} }

View File

@ -1,35 +1,36 @@
import { tryOnMounted, tryOnUnmounted, useDebounceFn } from '@vueuse/core' import { tryOnMounted, tryOnUnmounted } from '@vueuse/core';
import { useDebounceFn } from '@vueuse/core';
interface WindowSizeOptions { interface WindowSizeOptions {
once?: boolean once?: boolean;
immediate?: boolean immediate?: boolean;
listenerOptions?: AddEventListenerOptions | boolean listenerOptions?: AddEventListenerOptions | boolean;
} }
export function useWindowSizeFn<T>(fn: Fn<T>, wait = 150, options?: WindowSizeOptions) { export function useWindowSizeFn<T>(fn: Fn<T>, wait = 150, options?: WindowSizeOptions) {
let handler = () => { let handler = () => {
fn() fn();
} };
const handleSize = useDebounceFn(handler, wait) const handleSize = useDebounceFn(handler, wait);
handler = handleSize handler = handleSize;
const start = () => { const start = () => {
if (options && options.immediate) { if (options && options.immediate) {
handler() handler();
} }
window.addEventListener('resize', handler) window.addEventListener('resize', handler);
} };
const stop = () => { const stop = () => {
window.removeEventListener('resize', handler) window.removeEventListener('resize', handler);
} };
tryOnMounted(() => { tryOnMounted(() => {
start() start();
}) });
tryOnUnmounted(() => { tryOnUnmounted(() => {
stop() stop();
}) });
return [start, stop] return [start, stop];
} }

View File

@ -1,3 +1,3 @@
import { useAsync } from './use-async' import { useAsync } from './use-async';
export { useAsync } export { useAsync };

View File

@ -1,8 +1,8 @@
import { warn } from '@/utils/log' import { warn } from '@/utils/log';
import { getAppEnvConfig } from '@/utils/env' import { getAppEnvConfig } from '@/utils/env';
import type { GlobConfig } from '#/config' import { GlobConfig } from '#/config';
export function useGlobSetting(): Readonly<GlobConfig> { export const useGlobSetting = (): Readonly<GlobConfig> => {
const { const {
VITE_GLOB_APP_TITLE, VITE_GLOB_APP_TITLE,
VITE_GLOB_APP_TITLE_CN, VITE_GLOB_APP_TITLE_CN,
@ -12,12 +12,12 @@ export function useGlobSetting(): Readonly<GlobConfig> {
VITE_GLOB_UPLOAD_URL, VITE_GLOB_UPLOAD_URL,
VITE_GLOB_PROD_MOCK, VITE_GLOB_PROD_MOCK,
VITE_GLOB_IMG_URL, VITE_GLOB_IMG_URL,
} = getAppEnvConfig() } = getAppEnvConfig();
if (!/[a-zA-Z\_]*/.test(VITE_GLOB_APP_SHORT_NAME)) { if (!/[a-zA-Z\_]*/.test(VITE_GLOB_APP_SHORT_NAME)) {
warn( warn(
`VITE_GLOB_APP_SHORT_NAME Variables can only be characters/underscores, please modify in the environment variables and re-running.`, `VITE_GLOB_APP_SHORT_NAME Variables can only be characters/underscores, please modify in the environment variables and re-running.`
) );
} }
// Take global configuration // Take global configuration
@ -30,6 +30,6 @@ export function useGlobSetting(): Readonly<GlobConfig> {
uploadUrl: VITE_GLOB_UPLOAD_URL, uploadUrl: VITE_GLOB_UPLOAD_URL,
prodMock: VITE_GLOB_PROD_MOCK, prodMock: VITE_GLOB_PROD_MOCK,
imgUrl: VITE_GLOB_IMG_URL, imgUrl: VITE_GLOB_IMG_URL,
} };
return glob as Readonly<GlobConfig> return glob as Readonly<GlobConfig>;
} };

View File

@ -1,18 +1,18 @@
import { computed } from 'vue' import { computed } from 'vue';
import { useDesignSettingStore } from '@/store/modules/designSetting' import { useDesignSettingStore } from '@/store/modules/designSetting';
export function useDesignSetting() { export function useDesignSetting() {
const designStore = useDesignSettingStore() const designStore = useDesignSettingStore();
const getDarkMode = computed(() => designStore.darkMode) const getDarkMode = computed(() => designStore.darkMode);
const getAppTheme = computed(() => designStore.appTheme) const getAppTheme = computed(() => designStore.appTheme);
const getAppThemeList = computed(() => designStore.appThemeList) const getAppThemeList = computed(() => designStore.appThemeList);
const getIsPageAnimate = computed(() => designStore.isPageAnimate) const getIsPageAnimate = computed(() => designStore.isPageAnimate);
const getPageAnimateType = computed(() => designStore.pageAnimateType) const getPageAnimateType = computed(() => designStore.pageAnimateType);
return { return {
getDarkMode, getDarkMode,
@ -20,5 +20,5 @@ export function useDesignSetting() {
getAppThemeList, getAppThemeList,
getIsPageAnimate, getIsPageAnimate,
getPageAnimateType, getPageAnimateType,
} };
} }

View File

@ -1,16 +1,15 @@
import { isReactive, isRef } from 'vue' import { isReactive, isRef } from 'vue';
function setLoading(loading, val) { function setLoading(loading, val) {
if (loading !== undefined && isRef(loading)) { if (loading != undefined && isRef(loading)) {
loading.value = val loading.value = val;
} } else if (loading != undefined && isReactive(loading)) {
else if (loading !== undefined && isReactive(loading)) { loading.loading = val;
loading.loading = val
} }
} }
export async function useAsync(func: Promise<any>, loading: any): Promise<any> { export const useAsync = async (func: Promise<any>, loading: any): Promise<any> => {
setLoading(loading, true) setLoading(loading, true);
return await func.finally(() => setLoading(loading, false)) return await func.finally(() => setLoading(loading, false));
} };

View File

@ -1,23 +1,23 @@
import { onMounted, onUnmounted, ref } from 'vue' import { ref, onMounted, onUnmounted } from 'vue';
import { debounce } from 'lodash-es' import { debounce } from 'lodash-es';
/** /**
* description: 获取页面宽度 * description: 获取页面宽度
*/ */
export function useDomWidth() { export function useDomWidth() {
const domWidth = ref(window.innerWidth) const domWidth = ref(window.innerWidth);
function resize() { function resize() {
domWidth.value = document.body.clientWidth domWidth.value = document.body.clientWidth;
} }
onMounted(() => { onMounted(() => {
window.addEventListener('resize', debounce(resize, 80)) window.addEventListener('resize', debounce(resize, 80));
}) });
onUnmounted(() => { onUnmounted(() => {
window.removeEventListener('resize', resize) window.removeEventListener('resize', resize);
}) });
return domWidth return domWidth;
} }

View File

@ -1,30 +1,30 @@
import { onMounted, onUnmounted, ref } from 'vue' import { ref, onMounted, onUnmounted } from 'vue';
/** /**
* @description * @description
*/ * */
export function useOnline() { export function useOnline() {
const online = ref(true) const online = ref(true);
const showStatus = (val) => { const showStatus = (val) => {
online.value = typeof val == 'boolean' ? val : val.target.online online.value = typeof val == 'boolean' ? val : val.target.online;
} };
// 在页面加载后,设置正确的网络状态 // 在页面加载后,设置正确的网络状态
navigator.onLine ? showStatus(true) : showStatus(false) navigator.onLine ? showStatus(true) : showStatus(false);
onMounted(() => { onMounted(() => {
// 开始监听网络状态的变化 // 开始监听网络状态的变化
window.addEventListener('online', showStatus) window.addEventListener('online', showStatus);
window.addEventListener('offline', showStatus) window.addEventListener('offline', showStatus);
}) });
onUnmounted(() => { onUnmounted(() => {
// 移除监听网络状态的变化 // 移除监听网络状态的变化
window.removeEventListener('online', showStatus) window.removeEventListener('online', showStatus);
window.removeEventListener('offline', showStatus) window.removeEventListener('offline', showStatus);
}) });
return { online } return { online };
} }

View File

@ -1,33 +1,33 @@
import { onMounted, onUnmounted, ref } from 'vue' import { ref, onMounted, onUnmounted } from 'vue';
/** /**
* @description * @description
*/ */
export function useTime() { export function useTime() {
let timer // 定时器 let timer; // 定时器
const year = ref(0) // 年份 const year = ref(0); // 年份
const month = ref(0) // 月份 const month = ref(0); // 月份
const week = ref('') // 星期几 const week = ref(''); // 星期几
const day = ref(0) // 天数 const day = ref(0); // 天数
const hour = ref<number | string>(0) // 小时 const hour = ref<number | string>(0); // 小时
const minute = ref<number | string>(0) // 分钟 const minute = ref<number | string>(0); // 分钟
const second = ref(0) // 秒 const second = ref(0); // 秒
// 更新时间 // 更新时间
const updateTime = () => { const updateTime = () => {
const date = new Date() const date = new Date();
year.value = date.getFullYear() year.value = date.getFullYear();
month.value = date.getMonth() + 1 month.value = date.getMonth() + 1;
week.value = '日一二三四五六'.charAt(date.getDay()) week.value = '日一二三四五六'.charAt(date.getDay());
day.value = date.getDate() day.value = date.getDate();
hour.value hour.value =
= (`${date.getHours()}`)?.padStart(2, '0') (date.getHours() + '')?.padStart(2, '0') ||
|| new Intl.NumberFormat(undefined, { minimumIntegerDigits: 2 }).format(date.getHours()) new Intl.NumberFormat(undefined, { minimumIntegerDigits: 2 }).format(date.getHours());
minute.value minute.value =
= (`${date.getMinutes()}`)?.padStart(2, '0') (date.getMinutes() + '')?.padStart(2, '0') ||
|| new Intl.NumberFormat(undefined, { minimumIntegerDigits: 2 }).format(date.getMinutes()) new Intl.NumberFormat(undefined, { minimumIntegerDigits: 2 }).format(date.getMinutes());
second.value = date.getSeconds() second.value = date.getSeconds();
} };
// 原生时间格式化 // 原生时间格式化
// new Intl.DateTimeFormat('zh', { // new Intl.DateTimeFormat('zh', {
@ -40,16 +40,16 @@ export function useTime() {
// hour12: false // hour12: false
// }).format(new Date()) // }).format(new Date())
updateTime() updateTime();
onMounted(() => { onMounted(() => {
clearInterval(timer) clearInterval(timer);
timer = setInterval(() => updateTime(), 1000) timer = setInterval(() => updateTime(), 1000);
}) });
onUnmounted(() => { onUnmounted(() => {
clearInterval(timer) clearInterval(timer);
}) });
return { month, day, hour, minute, second, week } return { month, day, hour, minute, second, week };
} }

View File

@ -1,116 +1,111 @@
import type { EChartsOption } from 'echarts' import type { EChartsOption } from 'echarts';
import type { Ref } from 'vue' import type { Ref } from 'vue';
import type { Fn } from '@vueuse/core' import { useTimeoutFn } from '@/hooks/core/useTimeout';
import { tryOnUnmounted, useDebounceFn } from '@vueuse/core' import { Fn, tryOnUnmounted } from '@vueuse/core';
import { computed, nextTick, ref, unref, watch } from 'vue' import { unref, nextTick, watch, computed, ref } from 'vue';
import { useTimeoutFn } from '@/hooks/core/useTimeout' import { useDebounceFn } from '@vueuse/core';
import { useEventListener } from '@/hooks/event/useEventListener';
import { useEventListener } from '@/hooks/event/useEventListener' import { useBreakpoint } from '@/hooks/event/useBreakpoint';
import { useBreakpoint } from '@/hooks/event/useBreakpoint' import { useDesignSettingStore } from '@/store/modules/designSetting';
import { useDesignSettingStore } from '@/store/modules/designSetting' import echarts from '@/utils/lib/echarts';
import echarts from '@/utils/lib/echarts'
export function useECharts( export function useECharts(
elRef: Ref<HTMLDivElement>, elRef: Ref<HTMLDivElement>,
theme: 'light' | 'dark' | 'default' = 'default', theme: 'light' | 'dark' | 'default' = 'default'
) { ) {
const designStore = useDesignSettingStore() const designStore = useDesignSettingStore();
const getDarkMode = computed(() => { const getDarkMode = computed(() => {
return theme === 'default' ? designStore.getDarkMode : theme return theme === 'default' ? designStore.getDarkMode : theme;
}) });
let chartInstance: echarts.ECharts | null = null let chartInstance: echarts.ECharts | null = null;
let resizeFn: Fn = resize let resizeFn: Fn = resize;
const cacheOptions = ref({}) const cacheOptions = ref({});
let removeResizeFn: Fn = () => {} let removeResizeFn: Fn = () => {};
resizeFn = useDebounceFn(resize, 200) resizeFn = useDebounceFn(resize, 200);
const getOptions = computed((): EChartsOption => { const getOptions = computed((): EChartsOption => {
if (getDarkMode.value !== 'dark') { if (getDarkMode.value !== 'dark') {
return cacheOptions.value return cacheOptions.value;
} }
return { return {
backgroundColor: 'transparent', backgroundColor: 'transparent',
...cacheOptions.value, ...cacheOptions.value,
} };
}) });
function initCharts(t = theme) { function initCharts(t = theme) {
const el = unref(elRef) const el = unref(elRef);
if (!el || !unref(el)) { if (!el || !unref(el)) {
return return;
} }
chartInstance = echarts.init(el, t) chartInstance = echarts.init(el, t);
const { removeEvent } = useEventListener({ const { removeEvent } = useEventListener({
el: window, el: window,
name: 'resize', name: 'resize',
listener: resizeFn, listener: resizeFn,
}) });
removeResizeFn = removeEvent removeResizeFn = removeEvent;
const { widthRef, screenEnum } = useBreakpoint() const { widthRef, screenEnum } = useBreakpoint();
if (unref(widthRef) <= screenEnum.MD || el.offsetHeight === 0) { if (unref(widthRef) <= screenEnum.MD || el.offsetHeight === 0) {
useTimeoutFn(() => { useTimeoutFn(() => {
resizeFn() resizeFn();
}, 30) }, 30);
} }
} }
function setOptions(options: EChartsOption, clear = true) { function setOptions(options: EChartsOption, clear = true) {
cacheOptions.value = options cacheOptions.value = options;
if (unref(elRef)?.offsetHeight === 0) { if (unref(elRef)?.offsetHeight === 0) {
useTimeoutFn(() => { useTimeoutFn(() => {
setOptions(unref(getOptions)) setOptions(unref(getOptions));
}, 30) }, 30);
return return;
} }
nextTick(() => { nextTick(() => {
useTimeoutFn(() => { useTimeoutFn(() => {
if (!chartInstance) { if (!chartInstance) {
initCharts(getDarkMode.value as 'default') initCharts(getDarkMode.value as 'default');
if (!chartInstance) { if (!chartInstance) return;
return
}
} }
clear && chartInstance?.clear() clear && chartInstance?.clear();
chartInstance?.setOption(unref(getOptions)) chartInstance?.setOption(unref(getOptions));
}, 30) }, 30);
}) });
} }
function resize() { function resize() {
chartInstance?.resize() chartInstance?.resize();
} }
watch( watch(
() => getDarkMode.value, () => getDarkMode.value,
(theme) => { (theme) => {
if (chartInstance) { if (chartInstance) {
chartInstance.dispose() chartInstance.dispose();
initCharts(theme as 'default') initCharts(theme as 'default');
setOptions(cacheOptions.value) setOptions(cacheOptions.value);
} }
}, }
) );
tryOnUnmounted(() => { tryOnUnmounted(() => {
if (!chartInstance) { if (!chartInstance) return;
return removeResizeFn();
} chartInstance.dispose();
removeResizeFn() chartInstance = null;
chartInstance.dispose() });
chartInstance = null
})
function getInstance(): echarts.ECharts | null { function getInstance(): echarts.ECharts | null {
if (!chartInstance) { if (!chartInstance) {
initCharts(getDarkMode.value as 'default') initCharts(getDarkMode.value as 'default');
} }
return chartInstance return chartInstance;
} }
return { return {
@ -118,5 +113,5 @@ export function useECharts(
resize, resize,
echarts, echarts,
getInstance, getInstance,
} };
} }

View File

@ -1,58 +1,61 @@
<!-- eslint-disable prettier/prettier -->
<template> <template>
<div class="h-screen flex flex-col"> <div class="h-screen flex flex-col">
<van-nav-bar v-if="getShowHeader" placeholder fixed :title="getTitle" /> <van-nav-bar v-if="getShowHeader" fixed placeholder :title="getTitle" />
<routerView class="flex-1 overflow-x-hidden"> <routerView class="flex-1 overflow-x-hidden">
<template #default="{ Component, route }"> <template #default="{ Component, route }">
<!-- <!--
keep-alive 标签的 include 属性是根据组件的 name 判断的 keep-alive 标签的 include 属性是根据组件的 name 判断的
所以 index.vue list.vue 等页面 vue 文件里一定要写上 name 所以 index.vue list.vue 等页面 vue 文件里一定要写上 name
并且与 router 路由表中使用的 name 属性 一致否则无效 并且与 router 路由表中使用的 name 属性 一致否则无效
Vue 3.3 中新引入了 defineOptions 宏声明 name 属性 本项目使用了 vite-plugin-vue-setup-extend 插件
https://gist.github.com/sxzz/3995fc7251567c7c95de35f45539b9c2 可直接在 script 标签上书写 name
<script setup lang="ts" name="DashboardPage">
--> -->
<keep-alive v-if="keepAliveComponents" :include="keepAliveComponents"> <keep-alive v-if="keepAliveComponents" :include="keepAliveComponents">
<component :is="Component" :key="route.fullPath" /> <component :is="Component" :key="route.fullPath" />
</keep-alive> </keep-alive>
<component :is="Component" v-else :key="route.fullPath" /> <component v-else :is="Component" :key="route.fullPath" />
</template> </template>
</routerView> </routerView>
<van-tabbar placeholder route fixed> <van-tabbar route class="tabbar">
<van-tabbar-item <van-tabbar-item
fixed
replace
v-for="menu in getMenus" v-for="menu in getMenus"
:key="menu.name" :key="menu.name"
replace
:to="menu.path" :to="menu.path"
:icon="(menu.meta?.icon as string)" :icon="(menu.meta?.icon as string)"
> >{{ menu.meta?.title }}
{{ menu.meta?.title }}
</van-tabbar-item> </van-tabbar-item>
</van-tabbar> </van-tabbar>
</div> </div>
</template> </template>
<script setup lang="ts"> <script setup lang="ts">
import type { ComputedRef } from 'vue' import { computed } from 'vue';
import { computed } from 'vue' import { useRoute } from 'vue-router';
import { useRoute } from 'vue-router' import { useRouteStore } from '@/store/modules/route';
import type { RouteRecordRaw } from 'vue-router'
import { useRouteStore } from '@/store/modules/route'
const routeStore = useRouteStore() const routeStore = useRouteStore();
// //
const keepAliveComponents = computed(() => routeStore.keepAliveComponents) const keepAliveComponents = computed(() => routeStore.keepAliveComponents);
const currentRoute = useRoute() const currentRoute = useRoute();
const getTitle = computed(() => currentRoute.meta.title as string) const getTitle = computed(() => currentRoute.meta.title as string);
// //
const getMenus: ComputedRef<RouteRecordRaw[]> = computed(() => const getMenus = computed(() =>
routeStore.menus.filter((item) => { routeStore.menus.filter((item) => {
return !item.meta?.innerPage return !item.meta?.innerPage;
}), })
) );
const getShowHeader = computed(() => !currentRoute.meta.hiddenHeader) const getShowHeader = computed(() => !currentRoute.meta.hiddenHeader);
</script> </script>
<style scoped lang="less"></style> <style scoped lang="less">
.tabbar {
position: fixed;
width: 100%;
}
</style>

View File

@ -1,31 +1,28 @@
import 'virtual:uno.css' import 'virtual:windi.css';
import 'vant/es/toast/style' import 'vant/es/toast/style';
import 'vant/es/dialog/style' import 'vant/es/dialog/style';
import '@unocss/reset/tailwind.css'
import '@unocss/reset/tailwind-compat.css'
// Register icon sprite // Register icon sprite
import 'virtual:svg-icons-register' import 'virtual:svg-icons-register';
import { createApp } from 'vue' import { createApp } from 'vue';
import App from './App.vue' import App from './App.vue';
import router, { setupRouter } from './router' import { setupStore } from '@/store';
import { updateDarkSign } from './theme' import router, { setupRouter } from './router';
import { setupStore } from '@/store' import { updateDarkSign } from './theme';
async function bootstrap() { async function bootstrap() {
const app = createApp(App) const app = createApp(App);
// 挂载状态管理 // 挂载状态管理
setupStore(app) setupStore(app);
// 挂载路由 // 挂载路由
setupRouter(app) setupRouter(app);
await router.isReady() await router.isReady();
// 路由准备就绪后挂载APP实例 // 路由准备就绪后挂载APP实例
app.mount('#app', true) app.mount('#app', true);
// 根节点挂载 dark 标识 // 根节点挂载 dark 标识
const appDesignSetting = window.localStorage.getItem('DESIGN-SETTING') const appDesignSetting = window.localStorage.getItem('DESIGN-SETTING');
const darkMode = appDesignSetting && JSON.parse(appDesignSetting).darkMode const darkMode = appDesignSetting && JSON.parse(appDesignSetting).darkMode;
updateDarkSign(darkMode) updateDarkSign(darkMode);
} }
void bootstrap() void bootstrap();

View File

@ -1,7 +1,7 @@
import type { RouteRecordRaw } from 'vue-router' import { RouteRecordRaw } from 'vue-router';
import { PageEnum } from '@/enums/pageEnum' import { PageEnum } from '@/enums/pageEnum';
const Layout = () => import('@/layout/index.vue') const Layout = () => import('@/layout/index.vue');
// 404 on a page // 404 on a page
export const ErrorPageRoute: RouteRecordRaw = { export const ErrorPageRoute: RouteRecordRaw = {
@ -23,7 +23,7 @@ export const ErrorPageRoute: RouteRecordRaw = {
}, },
}, },
], ],
} };
export const RootRoute: RouteRecordRaw = { export const RootRoute: RouteRecordRaw = {
path: '/', path: '/',
@ -32,7 +32,7 @@ export const RootRoute: RouteRecordRaw = {
meta: { meta: {
title: 'Root', title: 'Root',
}, },
} };
export const LoginRoute: RouteRecordRaw = { export const LoginRoute: RouteRecordRaw = {
path: '/login', path: '/login',
@ -41,4 +41,4 @@ export const LoginRoute: RouteRecordRaw = {
meta: { meta: {
title: '登录', title: '登录',
}, },
} };

View File

@ -1,32 +1,31 @@
import type { App } from 'vue' import { App } from 'vue';
import type { RouteRecordRaw } from 'vue-router' import { createRouter, createWebHashHistory, RouteRecordRaw } from 'vue-router';
import { createRouter, createWebHashHistory } from 'vue-router' import { LoginRoute, RootRoute, ErrorPageRoute } from '@/router/base';
import { createRouterGuards } from './router-guards' import { createRouterGuards } from './router-guards';
import routeModuleList from './modules' import { useRouteStoreWidthOut } from '@/store/modules/route';
import { ErrorPageRoute, LoginRoute, RootRoute } from '@/router/base'
import { useRouteStoreWidthOut } from '@/store/modules/route'
// 菜单 // 菜单
import routeModuleList from './modules';
// 普通路由 // 普通路由
export const constantRouter: RouteRecordRaw[] = [LoginRoute, RootRoute, ErrorPageRoute] export const constantRouter: RouteRecordRaw[] = [LoginRoute, RootRoute, ErrorPageRoute];
const routeStore = useRouteStoreWidthOut() const routeStore = useRouteStoreWidthOut();
routeStore.setMenus(routeModuleList) routeStore.setMenus(routeModuleList);
routeStore.setRouters(constantRouter.concat(routeModuleList)) routeStore.setRouters(constantRouter.concat(routeModuleList));
const router = createRouter({ const router = createRouter({
history: createWebHashHistory(''), history: createWebHashHistory(''),
routes: constantRouter.concat(...routeModuleList), routes: constantRouter.concat(...routeModuleList),
strict: true, strict: true,
scrollBehavior: () => ({ left: 0, top: 0 }), scrollBehavior: () => ({ left: 0, top: 0 }),
}) });
export function setupRouter(app: App) { export function setupRouter(app: App) {
app.use(router) app.use(router);
// 创建路由守卫 // 创建路由守卫
createRouterGuards(router) createRouterGuards(router);
} }
export default router export default router;

View File

@ -1,6 +1,6 @@
import type { RouteRecordRaw } from 'vue-router' import { RouteRecordRaw } from 'vue-router';
const Layout = () => import('@/layout/index.vue') const Layout = () => import('@/layout/index.vue');
const routeModuleList: Array<RouteRecordRaw> = [ const routeModuleList: Array<RouteRecordRaw> = [
{ {
@ -121,6 +121,6 @@ const routeModuleList: Array<RouteRecordRaw> = [
}, },
component: () => import('@/views/my/ThemeSetting.vue'), component: () => import('@/views/my/ThemeSetting.vue'),
}, },
] ];
export default routeModuleList export default routeModuleList;

View File

@ -1,91 +1,88 @@
import type { Router } from 'vue-router' import { isNavigationFailure, Router } from 'vue-router';
import { isNavigationFailure } from 'vue-router' import { useRouteStoreWidthOut } from '@/store/modules/route';
import { useRouteStoreWidthOut } from '@/store/modules/route' import { useUserStoreWidthOut } from '@/store/modules/user';
import { useUserStoreWidthOut } from '@/store/modules/user' import { ACCESS_TOKEN } from '@/store/mutation-types';
import { ACCESS_TOKEN } from '@/store/mutation-types' import { storage } from '@/utils/Storage';
import { storage } from '@/utils/Storage' import { PageEnum } from '@/enums/pageEnum';
import { PageEnum } from '@/enums/pageEnum'
const LOGIN_PATH = PageEnum.BASE_LOGIN const LOGIN_PATH = PageEnum.BASE_LOGIN;
const whitePathList = [LOGIN_PATH] // no redirect whitelist const whitePathList = [LOGIN_PATH]; // no redirect whitelist
export function createRouterGuards(router: Router) { export function createRouterGuards(router: Router) {
router.beforeEach(async (to, from, next) => { router.beforeEach(async (to, from, next) => {
// to: 即将要进入的目标 // to: 即将要进入的目标
// from: 当前导航正要离开的路由 // from: 当前导航正要离开的路由
const userStore = useUserStoreWidthOut() const userStore = useUserStoreWidthOut();
if (from.path === LOGIN_PATH && to.name === PageEnum.ERROR_PAGE_NAME) { if (from.path === LOGIN_PATH && to.name === PageEnum.ERROR_PAGE_NAME) {
next(PageEnum.BASE_HOME) next(PageEnum.BASE_HOME);
return return;
} }
// Whitelist can be directly entered // Whitelist can be directly entered
if (whitePathList.includes(to.path as PageEnum)) { if (whitePathList.includes(to.path as PageEnum)) {
next() next();
return return;
} }
const token = storage.get(ACCESS_TOKEN) const token = storage.get(ACCESS_TOKEN);
if (!token) { if (!token) {
// redirect login page // redirect login page
next(LOGIN_PATH) next(LOGIN_PATH);
return return;
} }
// 当上次更新时间为空时获取用户信息 // 当上次更新时间为空时获取用户信息
if (userStore.getLastUpdateTime === 0) { if (userStore.getLastUpdateTime === 0) {
try { try {
await userStore.GetUserInfo() await userStore.GetUserInfo();
} } catch (err) {
catch (err) { next();
next() return;
return
} }
} }
next() next();
}) });
// 进入某个路由之后触发的钩子 // 进入某个路由之后触发的钩子
router.afterEach((to, _, failure) => { router.afterEach((to, _, failure) => {
// 设置每个页面的 title // 设置每个页面的 title
document.title = (to?.meta?.title as string) || document.title document.title = (to?.meta?.title as string) || document.title;
if (isNavigationFailure(failure)) { if (isNavigationFailure(failure)) {
console.warn('failed navigation', failure) console.log('failed navigation', failure);
} }
const routeStore = useRouteStoreWidthOut() const routeStore = useRouteStoreWidthOut();
// 在这里设置需要缓存的组件名称 // 在这里设置需要缓存的组件名称
const keepAliveComponents = routeStore.keepAliveComponents const keepAliveComponents = routeStore.keepAliveComponents;
// 获取当前组件名 // 获取当前组件名
const currentComName: any = to.matched.find(item => item.name === to.name)?.name const currentComName: any = to.matched.find((item) => item.name == to.name)?.name;
// 如果 currentComName 且 keepAliveComponents 不包含 currentComName 且 即将要进入的路由 meta 属性里 keepAlive 为 true则缓存该组件 // 如果 currentComName 且 keepAliveComponents 不包含 currentComName 且 即将要进入的路由 meta 属性里 keepAlive 为 true则缓存该组件
if (currentComName && !keepAliveComponents.includes(currentComName) && to.meta?.keepAlive) { if (currentComName && !keepAliveComponents.includes(currentComName) && to.meta?.keepAlive) {
// 需要缓存的组件 // 需要缓存的组件
keepAliveComponents.push(currentComName) keepAliveComponents.push(currentComName);
// keepAlive 为 false 则不缓存 // keepAlive 为 false 则不缓存
} } else if (!to.meta?.keepAlive) {
else if (!to.meta?.keepAlive) {
// 不需要缓存的组件 // 不需要缓存的组件
// 这里的作用一开始组件设置为缓存,之后又设置不缓存但是它还是存在 keepAliveComponents 数组中 // 这里的作用一开始组件设置为缓存,之后又设置不缓存但是它还是存在 keepAliveComponents 数组中
// keepAliveComponents 使用 findIndex 与 当前路由对比,如果存在则返回具体下标位置,不存在返回 -1 // keepAliveComponents 使用 findIndex 与 当前路由对比,如果存在则返回具体下标位置,不存在返回 -1
const index = routeStore.keepAliveComponents.findIndex(name => name === currentComName) const index = routeStore.keepAliveComponents.findIndex((name) => name == currentComName);
if (index !== -1) { if (index != -1) {
// 通过返回具体下标位置删除 keepAliveComponents 数组中缓存的 元素 // 通过返回具体下标位置删除 keepAliveComponents 数组中缓存的 元素
keepAliveComponents.splice(index, 1) keepAliveComponents.splice(index, 1);
} }
} }
routeStore.setKeepAliveComponents(keepAliveComponents) routeStore.setKeepAliveComponents(keepAliveComponents);
}) });
router.onError((error) => { router.onError((error) => {
console.error(error, '路由错误') console.error(error, '路由错误');
}) });
} }

View File

@ -5,4 +5,4 @@ export const animates = [
{ value: 'fade', text: '消退' }, { value: 'fade', text: '消退' },
{ value: 'fade-bottom', text: '底部消退' }, { value: 'fade-bottom', text: '底部消退' },
{ value: 'fade-scale', text: '缩放消退' }, { value: 'fade-scale', text: '缩放消退' },
] ];

View File

@ -1,15 +1,15 @@
export default { export default {
upload: { upload: {
// 考虑接口规范不同 //考虑接口规范不同
apiSetting: { apiSetting: {
// 集合字段名 // 集合字段名
infoField: 'result', infoField: 'result',
// 图片地址字段名 // 图片地址字段名
imgField: 'imagePath', imgField: 'imagePath',
}, },
// 最大上传图片大小 //最大上传图片大小
maxSize: 1, maxSize: 1,
// 图片上传类型 //图片上传类型
fileType: ['image/png', 'image/jpg', 'image/jpeg', 'image/gif', 'image/svg+xml'], fileType: ['image/png', 'image/jpg', 'image/jpeg', 'image/gif', 'image/svg+xml'],
}, },
} };

View File

@ -2,15 +2,15 @@
export interface DesignSettingState { export interface DesignSettingState {
// 系统主题 // 系统主题
darkMode: 'light' | 'dark' darkMode: 'light' | 'dark';
// 系统风格 // 系统风格
appTheme: string appTheme: string;
// 系统内置风格 // 系统内置风格
appThemeList: string[] appThemeList: string[];
// 是否开启路由动画 // 是否开启路由动画
isPageAnimate: boolean isPageAnimate: boolean;
// 路由动画类型 // 路由动画类型
pageAnimateType: string pageAnimateType: string;
} }
export const appThemeList: string[] = [ export const appThemeList: string[] = [
@ -33,19 +33,19 @@ export const appThemeList: string[] = [
'#FB9300', '#FB9300',
'#FC5404', '#FC5404',
'#8675ff', '#8675ff',
] ];
const setting: DesignSettingState = { const setting: DesignSettingState = {
// 深色主题 //深色主题
darkMode: 'light', darkMode: 'light',
// 系统主题色 //系统主题色
appTheme: '#5d9dfe', appTheme: '#5d9dfe',
// 系统内置主题色列表 //系统内置主题色列表
appThemeList, appThemeList,
// 是否开启路由动画 //是否开启路由动画
isPageAnimate: true, isPageAnimate: true,
// 路由动画类型 //路由动画类型
pageAnimateType: 'zoom-fade', pageAnimateType: 'zoom-fade',
} };
export default setting export default setting;

View File

@ -1,12 +1,12 @@
import type { App } from 'vue' import type { App } from 'vue';
import { createPinia } from 'pinia' import { createPinia } from 'pinia';
import piniaPersist from 'pinia-plugin-persist' import piniaPersist from 'pinia-plugin-persist';
const store = createPinia() const store = createPinia();
store.use(piniaPersist) store.use(piniaPersist);
export function setupStore(app: App<Element>) { export function setupStore(app: App<Element>) {
app.use(store) app.use(store);
} }
export { store } export { store };

View File

@ -1,9 +1,9 @@
import { defineStore } from 'pinia' import { defineStore } from 'pinia';
import { store } from '@/store' import { store } from '@/store';
import designSetting from '@/settings/designSetting' import designSetting from '@/settings/designSetting';
import type { DesignSettingState } from '@/settings/designSetting' import type { DesignSettingState } from '@/settings/designSetting';
const { darkMode, appTheme, appThemeList, isPageAnimate, pageAnimateType } = designSetting const { darkMode, appTheme, appThemeList, isPageAnimate, pageAnimateType } = designSetting;
export const useDesignSettingStore = defineStore({ export const useDesignSettingStore = defineStore({
id: 'app-design-setting', id: 'app-design-setting',
@ -16,27 +16,27 @@ export const useDesignSettingStore = defineStore({
}), }),
getters: { getters: {
getDarkMode(): 'light' | 'dark' { getDarkMode(): 'light' | 'dark' {
return this.darkMode return this.darkMode;
}, },
getAppTheme(): string { getAppTheme(): string {
return this.appTheme return this.appTheme;
}, },
getAppThemeList(): string[] { getAppThemeList(): string[] {
return this.appThemeList return this.appThemeList;
}, },
getIsPageAnimate(): boolean { getIsPageAnimate(): boolean {
return this.isPageAnimate return this.isPageAnimate;
}, },
getPageAnimateType(): string { getPageAnimateType(): string {
return this.pageAnimateType return this.pageAnimateType;
}, },
}, },
actions: { actions: {
setDarkMode(mode: 'light' | 'dark'): void { setDarkMode(mode: 'light' | 'dark'): void {
this.darkMode = mode this.darkMode = mode;
}, },
setPageAnimateType(type: string): void { setPageAnimateType(type: string): void {
this.pageAnimateType = type this.pageAnimateType = type;
}, },
}, },
// 持久化 // 持久化
@ -49,9 +49,9 @@ export const useDesignSettingStore = defineStore({
}, },
], ],
}, },
}) });
// Need to be used outside the setup // Need to be used outside the setup
export function useDesignSettingWithOut() { export function useDesignSettingWithOut() {
return useDesignSettingStore(store) return useDesignSettingStore(store);
} }

View File

@ -1,11 +1,11 @@
import { defineStore } from 'pinia' import { defineStore } from 'pinia';
import type { RouteRecordRaw } from 'vue-router' import { RouteRecordRaw } from 'vue-router';
import { store } from '@/store' import { store } from '@/store';
export interface IRouteState { export interface IRouteState {
menus: RouteRecordRaw[] menus: RouteRecordRaw[];
routers: RouteRecordRaw[] routers: RouteRecordRaw[];
keepAliveComponents: string[] keepAliveComponents: string[];
} }
export const useRouteStore = defineStore({ export const useRouteStore = defineStore({
@ -17,24 +17,24 @@ export const useRouteStore = defineStore({
}), }),
getters: { getters: {
getMenus(): RouteRecordRaw[] { getMenus(): RouteRecordRaw[] {
return this.menus return this.menus;
}, },
}, },
actions: { actions: {
setRouters(routers: RouteRecordRaw[]) { setRouters(routers: RouteRecordRaw[]) {
this.routers = routers this.routers = routers;
}, },
setMenus(menus: RouteRecordRaw[]) { setMenus(menus: RouteRecordRaw[]) {
this.menus = menus this.menus = menus;
}, },
setKeepAliveComponents(compNames: string[]) { setKeepAliveComponents(compNames: string[]) {
// 设置需要缓存的组件 // 设置需要缓存的组件
this.keepAliveComponents = compNames this.keepAliveComponents = compNames;
}, },
}, },
}) });
// Need to be used outside the setup // Need to be used outside the setup
export function useRouteStoreWidthOut() { export function useRouteStoreWidthOut() {
return useRouteStore(store) return useRouteStore(store);
} }

View File

@ -1,36 +1,35 @@
import { defineStore } from 'pinia' import { defineStore } from 'pinia';
import { createStorage } from '@/utils/Storage' import { createStorage } from '@/utils/Storage';
import { store } from '@/store' import { store } from '@/store';
import { ACCESS_TOKEN, CURRENT_USER } from '@/store/mutation-types' import { ACCESS_TOKEN, CURRENT_USER } from '@/store/mutation-types';
import { ResultEnum } from '@/enums/httpEnum' import { ResultEnum } from '@/enums/httpEnum';
import { doLogout, getUserInfo, login } from '@/api/system/user' const Storage = createStorage({ storage: localStorage });
import { PageEnum } from '@/enums/pageEnum' import { getUserInfo, login, doLogout } from '@/api/system/user';
import router from '@/router' import { PageEnum } from '@/enums/pageEnum';
import router from '@/router';
const Storage = createStorage({ storage: localStorage })
interface UserInfo { interface UserInfo {
userId: string | number userId: string | number;
username: string username: string;
realname: string realname: string;
nickname: string nickname: string;
avatar: string avatar: string;
cover: string cover: string;
gender: number gender: number;
phone: string phone: string;
sign?: string sign?: string;
industry?: number industry?: number;
} }
interface IUserState { interface IUserState {
token?: string token?: string;
userInfo: Nullable<UserInfo> userInfo: Nullable<UserInfo>;
lastUpdateTime: number lastUpdateTime: number;
} }
interface LoginParams { interface LoginParams {
username: string username: string;
password: string password: string;
} }
export const useUserStore = defineStore({ export const useUserStore = defineStore({
@ -42,38 +41,37 @@ export const useUserStore = defineStore({
}), }),
getters: { getters: {
getUserInfo(): UserInfo { getUserInfo(): UserInfo {
return this.userInfo || Storage.get(CURRENT_USER, '') || {} return this.userInfo || Storage.get(CURRENT_USER, '') || {};
}, },
getToken(): string { getToken(): string {
return this.token || Storage.get(ACCESS_TOKEN, '') return this.token || Storage.get(ACCESS_TOKEN, '');
}, },
getLastUpdateTime(): number { getLastUpdateTime(): number {
return this.lastUpdateTime return this.lastUpdateTime;
}, },
}, },
actions: { actions: {
setToken(token: string | undefined) { setToken(token: string | undefined) {
this.token = token || '' this.token = token ? token : '';
Storage.set(ACCESS_TOKEN, token) Storage.set(ACCESS_TOKEN, token);
}, },
setUserInfo(info: UserInfo | null) { setUserInfo(info: UserInfo | null) {
this.userInfo = info this.userInfo = info;
this.lastUpdateTime = new Date().getTime() this.lastUpdateTime = new Date().getTime();
Storage.set(CURRENT_USER, info) Storage.set(CURRENT_USER, info);
}, },
async Login(params: LoginParams) { async Login(params: LoginParams) {
try { try {
const response = await login(params) const response = await login(params);
const { result, code } = response const { result, code } = response;
if (code === ResultEnum.SUCCESS) { if (code === ResultEnum.SUCCESS) {
// save token // save token
this.setToken(result.token) this.setToken(result.token);
} }
return Promise.resolve(response) return Promise.resolve(response);
} } catch (error) {
catch (error) { return Promise.reject(error);
return Promise.reject(error)
} }
}, },
@ -81,35 +79,34 @@ export const useUserStore = defineStore({
return new Promise((resolve, reject) => { return new Promise((resolve, reject) => {
getUserInfo() getUserInfo()
.then((res) => { .then((res) => {
this.setUserInfo(res) this.setUserInfo(res);
resolve(res) resolve(res);
}) })
.catch((error) => { .catch((error) => {
reject(error) reject(error);
}) });
}) });
}, },
async Logout() { async Logout() {
if (this.getToken) { if (this.getToken) {
try { try {
await doLogout() await doLogout();
} } catch {
catch { console.error('注销Token失败');
console.error('注销Token失败')
} }
} }
this.setToken(undefined) this.setToken(undefined);
this.setUserInfo(null) this.setUserInfo(null);
Storage.remove(ACCESS_TOKEN) Storage.remove(ACCESS_TOKEN);
Storage.remove(CURRENT_USER) Storage.remove(CURRENT_USER);
router.push(PageEnum.BASE_LOGIN) router.push(PageEnum.BASE_LOGIN);
location.reload() location.reload();
}, },
}, },
}) });
// Need to be used outside the setup // Need to be used outside the setup
export function useUserStoreWidthOut() { export function useUserStoreWidthOut() {
return useUserStore(store) return useUserStore(store);
} }

View File

@ -1,4 +1,4 @@
export const FIRST_VISIT = 'FIRST-VISIT' // 是否首次访问 export const FIRST_VISIT = 'FIRST-VISIT'; // 是否首次访问
export const ACCESS_TOKEN = 'ACCESS-TOKEN' // 用户token export const ACCESS_TOKEN = 'ACCESS-TOKEN'; // 用户token
export const CURRENT_USER = 'CURRENT-USER' // 当前用户信息 export const CURRENT_USER = 'CURRENT-USER'; // 当前用户信息
export const DESIGN_SETTING = 'DESIGN-SETTING' // 当前用户主题信息 export const DESIGN_SETTING = 'DESIGN-SETTING'; // 当前用户主题信息

View File

@ -11,6 +11,7 @@ html {
} }
[data-theme='dark'] { [data-theme='dark'] {
&, &,
* { * {
color-scheme: dark !important; color-scheme: dark !important;
@ -23,6 +24,7 @@ html {
} }
[data-theme='light'] { [data-theme='light'] {
&, &,
* { * {
color-scheme: light !important; color-scheme: light !important;
@ -78,9 +80,7 @@ a:hover {
.zoom-fade-enter-active, .zoom-fade-enter-active,
.zoom-fade-leave-active { .zoom-fade-leave-active {
transition: transition: transform 0.35s, opacity 0.28s ease-in-out;
transform 0.35s,
opacity 0.28s ease-in-out;
} }
.zoom-fade-enter-from { .zoom-fade-enter-from {

View File

@ -1,168 +0,0 @@
* > .enter-x:nth-child(1) {
transform: translateX(50px);
}
* > .-enter-x:nth-child(1) {
transform: translateX(-50px);
}
* > .enter-x:nth-child(1),
* > .-enter-x:nth-child(1) {
z-index: 9;
opacity: 0;
animation: enter-x-animation 0.4s ease-in-out 0.3s;
animation-fill-mode: forwards;
animation-delay: 0.1s;
}
* > .enter-x:nth-child(2) {
transform: translateX(50px);
}
* > .-enter-x:nth-child(2) {
transform: translateX(-50px);
}
* > .enter-x:nth-child(2),
* > .-enter-x:nth-child(2) {
z-index: 8;
opacity: 0;
animation: enter-x-animation 0.4s ease-in-out 0.3s;
animation-fill-mode: forwards;
animation-delay: 0.2s;
}
* > .enter-x:nth-child(3) {
transform: translateX(50px);
}
* > .-enter-x:nth-child(3) {
transform: translateX(-50px);
}
* > .enter-x:nth-child(3),
* > .-enter-x:nth-child(3) {
z-index: 7;
opacity: 0;
animation: enter-x-animation 0.4s ease-in-out 0.3s;
animation-fill-mode: forwards;
animation-delay: 0.3s;
}
* > .enter-x:nth-child(4) {
transform: translateX(50px);
}
* > .-enter-x:nth-child(4) {
transform: translateX(-50px);
}
* > .enter-x:nth-child(4),
* > .-enter-x:nth-child(4) {
z-index: 6;
opacity: 0;
animation: enter-x-animation 0.4s ease-in-out 0.3s;
animation-fill-mode: forwards;
animation-delay: 0.4s;
}
* > .enter-x:nth-child(5) {
transform: translateX(50px);
}
* > .-enter-x:nth-child(5) {
transform: translateX(-50px);
}
* > .enter-x:nth-child(5),
* > .-enter-x:nth-child(5) {
z-index: 5;
opacity: 0;
animation: enter-x-animation 0.4s ease-in-out 0.3s;
animation-fill-mode: forwards;
animation-delay: 0.5s;
}
* > .enter-y:nth-child(1) {
transform: translateX(50px);
}
* > .-enter-y:nth-child(1) {
transform: translateX(-50px);
}
* > .enter-y:nth-child(1),
* > .-enter-y:nth-child(1) {
z-index: 9;
opacity: 0;
animation: enter-y-animation 0.4s ease-in-out 0.3s;
animation-fill-mode: forwards;
animation-delay: 0.1s;
}
* > .enter-y:nth-child(2) {
transform: translateX(50px);
}
* > .-enter-y:nth-child(2) {
transform: translateX(-50px);
}
* > .enter-y:nth-child(2),
* > .-enter-y:nth-child(2) {
z-index: 8;
opacity: 0;
animation: enter-y-animation 0.4s ease-in-out 0.3s;
animation-fill-mode: forwards;
animation-delay: 0.2s;
}
* > .enter-y:nth-child(3) {
transform: translateX(50px);
}
* > .-enter-y:nth-child(3) {
transform: translateX(-50px);
}
* > .enter-y:nth-child(3),
* > .-enter-y:nth-child(3) {
z-index: 7;
opacity: 0;
animation: enter-y-animation 0.4s ease-in-out 0.3s;
animation-fill-mode: forwards;
animation-delay: 0.3s;
}
* > .enter-y:nth-child(4) {
transform: translateX(50px);
}
* > .-enter-y:nth-child(4) {
transform: translateX(-50px);
}
* > .enter-y:nth-child(4),
* > .-enter-y:nth-child(4) {
z-index: 6;
opacity: 0;
animation: enter-y-animation 0.4s ease-in-out 0.3s;
animation-fill-mode: forwards;
animation-delay: 0.4s;
}
* > .enter-y:nth-child(5) {
transform: translateX(50px);
}
* > .-enter-y:nth-child(5) {
transform: translateX(-50px);
}
* > .enter-y:nth-child(5),
* > .-enter-y:nth-child(5) {
z-index: 5;
opacity: 0;
animation: enter-y-animation 0.4s ease-in-out 0.3s;
animation-fill-mode: forwards;
animation-delay: 0.5s;
}
@keyframes enter-x-animation {
to {
opacity: 1;
transform: translateX(0);
}
}
@keyframes enter-y-animation {
to {
opacity: 1;
transform: translateY(0);
}
}

View File

@ -1,4 +1,3 @@
@import './common.less'; @import './common.less';
@import 'transition/index.less'; @import 'transition/index.less';
@import 'entry.css';
@import './vant.less'; @import './vant.less';

View File

@ -31,9 +31,7 @@
// Speed: 1x // Speed: 1x
.fade-bottom-enter-active, .fade-bottom-enter-active,
.fade-bottom-leave-active { .fade-bottom-leave-active {
transition: transition: opacity 0.25s, transform 0.3s;
opacity 0.25s,
transform 0.3s;
} }
.fade-bottom-enter-from { .fade-bottom-enter-from {
@ -69,9 +67,7 @@
// Speed: 1x // Speed: 1x
.fade-top-enter-active, .fade-top-enter-active,
.fade-top-leave-active { .fade-top-leave-active {
transition: transition: opacity 0.2s, transform 0.25s;
opacity 0.2s,
transform 0.25s;
} }
.fade-top-enter-from { .fade-top-enter-from {

View File

@ -6,8 +6,5 @@
@import './zoom.less'; @import './zoom.less';
.collapse-transition { .collapse-transition {
transition: transition: 0.2s height ease-in-out, 0.2s padding-top ease-in-out, 0.2s padding-bottom ease-in-out;
0.2s height ease-in-out,
0.2s padding-top ease-in-out,
0.2s padding-bottom ease-in-out;
} }

View File

@ -1,9 +1,7 @@
// zoom-out // zoom-out
.zoom-out-enter-active, .zoom-out-enter-active,
.zoom-out-leave-active { .zoom-out-leave-active {
transition: transition: opacity 0.1 ease-in-out, transform 0.15s ease-out;
opacity 0.1 ease-in-out,
transform 0.15s ease-out;
} }
.zoom-out-enter-from, .zoom-out-enter-from,
@ -15,9 +13,7 @@
// zoom-fade // zoom-fade
.zoom-fade-enter-active, .zoom-fade-enter-active,
.zoom-fade-leave-active { .zoom-fade-leave-active {
transition: transition: transform 0.2s, opacity 0.3s ease-out;
transform 0.2s,
opacity 0.3s ease-out;
} }
.zoom-fade-enter-from { .zoom-fade-enter-from {

View File

@ -1,25 +1,24 @@
import { addClass, hasClass, removeClass } from '@/utils/domUtils' import { addClass, removeClass, hasClass } from '@/utils/domUtils';
/** /**
* html / * html /
*/ */
export function updateDarkSign(mode: 'light' | 'dark') { export function updateDarkSign(mode: 'light' | 'dark') {
const htmlRoot = document.getElementById('htmlRoot') const htmlRoot = document.getElementById('htmlRoot');
if (!htmlRoot) { if (!htmlRoot) {
return return;
} }
const hasDarkClass = hasClass(htmlRoot, 'dark') const hasDarkClass = hasClass(htmlRoot, 'dark');
if (mode === 'dark') { if (mode === 'dark') {
htmlRoot.setAttribute('data-theme', 'dark') htmlRoot.setAttribute('data-theme', 'dark');
if (!hasDarkClass) { if (!hasDarkClass) {
addClass(htmlRoot, 'dark') addClass(htmlRoot, 'dark');
} }
} } else {
else { htmlRoot.setAttribute('data-theme', 'light');
htmlRoot.setAttribute('data-theme', 'light')
if (hasDarkClass) { if (hasDarkClass) {
removeClass(htmlRoot, 'dark') removeClass(htmlRoot, 'dark');
} }
} }
} }

View File

@ -1,21 +1,22 @@
// 默认缓存期限为7天 // 默认缓存期限为7天
const DEFAULT_CACHE_TIME = 60 * 60 * 24 * 7 const DEFAULT_CACHE_TIME = 60 * 60 * 24 * 7;
/** /**
* *
* @param {string} prefixKey - * @param {string=} prefixKey -
* @param {Object} [storage=localStorage] - sessionStorage | localStorage
*/ */
export function createStorage({ prefixKey = '', storage = localStorage } = {}) { export const createStorage = ({ prefixKey = '', storage = localStorage } = {}) => {
/** /**
* *
* @class Storage * @class Storage
*/ */
const Storage = class { const Storage = class {
private storage = storage private storage = storage;
private prefixKey?: string = prefixKey private prefixKey?: string = prefixKey;
private getKey(key: string) { private getKey(key: string) {
return `${this.prefixKey}${key}`.toUpperCase() return `${this.prefixKey}${key}`.toUpperCase();
} }
/** /**
@ -28,8 +29,8 @@ export function createStorage({ prefixKey = '', storage = localStorage } = {}) {
const stringData = JSON.stringify({ const stringData = JSON.stringify({
value, value,
expire: expire !== null ? new Date().getTime() + expire * 1000 : null, expire: expire !== null ? new Date().getTime() + expire * 1000 : null,
}) });
this.storage.setItem(this.getKey(key), stringData) this.storage.setItem(this.getKey(key), stringData);
} }
/** /**
@ -38,22 +39,21 @@ export function createStorage({ prefixKey = '', storage = localStorage } = {}) {
* @param {*=} def * @param {*=} def
*/ */
get(key: string, def: any = null) { get(key: string, def: any = null) {
const item = this.storage.getItem(this.getKey(key)) const item = this.storage.getItem(this.getKey(key));
if (item) { if (item) {
try { try {
const data = JSON.parse(item) const data = JSON.parse(item);
const { value, expire } = data const { value, expire } = data;
// 在有效期内直接返回 // 在有效期内直接返回
if (expire === null || expire >= Date.now()) { if (expire === null || expire >= Date.now()) {
return value return value;
} }
this.remove(key) this.remove(key);
} } catch (e) {
catch (e) { return def;
return def
} }
} }
return def return def;
} }
/** /**
@ -61,7 +61,7 @@ export function createStorage({ prefixKey = '', storage = localStorage } = {}) {
* @param {string} key * @param {string} key
*/ */
remove(key: string) { remove(key: string) {
this.storage.removeItem(this.getKey(key)) this.storage.removeItem(this.getKey(key));
} }
/** /**
@ -69,7 +69,7 @@ export function createStorage({ prefixKey = '', storage = localStorage } = {}) {
* @memberOf Cache * @memberOf Cache
*/ */
clear(): void { clear(): void {
this.storage.clear() this.storage.clear();
} }
/** /**
@ -81,7 +81,7 @@ export function createStorage({ prefixKey = '', storage = localStorage } = {}) {
* @example * @example
*/ */
setCookie(name: string, value: any, expire: number | null = DEFAULT_CACHE_TIME) { setCookie(name: string, value: any, expire: number | null = DEFAULT_CACHE_TIME) {
document.cookie = `${this.getKey(name)}=${value}; Max-Age=${expire}` document.cookie = `${this.getKey(name)}=${value}; Max-Age=${expire}`;
} }
/** /**
@ -89,14 +89,14 @@ export function createStorage({ prefixKey = '', storage = localStorage } = {}) {
* @param name * @param name
*/ */
getCookie(name: string): string { getCookie(name: string): string {
const cookieArr = document.cookie.split('; ') const cookieArr = document.cookie.split('; ');
for (let i = 0, length = cookieArr.length; i < length; i++) { for (let i = 0, length = cookieArr.length; i < length; i++) {
const kv = cookieArr[i].split('=') const kv = cookieArr[i].split('=');
if (kv[0] === this.getKey(name)) { if (kv[0] === this.getKey(name)) {
return kv[1] return kv[1];
} }
} }
return '' return '';
} }
/** /**
@ -104,24 +104,24 @@ export function createStorage({ prefixKey = '', storage = localStorage } = {}) {
* @param {string} key * @param {string} key
*/ */
removeCookie(key: string) { removeCookie(key: string) {
this.setCookie(key, 1, -1) this.setCookie(key, 1, -1);
} }
/** /**
* cookie使cookie失效 * cookie使cookie失效
*/ */
clearCookie(): void { clearCookie(): void {
const keys = document.cookie.match(/[^ =;]+(?==)/g) const keys = document.cookie.match(/[^ =;]+(?==)/g);
if (keys) { if (keys) {
for (let i = keys.length; i--;) { for (let i = keys.length; i--; ) {
document.cookie = `${keys[i]}=0;expire=${new Date(0).toUTCString()}` document.cookie = keys[i] + '=0;expire=' + new Date(0).toUTCString();
} }
} }
} }
} };
return new Storage() return new Storage();
} };
export const storage = createStorage() export const storage = createStorage();
export default Storage export default Storage;

View File

@ -1,12 +1,12 @@
import { format } from 'date-fns' import { format } from 'date-fns';
const DATE_TIME_FORMAT = 'yyyy-MM-dd HH:mm:ss' const DATE_TIME_FORMAT = 'yyyy-MM-dd HH:mm:ss';
const DATE_FORMAT = 'YYYY-MM-DD ' const DATE_FORMAT = 'YYYY-MM-DD ';
export function formatToDateTime(date: number | Date, formatStr = DATE_TIME_FORMAT): string { export function formatToDateTime(date: number | Date, formatStr = DATE_TIME_FORMAT): string {
return format(date, formatStr) return format(date, formatStr);
} }
export function formatToDate(date: number | Date, formatStr = DATE_FORMAT): string { export function formatToDate(date: number | Date, formatStr = DATE_FORMAT): string {
return format(date, formatStr) return format(date, formatStr);
} }

View File

@ -1,92 +1,76 @@
/* eslint-disable ts/ban-ts-comment */ import type { FunctionArgs } from '@vueuse/core';
import type { FunctionArgs } from '@vueuse/core' import { upperFirst } from 'lodash-es';
import { upperFirst } from 'lodash-es'
export interface ViewportOffsetResult { export interface ViewportOffsetResult {
left: number left: number;
top: number top: number;
right: number right: number;
bottom: number bottom: number;
rightIncludeBody: number rightIncludeBody: number;
bottomIncludeBody: number bottomIncludeBody: number;
} }
export function getBoundingClientRect(element: Element): DOMRect | number { export function getBoundingClientRect(element: Element): DOMRect | number {
if (!element || !element.getBoundingClientRect) { if (!element || !element.getBoundingClientRect) {
return 0 return 0;
} }
return element.getBoundingClientRect() return element.getBoundingClientRect();
} }
function trim(string: string) { function trim(string: string) {
return (string || '').replace(/^[\s\uFEFF]+|[\s\uFEFF]+$/g, '') return (string || '').replace(/^[\s\uFEFF]+|[\s\uFEFF]+$/g, '');
} }
/* istanbul ignore next */ /* istanbul ignore next */
export function hasClass(el: Element, cls: string) { export function hasClass(el: Element, cls: string) {
if (!el || !cls) { if (!el || !cls) return false;
return false if (cls.indexOf(' ') !== -1) throw new Error('className should not contain space.');
}
if (cls.includes(' ')) {
throw new Error('className should not contain space.')
}
if (el.classList) { if (el.classList) {
return el.classList.contains(cls) return el.classList.contains(cls);
} } else {
else { return (' ' + el.className + ' ').indexOf(' ' + cls + ' ') > -1;
return (` ${el.className} `).includes(` ${cls} `)
} }
} }
/* istanbul ignore next */ /* istanbul ignore next */
export function addClass(el: Element, cls: string) { export function addClass(el: Element, cls: string) {
if (!el) { if (!el) return;
return let curClass = el.className;
} const classes = (cls || '').split(' ');
let curClass = el.className
const classes = (cls || '').split(' ')
for (let i = 0, j = classes.length; i < j; i++) { for (let i = 0, j = classes.length; i < j; i++) {
const clsName = classes[i] const clsName = classes[i];
if (!clsName) { if (!clsName) continue;
continue
}
if (el.classList) { if (el.classList) {
el.classList.add(clsName) el.classList.add(clsName);
} } else if (!hasClass(el, clsName)) {
else if (!hasClass(el, clsName)) { curClass += ' ' + clsName;
curClass += ` ${clsName}`
} }
} }
if (!el.classList) { if (!el.classList) {
el.className = curClass el.className = curClass;
} }
} }
/* istanbul ignore next */ /* istanbul ignore next */
export function removeClass(el: Element, cls: string) { export function removeClass(el: Element, cls: string) {
if (!el || !cls) { if (!el || !cls) return;
return const classes = cls.split(' ');
} let curClass = ' ' + el.className + ' ';
const classes = cls.split(' ')
let curClass = ` ${el.className} `
for (let i = 0, j = classes.length; i < j; i++) { for (let i = 0, j = classes.length; i < j; i++) {
const clsName = classes[i] const clsName = classes[i];
if (!clsName) { if (!clsName) continue;
continue
}
if (el.classList) { if (el.classList) {
el.classList.remove(clsName) el.classList.remove(clsName);
} } else if (hasClass(el, clsName)) {
else if (hasClass(el, clsName)) { curClass = curClass.replace(' ' + clsName + ' ', ' ');
curClass = curClass.replace(` ${clsName} `, ' ')
} }
} }
if (!el.classList) { if (!el.classList) {
el.className = trim(curClass) el.className = trim(curClass);
} }
} }
/** /**
@ -101,61 +85,61 @@ export function removeClass(el: Element, cls: string) {
* @description: * @description:
*/ */
export function getViewportOffset(element: Element): ViewportOffsetResult { export function getViewportOffset(element: Element): ViewportOffsetResult {
const doc = document.documentElement const doc = document.documentElement;
const docScrollLeft = doc.scrollLeft const docScrollLeft = doc.scrollLeft;
const docScrollTop = doc.scrollTop const docScrollTop = doc.scrollTop;
const docClientLeft = doc.clientLeft const docClientLeft = doc.clientLeft;
const docClientTop = doc.clientTop const docClientTop = doc.clientTop;
const pageXOffset = window.pageXOffset const pageXOffset = window.pageXOffset;
const pageYOffset = window.pageYOffset const pageYOffset = window.pageYOffset;
const box = getBoundingClientRect(element) const box = getBoundingClientRect(element);
const { left: retLeft, top: rectTop, width: rectWidth, height: rectHeight } = box as DOMRect const { left: retLeft, top: rectTop, width: rectWidth, height: rectHeight } = box as DOMRect;
const scrollLeft = (pageXOffset || docScrollLeft) - (docClientLeft || 0) const scrollLeft = (pageXOffset || docScrollLeft) - (docClientLeft || 0);
const scrollTop = (pageYOffset || docScrollTop) - (docClientTop || 0) const scrollTop = (pageYOffset || docScrollTop) - (docClientTop || 0);
const offsetLeft = retLeft + pageXOffset const offsetLeft = retLeft + pageXOffset;
const offsetTop = rectTop + pageYOffset const offsetTop = rectTop + pageYOffset;
const left = offsetLeft - scrollLeft const left = offsetLeft - scrollLeft;
const top = offsetTop - scrollTop const top = offsetTop - scrollTop;
const clientWidth = window.document.documentElement.clientWidth const clientWidth = window.document.documentElement.clientWidth;
const clientHeight = window.document.documentElement.clientHeight const clientHeight = window.document.documentElement.clientHeight;
return { return {
left, left: left,
top, top: top,
right: clientWidth - rectWidth - left, right: clientWidth - rectWidth - left,
bottom: clientHeight - rectHeight - top, bottom: clientHeight - rectHeight - top,
rightIncludeBody: clientWidth - left, rightIncludeBody: clientWidth - left,
bottomIncludeBody: clientHeight - top, bottomIncludeBody: clientHeight - top,
} };
} }
export function hackCss(attr: string, value: string) { export function hackCss(attr: string, value: string) {
const prefix: string[] = ['webkit', 'Moz', 'ms', 'OT'] const prefix: string[] = ['webkit', 'Moz', 'ms', 'OT'];
const styleObj: any = {} const styleObj: any = {};
prefix.forEach((item) => { prefix.forEach((item) => {
styleObj[`${item}${upperFirst(attr)}`] = value styleObj[`${item}${upperFirst(attr)}`] = value;
}) });
return { return {
...styleObj, ...styleObj,
[attr]: value, [attr]: value,
} };
} }
/* istanbul ignore next */ /* istanbul ignore next */
export function on( export function on(
element: Element | HTMLElement | Document | Window, element: Element | HTMLElement | Document | Window,
event: string, event: string,
handler: EventListenerOrEventListenerObject, handler: EventListenerOrEventListenerObject
): void { ): void {
if (element && event && handler) { if (element && event && handler) {
element.addEventListener(event, handler, false) element.addEventListener(event, handler, false);
} }
} }
@ -163,10 +147,10 @@ export function on(
export function off( export function off(
element: Element | HTMLElement | Document | Window, element: Element | HTMLElement | Document | Window,
event: string, event: string,
handler: Fn, handler: Fn
): void { ): void {
if (element && event && handler) { if (element && event && handler) {
element.removeEventListener(event, handler, false) element.removeEventListener(event, handler, false);
} }
} }
@ -174,26 +158,24 @@ export function off(
export function once(el: HTMLElement, event: string, fn: EventListener): void { export function once(el: HTMLElement, event: string, fn: EventListener): void {
const listener = function (this: any, ...args: unknown[]) { const listener = function (this: any, ...args: unknown[]) {
if (fn) { if (fn) {
// @ts-expect-error // @ts-ignore
fn.apply(this, args) fn.apply(this, args);
} }
off(el, event, listener) off(el, event, listener);
} };
on(el, event, listener) on(el, event, listener);
} }
export function useRafThrottle<T extends FunctionArgs>(fn: T): T { export function useRafThrottle<T extends FunctionArgs>(fn: T): T {
let locked = false let locked = false;
// @ts-expect-error // @ts-ignore
return function (...args: any[]) { return function (...args: any[]) {
if (locked) { if (locked) return;
return locked = true;
}
locked = true
window.requestAnimationFrame(() => { window.requestAnimationFrame(() => {
// @ts-expect-error // @ts-ignore
fn.apply(this, args) fn.apply(this, args);
locked = false locked = false;
}) });
} };
} }

View File

@ -1,26 +1,26 @@
import pkg from '../../package.json' import type { GlobEnvConfig } from '#/config';
import { getConfigFileName } from '../../build/getConfigFileName'
import type { GlobEnvConfig } from '#/config'
import { warn } from '@/utils/log' import { warn } from '@/utils/log';
import pkg from '../../package.json';
import { getConfigFileName } from '../../build/getConfigFileName';
export function getCommonStoragePrefix() { export function getCommonStoragePrefix() {
const { VITE_GLOB_APP_SHORT_NAME } = getAppEnvConfig() const { VITE_GLOB_APP_SHORT_NAME } = getAppEnvConfig();
return `${VITE_GLOB_APP_SHORT_NAME}__${getEnv()}`.toUpperCase() return `${VITE_GLOB_APP_SHORT_NAME}__${getEnv()}`.toUpperCase();
} }
// Generate cache key according to version // Generate cache key according to version
export function getStorageShortName() { export function getStorageShortName() {
return `${getCommonStoragePrefix()}${`__${pkg.version}`}__`.toUpperCase() return `${getCommonStoragePrefix()}${`__${pkg.version}`}__`.toUpperCase();
} }
export function getAppEnvConfig() { export function getAppEnvConfig() {
const ENV_NAME = getConfigFileName(import.meta.env) const ENV_NAME = getConfigFileName(import.meta.env);
// Get the global configuration (the configuration will be extracted independently when packaging)
const ENV = (import.meta.env.DEV const ENV = (import.meta.env.DEV
? (import.meta.env as unknown as GlobEnvConfig) ? // Get the global configuration (the configuration will be extracted independently when packaging)
: window[ENV_NAME as any]) as unknown as GlobEnvConfig (import.meta.env as unknown as GlobEnvConfig)
: window[ENV_NAME as any]) as unknown as GlobEnvConfig;
const { const {
VITE_GLOB_APP_TITLE, VITE_GLOB_APP_TITLE,
@ -31,12 +31,12 @@ export function getAppEnvConfig() {
VITE_GLOB_UPLOAD_URL, VITE_GLOB_UPLOAD_URL,
VITE_GLOB_PROD_MOCK, VITE_GLOB_PROD_MOCK,
VITE_GLOB_IMG_URL, VITE_GLOB_IMG_URL,
} = ENV } = ENV;
if (!/^[a-zA-Z\_]*$/.test(VITE_GLOB_APP_SHORT_NAME)) { if (!/^[a-zA-Z\_]*$/.test(VITE_GLOB_APP_SHORT_NAME)) {
warn( warn(
`VITE_GLOB_APP_SHORT_NAME Variables can only be characters/underscores, please modify in the environment variables and re-running.`, `VITE_GLOB_APP_SHORT_NAME Variables can only be characters/underscores, please modify in the environment variables and re-running.`
) );
} }
return { return {
@ -48,18 +48,18 @@ export function getAppEnvConfig() {
VITE_GLOB_UPLOAD_URL, VITE_GLOB_UPLOAD_URL,
VITE_GLOB_PROD_MOCK, VITE_GLOB_PROD_MOCK,
VITE_GLOB_IMG_URL, VITE_GLOB_IMG_URL,
} };
} }
/** /**
* @description: Development model * @description: Development model
*/ */
export const devMode = 'development' export const devMode = 'development';
/** /**
* @description: Production mode * @description: Production mode
*/ */
export const prodMode = 'production' export const prodMode = 'production';
/** /**
* @description: Get environment variables * @description: Get environment variables
@ -67,7 +67,7 @@ export const prodMode = 'production'
* @example: * @example:
*/ */
export function getEnv(): string { export function getEnv(): string {
return import.meta.env.MODE return import.meta.env.MODE;
} }
/** /**
@ -76,7 +76,7 @@ export function getEnv(): string {
* @example: * @example:
*/ */
export function isDevMode(): boolean { export function isDevMode(): boolean {
return import.meta.env.DEV return import.meta.env.DEV;
} }
/** /**
@ -85,5 +85,5 @@ export function isDevMode(): boolean {
* @example: * @example:
*/ */
export function isProdMode(): boolean { export function isProdMode(): boolean {
return import.meta.env.PROD return import.meta.env.PROD;
} }

View File

@ -1,32 +1,31 @@
/* eslint-disable ts/ban-ts-comment */ import type { AxiosRequestConfig, AxiosInstance, AxiosResponse } from 'axios';
import type { AxiosInstance, AxiosRequestConfig, AxiosResponse } from 'axios'
import axios from 'axios' import axios from 'axios';
import qs from 'qs' import qs from 'qs';
import { cloneDeep } from 'lodash-es' import { AxiosCanceler } from './axiosCancel';
import { AxiosCanceler } from './axiosCancel' import { isFunction } from '@/utils/is';
import type { CreateAxiosOptions, RequestOptions, Result, UploadFileParams } from './types' import { cloneDeep } from 'lodash-es';
import { isFunction } from '@/utils/is'
import { ContentTypeEnum, RequestEnum } from '@/enums/httpEnum' import type { RequestOptions, CreateAxiosOptions, Result, UploadFileParams } from './types';
import { ContentTypeEnum, RequestEnum } from '@/enums/httpEnum';
export * from './axiosTransform' export * from './axiosTransform';
/** /**
* @description: axios模块 * @description: axios模块
*/ */
export class VAxios { export class VAxios {
private axiosInstance: AxiosInstance private axiosInstance: AxiosInstance;
private options: CreateAxiosOptions private options: CreateAxiosOptions;
constructor(options: CreateAxiosOptions) { constructor(options: CreateAxiosOptions) {
this.options = options this.options = options;
this.axiosInstance = axios.create(options) this.axiosInstance = axios.create(options);
this.setupInterceptors() this.setupInterceptors();
} }
getAxios(): AxiosInstance { getAxios(): AxiosInstance {
return this.axiosInstance return this.axiosInstance;
} }
/** /**
@ -34,9 +33,9 @@ export class VAxios {
*/ */
configAxios(config: CreateAxiosOptions) { configAxios(config: CreateAxiosOptions) {
if (!this.axiosInstance) { if (!this.axiosInstance) {
return return;
} }
this.createAxios(config) this.createAxios(config);
} }
/** /**
@ -44,99 +43,97 @@ export class VAxios {
*/ */
setHeader(headers: any): void { setHeader(headers: any): void {
if (!this.axiosInstance) { if (!this.axiosInstance) {
return return;
} }
Object.assign(this.axiosInstance.defaults.headers, headers) Object.assign(this.axiosInstance.defaults.headers, headers);
} }
/** /**
* @description: * @description:
*/ */
request<T = any>(config: AxiosRequestConfig, options?: RequestOptions): Promise<T> { request<T = any>(config: AxiosRequestConfig, options?: RequestOptions): Promise<T> {
let conf: AxiosRequestConfig = cloneDeep(config) let conf: AxiosRequestConfig = cloneDeep(config);
const transform = this.getTransform() const transform = this.getTransform();
const { requestOptions } = this.options const { requestOptions } = this.options;
const opt: RequestOptions = { ...requestOptions, ...options } const opt: RequestOptions = Object.assign({}, requestOptions, options);
const { beforeRequestHook, requestCatch, transformRequestData } = transform || {} const { beforeRequestHook, requestCatch, transformRequestData } = transform || {};
if (beforeRequestHook && isFunction(beforeRequestHook)) { if (beforeRequestHook && isFunction(beforeRequestHook)) {
conf = beforeRequestHook(conf, opt) conf = beforeRequestHook(conf, opt);
} }
// 这里重新 赋值成最新的配置 //这里重新 赋值成最新的配置
// @ts-expect-error // @ts-ignore
conf.requestOptions = opt conf.requestOptions = opt;
// 支持 FormData // 支持 FormData
conf = this.supportFormData(conf) conf = this.supportFormData(conf);
return new Promise((resolve, reject) => { return new Promise((resolve, reject) => {
this.axiosInstance this.axiosInstance
.request<any, AxiosResponse<Result>>(conf) .request<any, AxiosResponse<Result>>(conf)
.then((res: AxiosResponse<Result>) => { .then((res: AxiosResponse<Result>) => {
// 请求是否被取消 // 请求是否被取消
const isCancel = axios.isCancel(res) const isCancel = axios.isCancel(res);
if (transformRequestData && isFunction(transformRequestData) && !isCancel) { if (transformRequestData && isFunction(transformRequestData) && !isCancel) {
try { try {
const ret = transformRequestData(res, opt) const ret = transformRequestData(res, opt);
resolve(ret) resolve(ret);
} catch (err) {
reject(err || new Error('request error!'));
} }
catch (err) { return;
reject(err || new Error('request error!'))
}
return
} }
resolve(res as unknown as Promise<T>) resolve(res as unknown as Promise<T>);
}) })
.catch((e: Error) => { .catch((e: Error) => {
if (requestCatch && isFunction(requestCatch)) { if (requestCatch && isFunction(requestCatch)) {
reject(requestCatch(e)) reject(requestCatch(e));
return return;
} }
reject(e) reject(e);
}) });
}) });
} }
/** /**
* @description: axios实例 * @description: axios实例
*/ */
private createAxios(config: CreateAxiosOptions): void { private createAxios(config: CreateAxiosOptions): void {
this.axiosInstance = axios.create(config) this.axiosInstance = axios.create(config);
} }
private getTransform() { private getTransform() {
const { transform } = this.options const { transform } = this.options;
return transform return transform;
} }
/** /**
* @description: * @description:
*/ */
uploadFile<T = any>(config: AxiosRequestConfig, params: UploadFileParams) { uploadFile<T = any>(config: AxiosRequestConfig, params: UploadFileParams) {
const formData = new window.FormData() const formData = new window.FormData();
const customFilename = params.name || 'file' const customFilename = params.name || 'file';
if (params.filename) { if (params.filename) {
formData.append(customFilename, params.file, params.filename) formData.append(customFilename, params.file, params.filename);
} } else {
else { formData.append(customFilename, params.file);
formData.append(customFilename, params.file)
} }
if (params.data) { if (params.data) {
Object.keys(params.data).forEach((key) => { Object.keys(params.data).forEach((key) => {
const value = params.data![key] const value = params.data![key];
if (Array.isArray(value)) { if (Array.isArray(value)) {
value.forEach((item) => { value.forEach((item) => {
formData.append(`${key}[]`, item) formData.append(`${key}[]`, item);
}) });
return return;
} }
formData.append(key, params.data![key]) formData.append(key, params.data![key]);
}) });
} }
return this.axiosInstance.request<T>({ return this.axiosInstance.request<T>({
@ -144,81 +141,80 @@ export class VAxios {
data: formData, data: formData,
headers: { headers: {
'Content-type': ContentTypeEnum.FORM_DATA, 'Content-type': ContentTypeEnum.FORM_DATA,
'ignoreCancelToken': true, ignoreCancelToken: true,
}, },
...config, ...config,
}) });
} }
// support form-data // support form-data
supportFormData(config: AxiosRequestConfig) { supportFormData(config: AxiosRequestConfig) {
const headers = config.headers || this.options.headers const headers = config.headers || this.options.headers;
const contentType = headers?.['Content-Type'] || headers?.['content-type'] const contentType = headers?.['Content-Type'] || headers?.['content-type'];
if ( if (
contentType !== ContentTypeEnum.FORM_URLENCODED contentType !== ContentTypeEnum.FORM_URLENCODED ||
|| !Reflect.has(config, 'data') !Reflect.has(config, 'data') ||
|| config.method?.toUpperCase() === RequestEnum.GET config.method?.toUpperCase() === RequestEnum.GET
) { ) {
return config return config;
} }
return { return {
...config, ...config,
data: qs.stringify(config.data, { arrayFormat: 'brackets' }), data: qs.stringify(config.data, { arrayFormat: 'brackets' }),
} };
} }
/** /**
* @description: * @description:
*/ */
private setupInterceptors() { private setupInterceptors() {
const transform = this.getTransform() const transform = this.getTransform();
if (!transform) { if (!transform) {
return return;
} }
const { const {
requestInterceptors, requestInterceptors,
requestInterceptorsCatch, requestInterceptorsCatch,
responseInterceptors, responseInterceptors,
responseInterceptorsCatch, responseInterceptorsCatch,
} = transform } = transform;
const axiosCanceler = new AxiosCanceler() const axiosCanceler = new AxiosCanceler();
// 请求拦截器配置处理 // 请求拦截器配置处理
// @ts-expect-error
this.axiosInstance.interceptors.request.use((config: AxiosRequestConfig) => { this.axiosInstance.interceptors.request.use((config: AxiosRequestConfig) => {
const { headers: { ignoreCancelToken } = { ignoreCancelToken: false } } = config const { headers: { ignoreCancelToken } = { ignoreCancelToken: false } } = config;
const ignoreCancel const ignoreCancel =
= ignoreCancelToken !== undefined ignoreCancelToken !== undefined
? ignoreCancelToken ? ignoreCancelToken
: this.options.requestOptions?.ignoreCancelToken : this.options.requestOptions?.ignoreCancelToken;
!ignoreCancel && axiosCanceler.addPending(config) !ignoreCancel && axiosCanceler.addPending(config);
if (requestInterceptors && isFunction(requestInterceptors)) { if (requestInterceptors && isFunction(requestInterceptors)) {
config = requestInterceptors(config, this.options) config = requestInterceptors(config, this.options);
} }
return config return config;
}, undefined) }, undefined);
// 请求拦截器错误捕获 // 请求拦截器错误捕获
requestInterceptorsCatch requestInterceptorsCatch &&
&& isFunction(requestInterceptorsCatch) isFunction(requestInterceptorsCatch) &&
&& this.axiosInstance.interceptors.request.use(undefined, requestInterceptorsCatch) this.axiosInstance.interceptors.request.use(undefined, requestInterceptorsCatch);
// 响应结果拦截器处理 // 响应结果拦截器处理
this.axiosInstance.interceptors.response.use((res: AxiosResponse<any>) => { this.axiosInstance.interceptors.response.use((res: AxiosResponse<any>) => {
res && axiosCanceler.removePending(res.config) res && axiosCanceler.removePending(res.config);
if (responseInterceptors && isFunction(responseInterceptors)) { if (responseInterceptors && isFunction(responseInterceptors)) {
res = responseInterceptors(res) res = responseInterceptors(res);
} }
return res return res;
}, undefined) }, undefined);
// 响应结果拦截器错误捕获 // 响应结果拦截器错误捕获
responseInterceptorsCatch responseInterceptorsCatch &&
&& isFunction(responseInterceptorsCatch) isFunction(responseInterceptorsCatch) &&
&& this.axiosInstance.interceptors.response.use(undefined, responseInterceptorsCatch) this.axiosInstance.interceptors.response.use(undefined, responseInterceptorsCatch);
} }
} }

View File

@ -1,33 +1,31 @@
import type { AxiosRequestConfig, Canceler } from 'axios' import axios, { AxiosRequestConfig, Canceler } from 'axios';
import axios from 'axios'
import qs from 'qs' import qs from 'qs';
import { isFunction } from '@/utils/is/index' import { isFunction } from '@/utils/is/index';
// 声明一个 Map 用于存储每个请求的标识 和 取消函数 // 声明一个 Map 用于存储每个请求的标识 和 取消函数
let pendingMap = new Map<string, Canceler>() let pendingMap = new Map<string, Canceler>();
export function getPendingUrl(config: AxiosRequestConfig) { export const getPendingUrl = (config: AxiosRequestConfig) =>
return [config.method, config.url, qs.stringify(config.data), qs.stringify(config.params)].join('&') [config.method, config.url, qs.stringify(config.data), qs.stringify(config.params)].join('&');
}
export class AxiosCanceler { export class AxiosCanceler {
/** /**
* *
* @param {object} config * @param {Object} config
*/ */
addPending(config: AxiosRequestConfig) { addPending(config: AxiosRequestConfig) {
this.removePending(config) this.removePending(config);
const url = getPendingUrl(config) const url = getPendingUrl(config);
config.cancelToken config.cancelToken =
= config.cancelToken config.cancelToken ||
|| new axios.CancelToken((cancel) => { new axios.CancelToken((cancel) => {
if (!pendingMap.has(url)) { if (!pendingMap.has(url)) {
// 如果 pending 中不存在当前请求,则添加进去 // 如果 pending 中不存在当前请求,则添加进去
pendingMap.set(url, cancel) pendingMap.set(url, cancel);
} }
}) });
} }
/** /**
@ -35,23 +33,23 @@ export class AxiosCanceler {
*/ */
removeAllPending() { removeAllPending() {
pendingMap.forEach((cancel) => { pendingMap.forEach((cancel) => {
cancel && isFunction(cancel) && cancel() cancel && isFunction(cancel) && cancel();
}) });
pendingMap.clear() pendingMap.clear();
} }
/** /**
* *
* @param {object} config * @param {Object} config
*/ */
removePending(config: AxiosRequestConfig) { removePending(config: AxiosRequestConfig) {
const url = getPendingUrl(config) const url = getPendingUrl(config);
if (pendingMap.has(url)) { if (pendingMap.has(url)) {
// 如果在 pending 中存在当前请求标识,需要取消当前请求,并且移除 // 如果在 pending 中存在当前请求标识,需要取消当前请求,并且移除
const cancel = pendingMap.get(url) const cancel = pendingMap.get(url);
cancel && cancel(url) cancel && cancel(url);
pendingMap.delete(url) pendingMap.delete(url);
} }
} }
@ -59,6 +57,6 @@ export class AxiosCanceler {
* @description: * @description:
*/ */
reset(): void { reset(): void {
pendingMap = new Map<string, Canceler>() pendingMap = new Map<string, Canceler>();
} }
} }

View File

@ -1,13 +1,13 @@
/** /**
* *
*/ */
import type { AxiosRequestConfig, AxiosResponse } from 'axios' import type { AxiosRequestConfig, AxiosResponse } from 'axios';
import type { RequestOptions, Result } from './types' import type { RequestOptions, Result } from './types';
export interface CreateAxiosOptions extends AxiosRequestConfig { export interface CreateAxiosOptions extends AxiosRequestConfig {
authenticationScheme?: string authenticationScheme?: string;
transform?: AxiosTransform transform?: AxiosTransform;
requestOptions?: RequestOptions requestOptions?: RequestOptions;
} }
export abstract class AxiosTransform { export abstract class AxiosTransform {
@ -15,17 +15,17 @@ export abstract class AxiosTransform {
* @description: * @description:
* @description: Process configuration before request * @description: Process configuration before request
*/ */
beforeRequestHook?: (config: AxiosRequestConfig, options: RequestOptions) => AxiosRequestConfig beforeRequestHook?: (config: AxiosRequestConfig, options: RequestOptions) => AxiosRequestConfig;
/** /**
* @description: * @description:
*/ */
transformRequestData?: (res: AxiosResponse<Result>, options: RequestOptions) => any transformRequestData?: (res: AxiosResponse<Result>, options: RequestOptions) => any;
/** /**
* @description: * @description:
*/ */
requestCatch?: (e: Error) => Promise<any> requestCatch?: (e: Error) => Promise<any>;
/** /**
* @description: * @description:
@ -33,20 +33,20 @@ export abstract class AxiosTransform {
requestInterceptors?: ( requestInterceptors?: (
config: AxiosRequestConfig, config: AxiosRequestConfig,
options: CreateAxiosOptions options: CreateAxiosOptions
) => AxiosRequestConfig ) => AxiosRequestConfig;
/** /**
* @description: * @description:
*/ */
responseInterceptors?: (res: AxiosResponse<any>) => AxiosResponse<any> responseInterceptors?: (res: AxiosResponse<any>) => AxiosResponse<any>;
/** /**
* @description: * @description:
*/ */
requestInterceptorsCatch?: (error: Error) => void requestInterceptorsCatch?: (error: Error) => void;
/** /**
* @description: * @description:
*/ */
responseInterceptorsCatch?: (error: Error) => void responseInterceptorsCatch?: (error: Error) => void;
} }

View File

@ -1,48 +1,48 @@
import { showFailToast } from 'vant' import { showFailToast } from 'vant';
export function checkStatus(status: number, msg: string): void { export function checkStatus(status: number, msg: string): void {
switch (status) { switch (status) {
case 400: case 400:
showFailToast(msg) showFailToast(msg);
break break;
// 401: 未登录 // 401: 未登录
// 未登录则跳转登录页面,并携带当前页面的路径 // 未登录则跳转登录页面,并携带当前页面的路径
// 在登录成功后返回当前页面,这一步需要在登录页操作。 // 在登录成功后返回当前页面,这一步需要在登录页操作。
case 401: case 401:
showFailToast('用户没有权限(令牌、用户名、密码错误)!') showFailToast('用户没有权限(令牌、用户名、密码错误)!');
break break;
case 403: case 403:
showFailToast('用户得到授权,但是访问是被禁止的。!') showFailToast('用户得到授权,但是访问是被禁止的。!');
break break;
// 404请求不存在 // 404请求不存在
case 404: case 404:
showFailToast('网络请求错误,未找到该资源!') showFailToast('网络请求错误,未找到该资源!');
break break;
case 405: case 405:
showFailToast('网络请求错误,请求方法未允许!') showFailToast('网络请求错误,请求方法未允许!');
break break;
case 408: case 408:
showFailToast('网络请求超时') showFailToast('网络请求超时');
break break;
case 500: case 500:
showFailToast('服务器错误,请联系管理员!') showFailToast('服务器错误,请联系管理员!');
break break;
case 501: case 501:
showFailToast('网络未实现') showFailToast('网络未实现');
break break;
case 502: case 502:
showFailToast('网络错误') showFailToast('网络错误');
break break;
case 503: case 503:
showFailToast('服务不可用,服务器暂时过载或维护!') showFailToast('服务不可用,服务器暂时过载或维护!');
break break;
case 504: case 504:
showFailToast('网络超时') showFailToast('网络超时');
break break;
case 505: case 505:
showFailToast('http版本不支持该请求!') showFailToast('http版本不支持该请求!');
break break;
default: default:
showFailToast(msg) showFailToast(msg);
} }
} }

View File

@ -1,21 +1,21 @@
import { isObject, isString } from '@/utils/is' import { isObject, isString } from '@/utils/is';
const DATE_TIME_FORMAT = 'YYYY-MM-DD HH:mm' const DATE_TIME_FORMAT = 'YYYY-MM-DD HH:mm';
export function joinTimestamp<T extends boolean>( export function joinTimestamp<T extends boolean>(
join: boolean, join: boolean,
restful: T restful: T
): T extends true ? string : object ): T extends true ? string : object;
export function joinTimestamp(join: boolean, restful = false): string | object { export function joinTimestamp(join: boolean, restful = false): string | object {
if (!join) { if (!join) {
return restful ? '' : {} return restful ? '' : {};
} }
const now = new Date().getTime() const now = new Date().getTime();
if (restful) { if (restful) {
return `?_t=${now}` return `?_t=${now}`;
} }
return { _t: now } return { _t: now };
} }
/** /**
@ -23,26 +23,25 @@ export function joinTimestamp(join: boolean, restful = false): string | object {
*/ */
export function formatRequestDate(params: Recordable) { export function formatRequestDate(params: Recordable) {
if (Object.prototype.toString.call(params) !== '[object Object]') { if (Object.prototype.toString.call(params) !== '[object Object]') {
return return;
} }
for (const key in params) { for (const key in params) {
if (params[key] && params[key]._isAMomentObject) { if (params[key] && params[key]._isAMomentObject) {
params[key] = params[key].format(DATE_TIME_FORMAT) params[key] = params[key].format(DATE_TIME_FORMAT);
} }
if (isString(key)) { if (isString(key)) {
const value = params[key] const value = params[key];
if (value) { if (value) {
try { try {
params[key] = isString(value) ? value.trim() : value params[key] = isString(value) ? value.trim() : value;
} } catch (error) {
catch (error) { throw new Error(error as any);
throw new Error(error as any)
} }
} }
} }
if (isObject(params[key])) { if (isObject(params[key])) {
formatRequestDate(params[key]) formatRequestDate(params[key]);
} }
} }
} }

View File

@ -1,27 +1,27 @@
// axios配置 可自行根据项目进行更改,只需更改该文件即可,其他文件可以不动 // axios配置 可自行根据项目进行更改,只需更改该文件即可,其他文件可以不动
import type { AxiosResponse } from 'axios' import { VAxios } from './Axios';
import axios from 'axios' import { AxiosTransform } from './axiosTransform';
import { showDialog, showFailToast } from 'vant' import axios, { AxiosResponse } from 'axios';
import { VAxios } from './Axios' import { checkStatus } from './checkStatus';
import type { AxiosTransform } from './axiosTransform' import { joinTimestamp, formatRequestDate } from './helper';
import { checkStatus } from './checkStatus' import { RequestEnum, ResultEnum, ContentTypeEnum } from '@/enums/httpEnum';
import { formatRequestDate, joinTimestamp } from './helper' import { PageEnum } from '@/enums/pageEnum';
import type { CreateAxiosOptions, RequestOptions, Result } from './types' import { useGlobSetting } from '@/hooks/setting';
import { ContentTypeEnum, RequestEnum, ResultEnum } from '@/enums/httpEnum'
import { PageEnum } from '@/enums/pageEnum'
import { useGlobSetting } from '@/hooks/setting'
import { isString } from '@/utils/is/' import { isString } from '@/utils/is/';
import { deepMerge, isUrl } from '@/utils' import { deepMerge, isUrl } from '@/utils';
import { setObjToUrlParams } from '@/utils/urlUtils' import { setObjToUrlParams } from '@/utils/urlUtils';
import { useUserStoreWidthOut } from '@/store/modules/user' import { RequestOptions, Result, CreateAxiosOptions } from './types';
import router from '@/router' import { useUserStoreWidthOut } from '@/store/modules/user';
import { storage } from '@/utils/Storage'
const globSetting = useGlobSetting() const globSetting = useGlobSetting();
const urlPrefix = globSetting.urlPrefix || '' const urlPrefix = globSetting.urlPrefix || '';
import router from '@/router';
import { storage } from '@/utils/Storage';
import { showFailToast, showDialog } from 'vant';
/** /**
* @description: 便 * @description: 便
@ -39,29 +39,29 @@ const transform: AxiosTransform = {
errorMessageText, errorMessageText,
isTransformResponse, isTransformResponse,
isReturnNativeResponse, isReturnNativeResponse,
} = options } = options;
// 是否返回原生响应头 比如:需要获取响应头时使用该属性 // 是否返回原生响应头 比如:需要获取响应头时使用该属性
if (isReturnNativeResponse) { if (isReturnNativeResponse) {
return res return res;
} }
// 不进行任何处理,直接返回 // 不进行任何处理,直接返回
// 用于页面代码可能需要直接获取codedatamessage这些信息时开启 // 用于页面代码可能需要直接获取codedatamessage这些信息时开启
if (!isTransformResponse) { if (!isTransformResponse) {
return res.data return res.data;
} }
const { data } = res const { data } = res;
if (!data) { if (!data) {
// return '[HTTP] Request has no return value'; // return '[HTTP] Request has no return value';
throw new Error('请求出错,请稍候重试') throw new Error('请求出错,请稍候重试');
} }
// 这里 coderesultmessage为 后台统一的字段,需要修改为项目自己的接口返回格式 // 这里 coderesultmessage为 后台统一的字段,需要修改为项目自己的接口返回格式
const { code, result, message } = data const { code, result, message } = data;
// 请求成功 // 请求成功
const hasSuccess = data && Reflect.has(data, 'code') && code === ResultEnum.SUCCESS const hasSuccess = data && Reflect.has(data, 'code') && code === ResultEnum.SUCCESS;
// 是否显示提示信息 // 是否显示提示信息
if (isShowMessage) { if (isShowMessage) {
if (hasSuccess && (successMessageText || isShowSuccessMessage)) { if (hasSuccess && (successMessageText || isShowSuccessMessage)) {
@ -69,116 +69,109 @@ const transform: AxiosTransform = {
message: successMessageText || message || '操作成功!', message: successMessageText || message || '操作成功!',
}).then(() => { }).then(() => {
// on close // on close
}) });
} } else if (!hasSuccess && (errorMessageText || isShowErrorMessage)) {
else if (!hasSuccess && (errorMessageText || isShowErrorMessage)) {
// 是否显示自定义信息提示 // 是否显示自定义信息提示
showFailToast(message || errorMessageText || '操作失败!') showFailToast(message || errorMessageText || '操作失败!');
} } else if (!hasSuccess && options.errorMessageMode === 'modal') {
else if (!hasSuccess && options.errorMessageMode === 'modal') {
// errorMessageMode=custom-modal的时候会显示modal错误弹窗而不是消息提示用于一些比较重要的错误 // errorMessageMode=custom-modal的时候会显示modal错误弹窗而不是消息提示用于一些比较重要的错误
showDialog({ showDialog({
title: '提示', title: '提示',
message, message: message,
}).then(() => { }).then(() => {
// on close // on close
}) });
} }
} }
// 接口请求成功,直接返回结果 // 接口请求成功,直接返回结果
if (code === ResultEnum.SUCCESS) { if (code === ResultEnum.SUCCESS) {
return result return result;
} }
// 接口请求错误,统一提示错误信息 这里逻辑可以根据项目进行修改 // 接口请求错误,统一提示错误信息 这里逻辑可以根据项目进行修改
let errorMsg = message let errorMsg = message;
const LoginName = PageEnum.BASE_LOGIN_NAME
const LoginPath = PageEnum.BASE_LOGIN
switch (code) { switch (code) {
// 请求失败 // 请求失败
case ResultEnum.ERROR: case ResultEnum.ERROR:
showFailToast(errorMsg) showFailToast(errorMsg);
break break;
// token 过期 // token 过期
case ResultEnum.TOKEN_EXPIRED: case ResultEnum.TOKEN_EXPIRED:
if (router.currentRoute.value?.name === LoginName) { const LoginName = PageEnum.BASE_LOGIN_NAME;
return const LoginPath = PageEnum.BASE_LOGIN;
} if (router.currentRoute.value?.name === LoginName) return;
// 到登录页 // 到登录页
errorMsg = '登录超时,请重新登录!' errorMsg = '登录超时,请重新登录!';
showDialog({ showDialog({
title: '提示', title: '提示',
message: '登录身份已失效,请重新登录!', message: '登录身份已失效,请重新登录!',
}) })
.then(() => { .then(() => {
storage.clear() storage.clear();
window.location.href = LoginPath window.location.href = LoginPath;
}) })
.catch(() => { .catch(() => {
// on cancel // on cancel
}) });
break break;
} }
throw new Error(errorMsg) throw new Error(errorMsg);
}, },
// 请求之前处理config // 请求之前处理config
beforeRequestHook: (config, options) => { beforeRequestHook: (config, options) => {
const { apiUrl, joinPrefix, joinParamsToUrl, formatDate, joinTime = true, urlPrefix } = options const { apiUrl, joinPrefix, joinParamsToUrl, formatDate, joinTime = true, urlPrefix } = options;
const isUrlStr = isUrl(config.url as string) const isUrlStr = isUrl(config.url as string);
if (!isUrlStr && joinPrefix) { if (!isUrlStr && joinPrefix) {
config.url = `${urlPrefix}${config.url}` config.url = `${urlPrefix}${config.url}`;
} }
if (!isUrlStr && apiUrl && isString(apiUrl)) { if (!isUrlStr && apiUrl && isString(apiUrl)) {
config.url = `${apiUrl}${config.url}` config.url = `${apiUrl}${config.url}`;
} }
const params = config.params || {} const params = config.params || {};
const data = config.data || false const data = config.data || false;
if (config.method?.toUpperCase() === RequestEnum.GET) { if (config.method?.toUpperCase() === RequestEnum.GET) {
if (!isString(params)) { if (!isString(params)) {
// 给 get 请求加上时间戳参数,避免从缓存中拿数据。 // 给 get 请求加上时间戳参数,避免从缓存中拿数据。
config.params = Object.assign(params || {}, joinTimestamp(joinTime, false)) config.params = Object.assign(params || {}, joinTimestamp(joinTime, false));
} } else {
else {
// 兼容restful风格 // 兼容restful风格
config.url = `${config.url + params}${joinTimestamp(joinTime, true)}` config.url = config.url + params + `${joinTimestamp(joinTime, true)}`;
config.params = undefined config.params = undefined;
} }
} } else {
else {
if (!isString(params)) { if (!isString(params)) {
formatDate && formatRequestDate(params) formatDate && formatRequestDate(params);
if ( if (
Reflect.has(config, 'data') Reflect.has(config, 'data') &&
&& config.data config.data &&
&& (Object.keys(config.data).length > 0 || config.data instanceof FormData) (Object.keys(config.data).length > 0 || config.data instanceof FormData)
) { ) {
config.data = data config.data = data;
config.params = params config.params = params;
} } else {
else {
// params 是添加到 url 的请求字符串中的,用于 get 请求 // params 是添加到 url 的请求字符串中的,用于 get 请求
// 非GET请求如果没有提供 data则将 params 视为 data // 非GET请求如果没有提供 data则将 params 视为 data
config.data = params config.data = params;
config.params = undefined config.params = undefined;
} }
if (joinParamsToUrl) { if (joinParamsToUrl) {
config.url = setObjToUrlParams( config.url = setObjToUrlParams(
config.url as string, config.url as string,
{ ...config.params, ...config.data }, Object.assign({}, config.params, config.data)
) );
} }
} } else {
else {
// 兼容restful风格 // 兼容restful风格
config.url = config.url + params config.url = config.url + params;
config.params = undefined config.params = undefined;
} }
} }
return config return config;
}, },
/** /**
@ -186,30 +179,30 @@ const transform: AxiosTransform = {
*/ */
requestInterceptors: (config, options) => { requestInterceptors: (config, options) => {
// 请求之前处理config // 请求之前处理config
const userStore = useUserStoreWidthOut() const userStore = useUserStoreWidthOut();
const token = userStore.getToken const token = userStore.getToken;
if (token && (config as Recordable)?.requestOptions?.withToken !== false) { if (token && (config as Recordable)?.requestOptions?.withToken !== false) {
// jwt token // jwt token
(config as Recordable).headers.Authorization = options.authenticationScheme (config as Recordable).headers.Authorization = options.authenticationScheme
? `${options.authenticationScheme} ${token}` ? `${options.authenticationScheme} ${token}`
: token : token;
} }
return config return config;
}, },
/** /**
* @description: * @description:
*/ */
responseInterceptorsCatch: (error: any) => { responseInterceptorsCatch: (error: any) => {
const { response, code, message } = error || {} const { response, code, message } = error || {};
// TODO 此处要根据后端接口返回格式修改 // TODO 此处要根据后端接口返回格式修改
const msg: string const msg: string =
= response && response.data && response.data.message ? response.data.message : '' response && response.data && response.data.message ? response.data.message : '';
const err: string = error.toString() const err: string = error.toString();
try { try {
if (code === 'ECONNABORTED' && message.includes('timeout')) { if (code === 'ECONNABORTED' && message.indexOf('timeout') !== -1) {
showFailToast('接口请求超时,请刷新页面重试!') showFailToast('接口请求超时,请刷新页面重试!');
return return;
} }
if (err && err.includes('Network Error')) { if (err && err.includes('Network Error')) {
showDialog({ showDialog({
@ -217,25 +210,23 @@ const transform: AxiosTransform = {
message: '请检查您的网络连接是否正常', message: '请检查您的网络连接是否正常',
}) })
.then(() => {}) .then(() => {})
.catch(() => {}) .catch(() => {});
return Promise.reject(error) return Promise.reject(error);
} }
} } catch (error) {
catch (error) { throw new Error(error as any);
throw new Error(error as any)
} }
// 请求是否被取消 // 请求是否被取消
const isCancel = axios.isCancel(error) const isCancel = axios.isCancel(error);
if (!isCancel) { if (!isCancel) {
checkStatus(error.response && error.response.status, msg) checkStatus(error.response && error.response.status, msg);
} else {
console.warn(error, '请求被取消!');
} }
else { //return Promise.reject(error);
console.warn(error, '请求被取消!') return Promise.reject(response?.data);
}
// return Promise.reject(error);
return Promise.reject(response?.data)
}, },
} };
function createAxios(opt?: Partial<CreateAxiosOptions>) { function createAxios(opt?: Partial<CreateAxiosOptions>) {
return new VAxios( return new VAxios(
@ -269,7 +260,7 @@ function createAxios(opt?: Partial<CreateAxiosOptions>) {
// 接口地址 // 接口地址
apiUrl: globSetting.apiUrl, apiUrl: globSetting.apiUrl,
// 接口拼接地址 // 接口拼接地址
urlPrefix, urlPrefix: urlPrefix,
// 是否加入时间戳 // 是否加入时间戳
joinTime: true, joinTime: true,
// 忽略重复请求 // 忽略重复请求
@ -279,12 +270,12 @@ function createAxios(opt?: Partial<CreateAxiosOptions>) {
}, },
withCredentials: false, withCredentials: false,
}, },
opt || {}, opt || {}
), )
) );
} }
export const http = createAxios() export const http = createAxios();
// 项目,多个不同 api 地址,直接在这里导出多个 // 项目,多个不同 api 地址,直接在这里导出多个
// src/api ts 里面接口,就可以单独使用这个请求, // src/api ts 里面接口,就可以单独使用这个请求,

View File

@ -1,65 +1,65 @@
import type { AxiosRequestConfig } from 'axios' import { AxiosRequestConfig } from 'axios';
import type { AxiosTransform } from './axiosTransform' import { AxiosTransform } from './axiosTransform';
export interface CreateAxiosOptions extends AxiosRequestConfig { export interface CreateAxiosOptions extends AxiosRequestConfig {
transform?: AxiosTransform transform?: AxiosTransform;
requestOptions?: RequestOptions requestOptions?: RequestOptions;
authenticationScheme?: string authenticationScheme?: string;
} }
// 上传文件 // 上传文件
export interface UploadFileParams { export interface UploadFileParams {
// 其他参数 // 其他参数
data?: Recordable data?: Recordable;
// 文件参数接口字段名 // 文件参数接口字段名
name?: string name?: string;
// 文件 // 文件
file: File | Blob file: File | Blob;
// 文件名称 // 文件名称
filename?: string filename?: string;
[key: string]: any [key: string]: any;
} }
export interface RequestOptions { export interface RequestOptions {
// 请求参数拼接到url // 请求参数拼接到url
joinParamsToUrl?: boolean joinParamsToUrl?: boolean;
// 格式化请求参数时间 // 格式化请求参数时间
formatDate?: boolean formatDate?: boolean;
// 是否显示提示信息 // 是否显示提示信息
isShowMessage?: boolean isShowMessage?: boolean;
// 是否解析成JSON // 是否解析成JSON
isParseToJson?: boolean isParseToJson?: boolean;
// 成功的文本信息 // 成功的文本信息
successMessageText?: string successMessageText?: string;
// 是否显示成功信息 // 是否显示成功信息
isShowSuccessMessage?: boolean isShowSuccessMessage?: boolean;
// 是否显示失败信息 // 是否显示失败信息
isShowErrorMessage?: boolean isShowErrorMessage?: boolean;
// 错误的文本信息 // 错误的文本信息
errorMessageText?: string errorMessageText?: string;
// 是否加入url // 是否加入url
joinPrefix?: boolean joinPrefix?: boolean;
// 接口地址, 不填则使用默认apiUrl // 接口地址, 不填则使用默认apiUrl
apiUrl?: string apiUrl?: string;
// 请求拼接路径 // 请求拼接路径
urlPrefix?: string urlPrefix?: string;
// 错误消息提示类型 // 错误消息提示类型
errorMessageMode?: 'none' | 'modal' errorMessageMode?: 'none' | 'modal';
// 是否添加时间戳 // 是否添加时间戳
joinTime?: boolean joinTime?: boolean;
// 不进行任何处理,直接返回 // 不进行任何处理,直接返回
isTransformResponse?: boolean isTransformResponse?: boolean;
// 是否返回原生响应头 // 是否返回原生响应头
isReturnNativeResponse?: boolean isReturnNativeResponse?: boolean;
// 忽略重复请求 //忽略重复请求
ignoreCancelToken?: boolean ignoreCancelToken?: boolean;
// 是否携带token // 是否携带token
withToken?: boolean withToken?: boolean;
} }
export interface Result<T = any> { export interface Result<T = any> {
code: number code: number;
type?: 'success' | 'error' | 'warning' type?: 'success' | 'error' | 'warning';
message: string message: string;
result?: T result?: T;
} }

View File

@ -1,11 +1,11 @@
import { isObject } from './is/index' import { isObject } from './is/index';
export function deepMerge<T = any>(src: any = {}, target: any = {}): T { export function deepMerge<T = any>(src: any = {}, target: any = {}): T {
let key: string let key: string;
for (key in target) { for (key in target) {
src[key] = isObject(src[key]) ? deepMerge(src[key], target[key]) : (src[key] = target[key]) src[key] = isObject(src[key]) ? deepMerge(src[key], target[key]) : (src[key] = target[key]);
} }
return src return src;
} }
/** /**
@ -15,9 +15,9 @@ export function deepMerge<T = any>(src: any = {}, target: any = {}): T {
* @returns {string} The processed part of the color * @returns {string} The processed part of the color
*/ */
function addLight(color: string, amount: number) { function addLight(color: string, amount: number) {
const cc = Number.parseInt(color, 16) + amount const cc = parseInt(color, 16) + amount;
const c = cc > 255 ? 255 : cc const c = cc > 255 ? 255 : cc;
return c.toString(16).length > 1 ? c.toString(16) : `0${c.toString(16)}` return c.toString(16).length > 1 ? c.toString(16) : `0${c.toString(16)}`;
} }
/** /**
@ -27,12 +27,12 @@ function addLight(color: string, amount: number) {
* @returns {string} The HEX representation of the processed color * @returns {string} The HEX representation of the processed color
*/ */
export function darken(color: string, amount: number) { export function darken(color: string, amount: number) {
color = color.includes('#') ? color.substring(1, color.length) : color color = color.indexOf('#') >= 0 ? color.substring(1, color.length) : color;
amount = Math.trunc((255 * amount) / 100) amount = Math.trunc((255 * amount) / 100);
return `#${subtractLight(color.substring(0, 2), amount)}${subtractLight( return `#${subtractLight(color.substring(0, 2), amount)}${subtractLight(
color.substring(2, 4), color.substring(2, 4),
amount, amount
)}${subtractLight(color.substring(4, 6), amount)}` )}${subtractLight(color.substring(4, 6), amount)}`;
} }
/** /**
@ -42,31 +42,31 @@ export function darken(color: string, amount: number) {
* @returns {string} The processed color represented as HEX * @returns {string} The processed color represented as HEX
*/ */
export function lighten(color: string, amount: number) { export function lighten(color: string, amount: number) {
color = color.includes('#') ? color.substring(1, color.length) : color color = color.indexOf('#') >= 0 ? color.substring(1, color.length) : color;
amount = Math.trunc((255 * amount) / 100) amount = Math.trunc((255 * amount) / 100);
return `#${addLight(color.substring(0, 2), amount)}${addLight( return `#${addLight(color.substring(0, 2), amount)}${addLight(
color.substring(2, 4), color.substring(2, 4),
amount, amount
)}${addLight(color.substring(4, 6), amount)}` )}${addLight(color.substring(4, 6), amount)}`;
} }
/** /**
* url * url
*/ * */
const RegExp = /^http(s)?:\/\//iu const RegExp = /^http(s)?:\/\//iu;
export function isUrl(url: string) { export function isUrl(url: string) {
return RegExp.test(url) return RegExp.test(url);
} }
/** /**
* *
*/ * */
export function arrayTrans(arr: number[]): number[][] { export function arrayTrans(arr: number[]): number[][] {
const newArr: number[][] = [] const newArr: number[][] = [];
while (arr.length > 0) { while (arr.length > 0) {
newArr.push(arr.splice(0, 2)) newArr.push(arr.splice(0, 2));
} }
return newArr return newArr;
} }
/** /**
@ -76,24 +76,24 @@ export function arrayTrans(arr: number[]): number[][] {
* @returns {string} The processed part of the color * @returns {string} The processed part of the color
*/ */
function subtractLight(color: string, amount: number) { function subtractLight(color: string, amount: number) {
const cc = Number.parseInt(color, 16) - amount const cc = parseInt(color, 16) - amount;
const c = cc < 0 ? 0 : cc const c = cc < 0 ? 0 : cc;
return c.toString(16).length > 1 ? c.toString(16) : `0${c.toString(16)}` return c.toString(16).length > 1 ? c.toString(16) : `0${c.toString(16)}`;
} }
export function hexToRgba(hex: string, opacity: number) { export function hexToRgba(hex: string, opacity: number) {
const shorthandRegex = /^#?([a-f\d])([a-f\d])([a-f\d])$/i const shorthandRegex = /^#?([a-f\d])([a-f\d])([a-f\d])$/i;
hex = hex.replace(shorthandRegex, (m, r, g, b) => { hex = hex.replace(shorthandRegex, function (m, r, g, b) {
return r + r + g + g + b + b return r + r + g + g + b + b;
}) });
const result = /^#?([a-f\d]{2})([a-f\d]{2})([a-f\d]{2})$/i.exec(hex) const result = /^#?([a-f\d]{2})([a-f\d]{2})([a-f\d]{2})$/i.exec(hex);
opacity = opacity >= 0 && opacity <= 1 ? Number(opacity) : 1 opacity = opacity >= 0 && opacity <= 1 ? Number(opacity) : 1;
return result return result
? `rgba(${ ? 'rgba(' +
[Number.parseInt(result[1], 16), Number.parseInt(result[2], 16), Number.parseInt(result[3], 16), opacity].join( [parseInt(result[1], 16), parseInt(result[2], 16), parseInt(result[3], 16), opacity].join(
',', ','
) ) +
})` ')'
: hex : hex;
} }

View File

@ -1,125 +1,125 @@
const toString = Object.prototype.toString const toString = Object.prototype.toString;
/** /**
* @description: * @description:
*/ */
export function is(val: unknown, type: string) { export function is(val: unknown, type: string) {
return toString.call(val) === `[object ${type}]` return toString.call(val) === `[object ${type}]`;
} }
/** /**
* @description: * @description:
*/ */
export function isFunction<T = Function>(val: unknown): val is T { export function isFunction<T = Function>(val: unknown): val is T {
return is(val, 'Function') return is(val, 'Function');
} }
/** /**
* @description: * @description:
*/ */
export function isDef<T = unknown>(val?: T): val is T { export const isDef = <T = unknown>(val?: T): val is T => {
return typeof val !== 'undefined' return typeof val !== 'undefined';
} };
export function isUnDef<T = unknown>(val?: T): val is T { export const isUnDef = <T = unknown>(val?: T): val is T => {
return !isDef(val) return !isDef(val);
} };
/** /**
* @description: * @description:
*/ */
export function isObject(val: any): val is Record<any, any> { export const isObject = (val: any): val is Record<any, any> => {
return val !== null && is(val, 'Object') return val !== null && is(val, 'Object');
} };
/** /**
* @description: * @description:
*/ */
export function isDate(val: unknown): val is Date { export function isDate(val: unknown): val is Date {
return is(val, 'Date') return is(val, 'Date');
} }
/** /**
* @description: * @description:
*/ */
export function isNumber(val: unknown): val is number { export function isNumber(val: unknown): val is number {
return is(val, 'Number') return is(val, 'Number');
} }
/** /**
* @description: AsyncFunction * @description: AsyncFunction
*/ */
export function isAsyncFunction<T = any>(val: unknown): val is () => Promise<T> { export function isAsyncFunction<T = any>(val: unknown): val is () => Promise<T> {
return is(val, 'AsyncFunction') return is(val, 'AsyncFunction');
} }
/** /**
* @description: promise * @description: promise
*/ */
export function isPromise<T = any>(val: unknown): val is Promise<T> { export function isPromise<T = any>(val: unknown): val is Promise<T> {
return is(val, 'Promise') && isObject(val) && isFunction(val.then) && isFunction(val.catch) return is(val, 'Promise') && isObject(val) && isFunction(val.then) && isFunction(val.catch);
} }
/** /**
* @description: * @description:
*/ */
export function isString(val: unknown): val is string { export function isString(val: unknown): val is string {
return is(val, 'String') return is(val, 'String');
} }
/** /**
* @description: boolean类型 * @description: boolean类型
*/ */
export function isBoolean(val: unknown): val is boolean { export function isBoolean(val: unknown): val is boolean {
return is(val, 'Boolean') return is(val, 'Boolean');
} }
/** /**
* @description: * @description:
*/ */
export function isArray(val: any): val is Array<any> { export function isArray(val: any): val is Array<any> {
return val && Array.isArray(val) return val && Array.isArray(val);
} }
/** /**
* @description: * @description:
*/ */
export function isClient() { export const isClient = () => {
return typeof window !== 'undefined' return typeof window !== 'undefined';
} };
/** /**
* @description: * @description:
*/ */
export function isWindow(val: any): val is Window { export const isWindow = (val: any): val is Window => {
return typeof window !== 'undefined' && is(val, 'Window') return typeof window !== 'undefined' && is(val, 'Window');
} };
export function isElement(val: unknown): val is Element { export const isElement = (val: unknown): val is Element => {
return isObject(val) && !!val.tagName return isObject(val) && !!val.tagName;
} };
export const isServer = typeof window === 'undefined' export const isServer = typeof window === 'undefined';
// 是否为图片节点 // 是否为图片节点
export function isImageDom(o: Element) { export function isImageDom(o: Element) {
return o && ['IMAGE', 'IMG'].includes(o.tagName) return o && ['IMAGE', 'IMG'].includes(o.tagName);
} }
export function isNull(val: unknown): val is null { export function isNull(val: unknown): val is null {
return val === null return val === null;
} }
export function isNullAndUnDef(val: unknown): val is null | undefined { export function isNullAndUnDef(val: unknown): val is null | undefined {
return isUnDef(val) && isNull(val) return isUnDef(val) && isNull(val);
} }
export function isNullOrUnDef(val: unknown): val is null | undefined { export function isNullOrUnDef(val: unknown): val is null | undefined {
return isUnDef(val) || isNull(val) return isUnDef(val) || isNull(val);
} }
/** /**
* @description: https * @description: https
*/ */
export function isHttps(val: string): boolean { export function isHttps(val: string): boolean {
return /^https?:\/\//.test(val) return /^https?:\/\//.test(val);
} }

View File

@ -1,33 +1,33 @@
import * as echarts from 'echarts/core' import * as echarts from 'echarts/core';
import { import {
BarChart, BarChart,
GaugeChart,
LineChart, LineChart,
PieChart,
MapChart, MapChart,
PictorialBarChart, PictorialBarChart,
PieChart,
RadarChart, RadarChart,
} from 'echarts/charts' GaugeChart,
} from 'echarts/charts';
import { import {
AriaComponent,
CalendarComponent,
DataZoomComponent,
GridComponent,
LegendComponent,
MarkLineComponent,
ParallelComponent,
PolarComponent,
RadarComponent,
TimelineComponent,
TitleComponent, TitleComponent,
ToolboxComponent,
TooltipComponent, TooltipComponent,
GridComponent,
PolarComponent,
AriaComponent,
ParallelComponent,
LegendComponent,
RadarComponent,
ToolboxComponent,
DataZoomComponent,
VisualMapComponent, VisualMapComponent,
} from 'echarts/components' TimelineComponent,
CalendarComponent,
MarkLineComponent,
} from 'echarts/components';
import { SVGRenderer } from 'echarts/renderers' import { SVGRenderer } from 'echarts/renderers';
echarts.use([ echarts.use([
LegendComponent, LegendComponent,
@ -52,6 +52,6 @@ echarts.use([
TimelineComponent, TimelineComponent,
CalendarComponent, CalendarComponent,
MarkLineComponent, MarkLineComponent,
]) ]);
export default echarts export default echarts;

View File

@ -1,9 +1,9 @@
const projectName = import.meta.env.VITE_GLOB_APP_TITLE const projectName = import.meta.env.VITE_GLOB_APP_TITLE;
export function warn(message: string) { export function warn(message: string) {
console.warn(`[${projectName} warn]:${message}`) console.warn(`[${projectName} warn]:${message}`);
} }
export function error(message: string) { export function error(message: string) {
throw new Error(`[${projectName} error]:${message}`) throw new Error(`[${projectName} error]:${message}`);
} }

View File

@ -9,17 +9,16 @@
* ==>www.baidu.com?a=3&b=4 * ==>www.baidu.com?a=3&b=4
*/ */
export function setObjToUrlParams(baseUrl: string, obj: object): string { export function setObjToUrlParams(baseUrl: string, obj: object): string {
let parameters = '' let parameters = '';
let url = '' let url = '';
for (const key in obj) { for (const key in obj) {
parameters += `${key}=${encodeURIComponent(obj[key])}&` parameters += key + '=' + encodeURIComponent(obj[key]) + '&';
} }
parameters = parameters.replace(/&$/, '') parameters = parameters.replace(/&$/, '');
if (/\?$/.test(baseUrl)) { if (/\?$/.test(baseUrl)) {
url = baseUrl + parameters url = baseUrl + parameters;
} else {
url = baseUrl.replace(/\/?$/, '?') + parameters;
} }
else { return url;
url = baseUrl.replace(/\/?$/, '?') + parameters
}
return url
} }

View File

@ -1,23 +1,19 @@
<template> <template>
<div class="h-screen flex flex-col items-center justify-center p-60px"> <div class="flex flex-col justify-center items-center h-screen p-60px">
<div class="wel-box w-full flex flex-col items-center justify-between"> <div class="wel-box flex flex-col items-center justify-between w-full">
<SvgIcon class="logo" :size="130" name="logo" /> <SvgIcon class="logo" :size="130" name="logo" />
<div class="text-darkBlue dark:text-garyWhite mb-4 mt-12 text-center text-2xl font-black"> <div class="text-darkBlue dark:text-garyWhite text-2xl font-black mt-12 mb-4 text-center"
{{ title }} >欢迎来到 {{ title }}</div
</div> >
<div class="mb-6 mt-4 w-full"> <div class="w-full mt-4 mb-6">
<van-swipe class="h-30" :autoplay="3000" :indicator-color="designStore.appTheme"> <van-swipe class="h-30" :autoplay="3000" :indicator-color="designStore.appTheme">
<van-swipe-item <van-swipe-item
class="text-gray-700 dark:text-gray-400 leading-relaxed text-center"
v-for="(text, index) in getSwipeText" v-for="(text, index) in getSwipeText"
:key="index" :key="index"
class="text-center text-gray-700 leading-relaxed dark:text-gray-400"
> >
<p class="text-lg"> <p class="text-lg">{{ text.title }}</p>
{{ text.title }} <p class="text-sm">{{ text.details }}</p>
</p>
<p class="text-sm">
{{ text.details }}
</p>
</van-swipe-item> </van-swipe-item>
</van-swipe> </van-swipe>
</div> </div>
@ -25,49 +21,45 @@
</div> </div>
</template> </template>
<script setup lang="ts"> <script setup lang="ts" name="DashboardPage">
import { computed } from 'vue' import { computed } from 'vue';
import { useDesignSettingStore } from '@/store/modules/designSetting' import { useDesignSettingStore } from '@/store/modules/designSetting';
import SvgIcon from '@/components/SvgIcon.vue' import SvgIcon from '@/components/SvgIcon.vue';
import { useGlobSetting } from '@/hooks/setting' import { useGlobSetting } from '@/hooks/setting';
defineOptions({ const designStore = useDesignSettingStore();
name: 'DashboardPage', const globSetting = useGlobSetting();
})
const designStore = useDesignSettingStore() const { title } = globSetting;
const globSetting = useGlobSetting()
const { title } = globSetting const getSwipeText = computed(() => {
return [
const getSwipeText = computed(() => { {
return [ title: '💡 最新技术栈',
{ details: '基于Vue3、Vant4、Vite、TypeScript、windiCss等最新技术栈开发',
title: '💡 最新技术栈', },
details: '基于Vue3、Vant4、Vite、TypeScript、UnoCSS等最新技术栈开发', {
}, title: '⚡️ 轻量快速的热重载',
{ details: '无论应用程序大小如何都始终极快的模块热重载HMR',
title: '⚡️ 轻量快速的热重载', },
details: '无论应用程序大小如何都始终极快的模块热重载HMR', {
}, title: '🔩 主题配置',
{ details: '具备主题配置及黑暗主题适配,且持久化保存',
title: '🔩 主题配置', },
details: '具备主题配置及黑暗主题适配,且持久化保存', {
}, title: '🛠️ 丰富的 Vite 插件',
{ details: '集成大部分 Vite 插件,无需繁琐配置,开箱即用',
title: '🛠️ 丰富的 Vite 插件', },
details: '集成大部分 Vite 插件,无需繁琐配置,开箱即用', {
}, title: '📊 内置 useEcharts hooks',
{ details: '满足大部分图表展示,只需要写你的 options',
title: '📊 内置 useEcharts hooks', },
details: '满足大部分图表展示,只需要写你的 options', {
}, title: '🥳 完善的登录系统、路由、Axios配置',
{ details: '所有架构已搭建完毕,你可以直接开发你的业务需求',
title: '🥳 完善的登录系统、路由、Axios配置', },
details: '所有架构已搭建完毕,你可以直接开发你的业务需求', ];
}, });
]
})
</script> </script>
<style scoped lang="less"></style> <style scoped lang="less"></style>

View File

@ -1,45 +1,40 @@
<template> <template>
<div class="page-container flex flex-col justify-center"> <div class="flex flex-col justify-center page-container">
<div class="text-center"> <div class="text-center">
<img src="~@/assets/icons/exception/403.svg" alt=""> <img src="~@/assets/icons/exception/403.svg" alt="" />
</div> </div>
<div class="text-center"> <div class="text-center">
<h1 class="text-base text-gray-500"> <h1 class="text-base text-gray-500">抱歉你无权访问该页面</h1>
抱歉你无权访问该页面 <n-button type="info" @click="goHome">回到首页</n-button>
</h1>
<n-button type="info" @click="goHome">
回到首页
</n-button>
</div> </div>
</div> </div>
</template> </template>
<script lang="ts" setup> <script lang="ts" setup>
import { useRouter } from 'vue-router' import { useRouter } from 'vue-router';
const router = useRouter();
const router = useRouter() function goHome() {
function goHome() { router.push('/');
router.push('/') }
}
</script> </script>
<style lang="less" scoped> <style lang="less" scoped>
.page-container { .page-container {
width: 100%; width: 100%;
border-radius: 4px; border-radius: 4px;
padding: 50px 0; padding: 50px 0;
height: 100vh; height: 100vh;
.text-center { .text-center {
h1 { h1 {
color: #666; color: #666;
padding: 20px 0; padding: 20px 0;
}
}
img {
width: 350px;
margin: 0 auto;
} }
} }
img {
width: 350px;
margin: 0 auto;
}
}
</style> </style>

View File

@ -1,42 +1,37 @@
<template> <template>
<div class="page-container flex flex-col justify-center"> <div class="flex flex-col justify-center page-container">
<div class="text-center"> <div class="text-center">
<img src="~@/assets/icons/exception/404.svg" alt=""> <img src="~@/assets/icons/exception/404.svg" alt="" />
</div> </div>
<div class="text-center"> <div class="text-center">
<h1 class="text-base text-gray-500"> <h1 class="text-base text-gray-500">抱歉你访问的页面不存在</h1>
抱歉你访问的页面不存在 <van-button type="primary" @click="goHome">回到首页</van-button>
</h1>
<van-button type="primary" @click="goHome">
回到首页
</van-button>
</div> </div>
</div> </div>
</template> </template>
<script lang="ts" setup> <script lang="ts" setup>
import { useRouter } from 'vue-router' import { useRouter } from 'vue-router';
const router = useRouter();
const router = useRouter() function goHome() {
function goHome() { router.push('/');
router.push('/') }
}
</script> </script>
<style lang="less" scoped> <style lang="less" scoped>
.page-container { .page-container {
height: 100%; height: 100%;
.text-center { .text-center {
h1 { h1 {
color: #666; color: #666;
padding: 20px 0; padding: 20px 0;
}
}
img {
width: 350px;
margin: 0 auto;
} }
} }
img {
width: 350px;
margin: 0 auto;
}
}
</style> </style>

View File

@ -1,45 +1,40 @@
<template> <template>
<div class="page-container flex flex-col justify-center"> <div class="flex flex-col justify-center page-container">
<div class="text-center"> <div class="text-center">
<img src="~@/assets/icons/exception/500.svg" alt=""> <img src="~@/assets/icons/exception/500.svg" alt="" />
</div> </div>
<div class="text-center"> <div class="text-center">
<h1 class="text-base text-gray-500"> <h1 class="text-base text-gray-500">抱歉服务器出错了</h1>
抱歉服务器出错了 <n-button type="info" @click="goHome">回到首页</n-button>
</h1>
<n-button type="info" @click="goHome">
回到首页
</n-button>
</div> </div>
</div> </div>
</template> </template>
<script lang="ts" setup> <script lang="ts" setup>
import { useRouter } from 'vue-router' import { useRouter } from 'vue-router';
const router = useRouter();
const router = useRouter() function goHome() {
function goHome() { router.push('/');
router.push('/') }
}
</script> </script>
<style lang="less" scoped> <style lang="less" scoped>
.page-container { .page-container {
width: 100%; width: 100%;
border-radius: 4px; border-radius: 4px;
padding: 50px 0; padding: 50px 0;
height: 100vh; height: 100vh;
.text-center { .text-center {
h1 { h1 {
color: #666; color: #666;
padding: 20px 0; padding: 20px 0;
}
}
img {
width: 350px;
margin: 0 auto;
} }
} }
img {
width: 350px;
margin: 0 auto;
}
}
</style> </style>

View File

@ -1,8 +1,8 @@
<template> <template>
<van-form v-if="getShow" ref="formRef" class="flex flex-col items-center" @submit="handleReset"> <van-form ref="formRef" v-if="getShow" class="flex flex-col items-center" @submit="handleReset">
<van-field <van-field
class="enter-y items-center mb-25px !rounded-md"
v-model="formData.username" v-model="formData.username"
class="enter-y mb-25px items-center !rounded-md"
name="username" name="username"
placeholder="用户名" placeholder="用户名"
:rules="getFormRules.username" :rules="getFormRules.username"
@ -15,8 +15,8 @@
</van-field> </van-field>
<van-field <van-field
class="enter-y items-center mb-25px !rounded-md"
v-model="formData.mobile" v-model="formData.mobile"
class="enter-y mb-25px items-center !rounded-md"
name="password" name="password"
placeholder="手机号码" placeholder="手机号码"
:rules="getFormRules.mobile" :rules="getFormRules.mobile"
@ -29,8 +29,8 @@
</van-field> </van-field>
<van-field <van-field
class="enter-y items-center mb-70px !rounded-md"
v-model="formData.sms" v-model="formData.sms"
class="enter-y mb-70px items-center !rounded-md"
center center
clearable clearable
placeholder="请输入短信验证码" placeholder="请输入短信验证码"
@ -42,9 +42,7 @@
</Icon> </Icon>
</template> </template>
<template #button> <template #button>
<van-button size="small" type="primary"> <van-button size="small" type="primary">发送验证码</van-button>
发送验证码
</van-button>
</template> </template>
</van-field> </van-field>
@ -71,40 +69,39 @@
</template> </template>
<script setup lang="ts"> <script setup lang="ts">
import { computed, reactive, ref, unref } from 'vue' import { computed, reactive, ref, unref } from 'vue';
import type { FormInstance } from 'vant' import type { FormInstance } from 'vant';
import { Icon } from '@vicons/utils' import { Icon } from '@vicons/utils';
import { EditOutlined, MobileOutlined, UserOutlined } from '@vicons/antd' import { UserOutlined, MobileOutlined, EditOutlined } from '@vicons/antd';
import { LoginStateEnum, useFormRules, useLoginState } from './useLogin' import { LoginStateEnum, useLoginState, useFormRules } from './useLogin';
const { handleBackLogin, getLoginState } = useLoginState() const { handleBackLogin, getLoginState } = useLoginState();
const { getFormRules } = useFormRules() const { getFormRules } = useFormRules();
const getShow = computed(() => unref(getLoginState) === LoginStateEnum.RESET_PASSWORD) const getShow = computed(() => unref(getLoginState) === LoginStateEnum.RESET_PASSWORD);
const loading = ref(false) const loading = ref(false);
const formRef = ref<FormInstance>() const formRef = ref<FormInstance>();
const formData = reactive({ const formData = reactive({
username: '', username: '',
mobile: '', mobile: '',
sms: '', sms: '',
}) });
function handleReset() { function handleReset() {
formRef.value formRef.value
?.validate() ?.validate()
.then(async () => { .then(async () => {
try { try {
loading.value = true loading.value = true;
// do something // do something
} } finally {
finally { loading.value = false;
loading.value = false }
} })
}) .catch(() => {
.catch(() => { console.error('验证失败');
console.error('验证失败') });
}) }
}
</script> </script>
<style scoped lang="less"></style> <style scoped lang="less"></style>

View File

@ -1,7 +1,7 @@
<template> <template>
<div> <div>
<div class="h-screen flex justify-center p-8"> <div class="flex justify-center h-screen p-8">
<div class="w-full flex flex-col"> <div class="flex flex-col w-full">
<LoginTitle /> <LoginTitle />
<LoginForm /> <LoginForm />
<ForgetPasswordForm /> <ForgetPasswordForm />
@ -13,18 +13,18 @@
</template> </template>
<script setup lang="ts"> <script setup lang="ts">
import LoginTitle from './LoginTitle.vue' import LoginTitle from './LoginTitle.vue';
import LoginForm from './LoginForm.vue' import LoginForm from './LoginForm.vue';
import ForgetPasswordForm from './ForgetPasswordForm.vue' import ForgetPasswordForm from './ForgetPasswordForm.vue';
import RegisterForm from './RegisterForm.vue' import RegisterForm from './RegisterForm.vue';
import LoginWave from './LoginWave.vue' import LoginWave from './LoginWave.vue';
</script> </script>
<style scoped lang="less"> <style scoped lang="less">
:deep(.van-field__left-icon) { :deep(.van-field__left-icon) {
display: flex; display: flex;
} }
:deep(.van-field__right-icon) { :deep(.van-field__right-icon) {
display: flex; display: flex;
} }
</style> </style>

View File

@ -1,8 +1,8 @@
<template> <template>
<van-form v-if="getShow" ref="formRef" class="flex flex-col items-center" @submit="handleSubmit"> <van-form ref="formRef" v-if="getShow" class="flex flex-col items-center" @submit="handleSubmit">
<van-field <van-field
class="enter-y items-center mb-25px !rounded-md"
v-model="formData.username" v-model="formData.username"
class="enter-y mb-25px items-center !rounded-md"
name="username" name="username"
placeholder="用户名" placeholder="用户名"
:rules="getFormRules.username" :rules="getFormRules.username"
@ -14,8 +14,8 @@
</template> </template>
</van-field> </van-field>
<van-field <van-field
class="enter-y items-center mb-25px !rounded-md"
v-model="formData.password" v-model="formData.password"
class="enter-y mb-25px items-center !rounded-md"
:type="switchPassType ? 'password' : 'text'" :type="switchPassType ? 'password' : 'text'"
name="password" name="password"
placeholder="密码" placeholder="密码"
@ -37,16 +37,16 @@
</template> </template>
</van-field> </van-field>
<div class="enter-y mb-100px w-full flex justify-between px-5px"> <div class="enter-y w-full px-5px flex justify-between mb-100px">
<div class="flex items-center"> <div class="flex items-center">
<van-switch v-model="rememberMe" class="mr-8px !text-30px" /> <van-switch class="mr-8px !text-30px" v-model="rememberMe" />
<span class="!text-25px">记住我</span> <span class="!text-25px">记住我</span>
</div> </div>
<a class="!text-25px" @click="setLoginState(LoginStateEnum.RESET_PASSWORD)">忘记密码?</a> <a class="!text-25px" @click="setLoginState(LoginStateEnum.RESET_PASSWORD)">忘记密码?</a>
</div> </div>
<van-button <van-button
class="enter-y !mb-25px !rounded-md" class="enter-y !rounded-md !mb-25px"
type="primary" type="primary"
block block
native-type="submit" native-type="submit"
@ -67,69 +67,64 @@
</template> </template>
<script setup lang="ts"> <script setup lang="ts">
import { computed, onMounted, reactive, ref, unref } from 'vue' import { computed, onMounted, reactive, ref, unref } from 'vue';
import { useRoute, useRouter } from 'vue-router' import { useRoute, useRouter } from 'vue-router';
import { showFailToast, showLoadingToast, showSuccessToast } from 'vant' import { showFailToast, showLoadingToast, showSuccessToast } from 'vant';
import type { FormInstance } from 'vant' import type { FormInstance } from 'vant';
import { Icon } from '@vicons/utils' import { Icon } from '@vicons/utils';
import { EyeInvisibleOutlined, EyeOutlined, LockOutlined, UserOutlined } from '@vicons/antd' import { UserOutlined, LockOutlined, EyeOutlined, EyeInvisibleOutlined } from '@vicons/antd';
import { LoginStateEnum, useFormRules, useLoginState } from './useLogin' import { useUserStore } from '@/store/modules/user';
import { useUserStore } from '@/store/modules/user' import { ResultEnum } from '@/enums/httpEnum';
import { ResultEnum } from '@/enums/httpEnum' import { PageEnum } from '@/enums/pageEnum';
import { PageEnum } from '@/enums/pageEnum' import { LoginStateEnum, useLoginState, useFormRules } from './useLogin';
const { setLoginState, getLoginState } = useLoginState() const { setLoginState, getLoginState } = useLoginState();
const { getFormRules } = useFormRules() const { getFormRules } = useFormRules();
const userStore = useUserStore() const userStore = useUserStore();
const router = useRouter() const router = useRouter();
const route = useRoute() const route = useRoute();
const formRef = ref<FormInstance>() const formRef = ref<FormInstance>();
const loading = ref(false) const loading = ref(false);
const rememberMe = ref(false) const rememberMe = ref(false);
const switchPassType = ref(true) const switchPassType = ref(true);
const formData = reactive({ const formData = reactive({
username: 'admin', username: 'admin',
password: '123456', password: '123456',
}) });
const getShow = computed(() => unref(getLoginState) === LoginStateEnum.LOGIN) const getShow = computed(() => unref(getLoginState) === LoginStateEnum.LOGIN);
function handleSubmit() { function handleSubmit() {
formRef.value formRef.value
?.validate() ?.validate()
.then(async () => { .then(async () => {
try { try {
loading.value = true loading.value = true;
showLoadingToast('登录中...') showLoadingToast('登录中...');
const { code, message: msg } = await userStore.Login({ const { code, message: msg } = await userStore.Login({
username: formData.username, username: formData.username,
password: formData.password, password: formData.password,
}) });
if (code === ResultEnum.SUCCESS) { if (code == ResultEnum.SUCCESS) {
const toPath = decodeURIComponent((route.query?.redirect || '/') as string) const toPath = decodeURIComponent((route.query?.redirect || '/') as string);
showSuccessToast('登录成功,即将进入系统') showSuccessToast('登录成功,即将进入系统');
if (route.name === PageEnum.BASE_LOGIN_NAME) { if (route.name === PageEnum.BASE_LOGIN_NAME) {
router.replace('/') router.replace('/');
} } else router.replace(toPath);
else { } else {
router.replace(toPath) showFailToast(msg || '登录失败');
} }
} finally {
loading.value = false;
} }
else { })
showFailToast(msg || '登录失败') .catch(() => {
} console.error('验证失败');
} });
finally { }
loading.value = false
}
})
.catch(() => {
console.error('验证失败')
})
}
onMounted(() => {}) onMounted(() => {});
</script> </script>
<style scoped lang="less"></style> <style scoped lang="less"></style>

View File

@ -1,20 +1,20 @@
<template> <template>
<div class="flex flex-col items-center justify-center"> <div class="flex flex-col items-center justify-center">
<div class="logo enter-y my-35px"> <div class="logo my-35px enter-y">
<SvgIcon class="!h-250px !w-250px" name="logo" /> <SvgIcon class="!h-250px !w-250px" name="logo" />
</div> </div>
<div class="text-darkBlue dark:text-garyWhite enter-y mb-80px text-45px font-black"> <div class="mb-80px text-darkBlue dark:text-garyWhite text-45px font-black enter-y">
{{ title }} {{ title }}
</div> </div>
</div> </div>
</template> </template>
<script setup lang="ts"> <script setup lang="ts">
import { useGlobSetting } from '@/hooks/setting' import { useGlobSetting } from '@/hooks/setting';
const globSetting = useGlobSetting() const globSetting = useGlobSetting();
const { title } = globSetting const { title } = globSetting;
</script> </script>
<style scoped lang="less"></style> <style scoped lang="less"></style>

View File

@ -1,5 +1,5 @@
<template> <template>
<div class="enter-y wave-wrapper fixed bottom-0 w-full !-z-5"> <div class="enter-y fixed bottom-0 w-full !-z-5 wave-wrapper">
<svg <svg
class="ignore-waves" class="ignore-waves"
xmlns="http://www.w3.org/2000/svg" xmlns="http://www.w3.org/2000/svg"
@ -45,61 +45,61 @@
</template> </template>
<script setup lang="ts"> <script setup lang="ts">
import { useDesignSettingStore } from '@/store/modules/designSetting' import { useDesignSettingStore } from '@/store/modules/designSetting';
import { hexToRgba } from '@/utils/index' import { hexToRgba } from '@/utils/index';
const designStore = useDesignSettingStore() const designStore = useDesignSettingStore();
</script> </script>
<style scoped lang="less"> <style scoped lang="less">
.wave-wrapper { .wave-wrapper {
position: fixed; position: fixed;
width: 100%; width: 100%;
left: 0; left: 0;
bottom: 0; bottom: 0;
} }
.ignore-waves { .ignore-waves {
position: relative; position: relative;
display: block; display: block;
width: 100%; width: 100%;
height: 50px; height: 50px;
min-height: 40px; min-height: 40px;
max-height: 80px; max-height: 80px;
}
/* Animation */
.parallax > use {
animation: move-forever 12s cubic-bezier(0.55, 0.5, 0.45, 0.5) infinite;
animation: move-forever 12s linear infinite;
}
.parallax > use:nth-child(1) {
animation-delay: -2s;
animation-duration: 7s;
}
.parallax > use:nth-child(2) {
animation-delay: -3s;
animation-duration: 10s;
}
.parallax > use:nth-child(3) {
animation-delay: -4s;
animation-duration: 13s;
}
.parallax > use:nth-child(4) {
animation-delay: -5s;
animation-duration: 16s;
}
@keyframes move-forever {
0% {
transform: translate3d(-90px, 0, 0);
} }
100% { /* Animation */
transform: translate3d(85px, 0, 0); .parallax > use {
animation: move-forever 12s cubic-bezier(0.55, 0.5, 0.45, 0.5) infinite;
animation: move-forever 12s linear infinite;
}
.parallax > use:nth-child(1) {
animation-delay: -2s;
animation-duration: 7s;
}
.parallax > use:nth-child(2) {
animation-delay: -3s;
animation-duration: 10s;
}
.parallax > use:nth-child(3) {
animation-delay: -4s;
animation-duration: 13s;
}
.parallax > use:nth-child(4) {
animation-delay: -5s;
animation-duration: 16s;
}
@keyframes move-forever {
0% {
transform: translate3d(-90px, 0, 0);
}
100% {
transform: translate3d(85px, 0, 0);
}
} }
}
</style> </style>

View File

@ -1,9 +1,9 @@
<template> <template>
<van-form v-if="getShow" ref="formRef" class="flex flex-col" @submit="handleRegister"> <van-form ref="formRef" v-if="getShow" class="flex flex-col" @submit="handleRegister">
<van-cell-group inset class="enter-y !mx-0 !mb-60px"> <van-cell-group inset class="enter-y !mx-0 !mb-60px">
<van-field <van-field
v-model="formData.username"
class="enter-y items-center !rounded-md" class="enter-y items-center !rounded-md"
v-model="formData.username"
name="username" name="username"
placeholder="用户名" placeholder="用户名"
:rules="getFormRules.username" :rules="getFormRules.username"
@ -16,8 +16,8 @@
</van-field> </van-field>
<van-field <van-field
v-model="formData.mobile"
class="enter-y items-center !rounded-md" class="enter-y items-center !rounded-md"
v-model="formData.mobile"
name="password" name="password"
placeholder="手机号码" placeholder="手机号码"
:rules="getFormRules.mobile" :rules="getFormRules.mobile"
@ -30,8 +30,8 @@
</van-field> </van-field>
<van-field <van-field
v-model="formData.sms"
class="enter-y items-center !rounded-md" class="enter-y items-center !rounded-md"
v-model="formData.sms"
center center
clearable clearable
placeholder="请输入短信验证码" placeholder="请输入短信验证码"
@ -43,15 +43,13 @@
</Icon> </Icon>
</template> </template>
<template #button> <template #button>
<van-button size="small" type="primary"> <van-button size="small" type="primary">发送验证码</van-button>
发送验证码
</van-button>
</template> </template>
</van-field> </van-field>
<van-field <van-field
v-model="formData.password"
class="enter-y items-center !rounded-md" class="enter-y items-center !rounded-md"
v-model="formData.password"
:type="switchPassType ? 'password' : 'text'" :type="switchPassType ? 'password' : 'text'"
name="password" name="password"
placeholder="密码" placeholder="密码"
@ -74,8 +72,8 @@
</van-field> </van-field>
<van-field <van-field
v-model="formData.confirmPassword"
class="enter-y items-center !rounded-md" class="enter-y items-center !rounded-md"
v-model="formData.confirmPassword"
:type="switchConfirmPassType ? 'password' : 'text'" :type="switchConfirmPassType ? 'password' : 'text'"
name="confirmPassword" name="confirmPassword"
placeholder="确认密码" placeholder="确认密码"
@ -133,59 +131,58 @@
</template> </template>
<script setup lang="ts"> <script setup lang="ts">
import { computed, reactive, ref, unref } from 'vue' import { computed, reactive, ref, unref } from 'vue';
import type { FormInstance } from 'vant' import type { FormInstance } from 'vant';
import { Icon } from '@vicons/utils' import { Icon } from '@vicons/utils';
import { import {
EditOutlined, UserOutlined,
EyeInvisibleOutlined, MobileOutlined,
EyeOutlined, EditOutlined,
LockOutlined, LockOutlined,
MobileOutlined, EyeOutlined,
UserOutlined, EyeInvisibleOutlined,
} from '@vicons/antd' } from '@vicons/antd';
import { LoginStateEnum, useFormRules, useLoginState } from './useLogin' import { LoginStateEnum, useLoginState, useFormRules } from './useLogin';
const { handleBackLogin, getLoginState } = useLoginState() const { handleBackLogin, getLoginState } = useLoginState();
const getShow = computed(() => unref(getLoginState) === LoginStateEnum.REGISTER) const getShow = computed(() => unref(getLoginState) === LoginStateEnum.REGISTER);
const loading = ref(false) const loading = ref(false);
const formRef = ref<FormInstance>() const formRef = ref<FormInstance>();
const formData = reactive({ const formData = reactive({
username: '', username: '',
mobile: '', mobile: '',
sms: '', sms: '',
password: '', password: '',
confirmPassword: '', confirmPassword: '',
policy: false, policy: false,
}) });
const { getFormRules } = useFormRules(formData) const { getFormRules } = useFormRules(formData);
const switchPassType = ref(true) const switchPassType = ref(true);
const switchConfirmPassType = ref(true) const switchConfirmPassType = ref(true);
function handleRegister() { function handleRegister() {
formRef.value formRef.value
?.validate() ?.validate()
.then(async () => { .then(async () => {
try { try {
loading.value = true loading.value = true;
// do something // do something
console.log('%c [ ]-167', 'font-size:13px; background:pink; color:#bf2c9f;') console.log('%c [ ]-167', 'font-size:13px; background:pink; color:#bf2c9f;');
} } finally {
finally { loading.value = false;
loading.value = false
console.log('%c [ ]-171', 'font-size:13px; background:pink; color:#bf2c9f;') console.log('%c [ ]-171', 'font-size:13px; background:pink; color:#bf2c9f;');
} }
}) })
.catch(() => { .catch(() => {
console.error('验证失败') console.error('验证失败');
}) });
} }
</script> </script>
<style scoped lang="less"></style> <style scoped lang="less"></style>

View File

@ -1,5 +1,5 @@
import type { FieldRule } from 'vant' import type { FieldRule } from 'vant';
import { computed, ref, unref } from 'vue' import { computed, ref, unref } from 'vue';
export enum LoginStateEnum { export enum LoginStateEnum {
LOGIN, LOGIN,
@ -7,54 +7,54 @@ export enum LoginStateEnum {
RESET_PASSWORD, RESET_PASSWORD,
} }
const currentState = ref(LoginStateEnum.LOGIN) const currentState = ref(LoginStateEnum.LOGIN);
export function useLoginState() { export function useLoginState() {
function setLoginState(state: LoginStateEnum) { function setLoginState(state: LoginStateEnum) {
currentState.value = state currentState.value = state;
} }
const getLoginState = computed(() => currentState.value) const getLoginState = computed(() => currentState.value);
function handleBackLogin() { function handleBackLogin() {
setLoginState(LoginStateEnum.LOGIN) setLoginState(LoginStateEnum.LOGIN);
} }
return { setLoginState, getLoginState, handleBackLogin } return { setLoginState, getLoginState, handleBackLogin };
} }
export function useFormRules(formData?: Recordable) { export function useFormRules(formData?: Recordable) {
const getUsernameFormRule = computed(() => createRule('请输入用户名')) const getUsernameFormRule = computed(() => createRule('请输入用户名'));
const getPasswordFormRule = computed(() => createRule('请输入密码')) const getPasswordFormRule = computed(() => createRule('请输入密码'));
const getSmsFormRule = computed(() => createRule('请输入短信验证码')) const getSmsFormRule = computed(() => createRule('请输入短信验证码'));
const getMobileFormRule = computed(() => createRule('请输入手机号码')) const getMobileFormRule = computed(() => createRule('请输入手机号码'));
const validatePolicy = async (value: any, _: FieldRule) => { const validatePolicy = async (value: any, _: FieldRule) => {
return !value ? Promise.resolve('勾选后才能注册') : Promise.resolve(true) return !value ? Promise.resolve('勾选后才能注册') : Promise.resolve(true);
} };
const validateConfirmPassword = (password: string) => { const validateConfirmPassword = (password: string) => {
return async (value: string) => { return async (value: string) => {
if (!value) { if (!value) {
return Promise.resolve('请输入确认密码') return Promise.resolve('请输入确认密码');
} }
if (value !== password) { if (value !== password) {
return Promise.resolve('两次输入密码不一致') return Promise.resolve('两次输入密码不一致');
} }
return Promise.resolve(true) return Promise.resolve(true);
} };
} };
const getFormRules = computed((): { [k: string]: FieldRule[] } => { const getFormRules = computed((): { [k: string]: FieldRule[] } => {
const usernameFormRule = unref(getUsernameFormRule) const usernameFormRule = unref(getUsernameFormRule);
const passwordFormRule = unref(getPasswordFormRule) const passwordFormRule = unref(getPasswordFormRule);
const smsFormRule = unref(getSmsFormRule) const smsFormRule = unref(getSmsFormRule);
const mobileFormRule = unref(getMobileFormRule) const mobileFormRule = unref(getMobileFormRule);
const mobileRule = { const mobileRule = {
sms: smsFormRule, sms: smsFormRule,
mobile: mobileFormRule, mobile: mobileFormRule,
} };
switch (unref(currentState)) { switch (unref(currentState)) {
// register form rules // register form rules
case LoginStateEnum.REGISTER: case LoginStateEnum.REGISTER:
@ -66,24 +66,24 @@ export function useFormRules(formData?: Recordable) {
], ],
policy: [{ validator: validatePolicy, trigger: 'onBlur' }], policy: [{ validator: validatePolicy, trigger: 'onBlur' }],
...mobileRule, ...mobileRule,
} };
// reset password form rules // reset password form rules
case LoginStateEnum.RESET_PASSWORD: case LoginStateEnum.RESET_PASSWORD:
return { return {
username: usernameFormRule, username: usernameFormRule,
...mobileRule, ...mobileRule,
} };
// login form rules // login form rules
default: default:
return { return {
username: usernameFormRule, username: usernameFormRule,
password: passwordFormRule, password: passwordFormRule,
} };
} }
}) });
return { getFormRules } return { getFormRules };
} }
function createRule(message: string): FieldRule[] { function createRule(message: string): FieldRule[] {
@ -93,5 +93,5 @@ function createRule(message: string): FieldRule[] {
message, message,
trigger: 'onBlur', trigger: 'onBlur',
}, },
] ];
} }

View File

@ -1,107 +1,106 @@
<template> <template>
<div class="my-card m-40px rounded-2xl p-30px shadow-xl"> <div class="my-card m-40px p-30px rounded-2xl shadow-xl">
<div ref="chartRef" :style="{ height: '350px' }" /> <div ref="chartRef" :style="{ height: '350px' }"></div>
</div> </div>
</template> </template>
<script setup lang="ts"> <script setup lang="ts">
import type { Ref } from 'vue' import { useECharts } from '@/hooks/web/useECharts';
import { onMounted, ref } from 'vue' import { onMounted, ref, Ref } from 'vue';
import type { EChartsOption } from 'echarts' import type { EChartsOption } from 'echarts';
import { useECharts } from '@/hooks/web/useECharts'
const chartRef = ref<HTMLDivElement | null>(null) const chartRef = ref<HTMLDivElement | null>(null);
const { setOptions } = useECharts(chartRef as Ref<HTMLDivElement>) const { setOptions } = useECharts(chartRef as Ref<HTMLDivElement>);
const chartOptions: EChartsOption = { const chartOptions: EChartsOption = {
tooltip: { tooltip: {
trigger: 'axis', trigger: 'axis',
axisPointer: { axisPointer: {
// Use axis to trigger tooltip // Use axis to trigger tooltip
type: 'shadow', // 'shadow' as default; can also be 'line' or 'shadow' type: 'shadow', // 'shadow' as default; can also be 'line' or 'shadow'
},
}, },
}, legend: {},
legend: {}, grid: {
grid: { left: '1%',
left: '1%', right: '7%',
right: '7%', bottom: '3%',
bottom: '3%', containLabel: true,
containLabel: true,
},
xAxis: {
type: 'value',
},
yAxis: {
type: 'category',
data: ['Mon', 'Tue', 'Wed', 'Thu', 'Fri', 'Sat', 'Sun'],
},
series: [
{
name: 'Direct',
type: 'bar',
stack: 'total',
label: {
show: true,
},
emphasis: {
focus: 'series',
},
data: [320, 302, 301, 334, 390, 330, 320],
}, },
{ xAxis: {
name: 'Mail Ad', type: 'value',
type: 'bar',
stack: 'total',
label: {
show: true,
},
emphasis: {
focus: 'series',
},
data: [120, 132, 101, 134, 90, 230, 210],
}, },
{ yAxis: {
name: 'Affiliate Ad', type: 'category',
type: 'bar', data: ['Mon', 'Tue', 'Wed', 'Thu', 'Fri', 'Sat', 'Sun'],
stack: 'total',
label: {
show: true,
},
emphasis: {
focus: 'series',
},
data: [220, 182, 191, 234, 290, 330, 310],
}, },
{ series: [
name: 'Video Ad', {
type: 'bar', name: 'Direct',
stack: 'total', type: 'bar',
label: { stack: 'total',
show: true, label: {
show: true,
},
emphasis: {
focus: 'series',
},
data: [320, 302, 301, 334, 390, 330, 320],
}, },
emphasis: { {
focus: 'series', name: 'Mail Ad',
type: 'bar',
stack: 'total',
label: {
show: true,
},
emphasis: {
focus: 'series',
},
data: [120, 132, 101, 134, 90, 230, 210],
}, },
data: [150, 212, 201, 154, 190, 330, 410], {
}, name: 'Affiliate Ad',
{ type: 'bar',
name: 'Search Engine', stack: 'total',
type: 'bar', label: {
stack: 'total', show: true,
label: { },
show: true, emphasis: {
focus: 'series',
},
data: [220, 182, 191, 234, 290, 330, 310],
}, },
emphasis: { {
focus: 'series', name: 'Video Ad',
type: 'bar',
stack: 'total',
label: {
show: true,
},
emphasis: {
focus: 'series',
},
data: [150, 212, 201, 154, 190, 330, 410],
}, },
data: [820, 832, 901, 934, 1290, 1330, 1320], {
}, name: 'Search Engine',
], type: 'bar',
} stack: 'total',
label: {
show: true,
},
emphasis: {
focus: 'series',
},
data: [820, 832, 901, 934, 1290, 1330, 1320],
},
],
};
onMounted(() => { onMounted(() => {
setOptions(chartOptions) setOptions(chartOptions);
}) });
</script> </script>
<style scoped></style> <style scoped></style>

View File

@ -7,9 +7,9 @@
</template> </template>
<script setup lang="ts"> <script setup lang="ts">
import lineChart from './lineChart.vue' import lineChart from './lineChart.vue';
import barChart from './barChart.vue' import barChart from './barChart.vue';
import pieChart from './pieChart.vue' import pieChart from './pieChart.vue';
</script> </script>
<style scoped></style> <style scoped></style>

View File

@ -1,120 +1,119 @@
<template> <template>
<div class="my-card m-40px rounded-2xl p-30px shadow-xl"> <div class="my-card m-40px p-30px rounded-2xl shadow-xl">
<div ref="chartRef" :style="{ height: '350px' }" /> <div ref="chartRef" :style="{ height: '350px' }"></div>
</div> </div>
</template> </template>
<script setup lang="ts"> <script setup lang="ts">
import type { Ref } from 'vue' import { useECharts } from '@/hooks/web/useECharts';
import { onMounted, ref } from 'vue' import { onMounted, ref, Ref } from 'vue';
import type { EChartsOption } from 'echarts' import type { EChartsOption } from 'echarts';
import { useECharts } from '@/hooks/web/useECharts'
const chartRef = ref<HTMLDivElement | null>(null) const chartRef = ref<HTMLDivElement | null>(null);
const { setOptions } = useECharts(chartRef as Ref<HTMLDivElement>) const { setOptions } = useECharts(chartRef as Ref<HTMLDivElement>);
const chartOptions: EChartsOption = { const chartOptions: EChartsOption = {
title: { title: {
text: 'Stacked Area Chart', text: 'Stacked Area Chart',
}, },
tooltip: { tooltip: {
trigger: 'axis', trigger: 'axis',
axisPointer: { axisPointer: {
type: 'cross', type: 'cross',
label: { label: {
backgroundColor: '#6a7985', backgroundColor: '#6a7985',
},
}, },
}, },
}, legend: {
legend: { data: ['Email', 'Union Ads', 'Video Ads', 'Direct', 'Search Engine'],
data: ['Email', 'Union Ads', 'Video Ads', 'Direct', 'Search Engine'], top: '10%',
top: '10%',
},
toolbox: {
feature: {
saveAsImage: {},
}, },
}, toolbox: {
grid: { feature: {
top: '30%', saveAsImage: {},
left: '3%',
right: '4%',
bottom: '3%',
containLabel: true,
},
xAxis: [
{
type: 'category',
boundaryGap: false,
data: ['Mon', 'Tue', 'Wed', 'Thu', 'Fri', 'Sat', 'Sun'],
},
],
yAxis: [
{
type: 'value',
},
],
series: [
{
name: 'Email',
type: 'line',
stack: 'Total',
areaStyle: {},
emphasis: {
focus: 'series',
}, },
data: [120, 132, 101, 134, 90, 230, 210],
}, },
{ grid: {
name: 'Union Ads', top: '30%',
type: 'line', left: '3%',
stack: 'Total', right: '4%',
areaStyle: {}, bottom: '3%',
emphasis: { containLabel: true,
focus: 'series',
},
data: [220, 182, 191, 234, 290, 330, 310],
}, },
{ xAxis: [
name: 'Video Ads', {
type: 'line', type: 'category',
stack: 'Total', boundaryGap: false,
areaStyle: {}, data: ['Mon', 'Tue', 'Wed', 'Thu', 'Fri', 'Sat', 'Sun'],
emphasis: {
focus: 'series',
}, },
data: [150, 232, 201, 154, 190, 330, 410], ],
}, yAxis: [
{ {
name: 'Direct', type: 'value',
type: 'line',
stack: 'Total',
areaStyle: {},
emphasis: {
focus: 'series',
}, },
data: [320, 332, 301, 334, 390, 330, 320], ],
}, series: [
{ {
name: 'Search Engine', name: 'Email',
type: 'line', type: 'line',
stack: 'Total', stack: 'Total',
label: { areaStyle: {},
show: true, emphasis: {
position: 'top', focus: 'series',
},
data: [120, 132, 101, 134, 90, 230, 210],
}, },
areaStyle: {}, {
emphasis: { name: 'Union Ads',
focus: 'series', type: 'line',
stack: 'Total',
areaStyle: {},
emphasis: {
focus: 'series',
},
data: [220, 182, 191, 234, 290, 330, 310],
}, },
data: [820, 932, 901, 934, 1290, 1330, 1320], {
}, name: 'Video Ads',
], type: 'line',
} stack: 'Total',
areaStyle: {},
emphasis: {
focus: 'series',
},
data: [150, 232, 201, 154, 190, 330, 410],
},
{
name: 'Direct',
type: 'line',
stack: 'Total',
areaStyle: {},
emphasis: {
focus: 'series',
},
data: [320, 332, 301, 334, 390, 330, 320],
},
{
name: 'Search Engine',
type: 'line',
stack: 'Total',
label: {
show: true,
position: 'top',
},
areaStyle: {},
emphasis: {
focus: 'series',
},
data: [820, 932, 901, 934, 1290, 1330, 1320],
},
],
};
onMounted(() => { onMounted(() => {
setOptions(chartOptions) setOptions(chartOptions);
}) });
</script> </script>
<style scoped></style> <style scoped></style>

View File

@ -1,66 +1,65 @@
<template> <template>
<div class="my-card m-40px rounded-2xl p-30px shadow-xl"> <div class="my-card m-40px p-30px rounded-2xl shadow-xl">
<div ref="chartRef" :style="{ height: '350px' }" /> <div ref="chartRef" :style="{ height: '350px' }"></div>
</div> </div>
</template> </template>
<script setup lang="ts"> <script setup lang="ts">
import type { Ref } from 'vue' import { useECharts } from '@/hooks/web/useECharts';
import { onMounted, ref } from 'vue' import { onMounted, ref, Ref } from 'vue';
import type { EChartsOption } from 'echarts' import type { EChartsOption } from 'echarts';
import { useECharts } from '@/hooks/web/useECharts'
const chartRef = ref<HTMLDivElement | null>(null) const chartRef = ref<HTMLDivElement | null>(null);
const { setOptions } = useECharts(chartRef as Ref<HTMLDivElement>) const { setOptions } = useECharts(chartRef as Ref<HTMLDivElement>);
const chartOptions: EChartsOption = { const chartOptions: EChartsOption = {
tooltip: { tooltip: {
trigger: 'item', trigger: 'item',
},
legend: {
top: '5%',
left: 'center',
},
series: [
{
name: 'Access From',
type: 'pie',
radius: ['40%', '70%'],
center: ['50%', '60%'],
avoidLabelOverlap: false,
itemStyle: {
borderRadius: 10,
borderColor: '#fff',
borderWidth: 2,
},
label: {
show: false,
position: 'center',
},
emphasis: {
label: {
show: true,
fontSize: '40',
fontWeight: 'bold',
},
},
labelLine: {
show: false,
},
data: [
{ value: 1048, name: 'Search Engine' },
{ value: 735, name: 'Direct' },
{ value: 580, name: 'Email' },
{ value: 484, name: 'Union Ads' },
{ value: 300, name: 'Video Ads' },
],
}, },
], legend: {
} top: '5%',
left: 'center',
},
series: [
{
name: 'Access From',
type: 'pie',
radius: ['40%', '70%'],
center: ['50%', '60%'],
avoidLabelOverlap: false,
itemStyle: {
borderRadius: 10,
borderColor: '#fff',
borderWidth: 2,
},
label: {
show: false,
position: 'center',
},
emphasis: {
label: {
show: true,
fontSize: '40',
fontWeight: 'bold',
},
},
labelLine: {
show: false,
},
data: [
{ value: 1048, name: 'Search Engine' },
{ value: 735, name: 'Direct' },
{ value: 580, name: 'Email' },
{ value: 484, name: 'Union Ads' },
{ value: 300, name: 'Video Ads' },
],
},
],
};
onMounted(() => { onMounted(() => {
setOptions(chartOptions) setOptions(chartOptions);
}) });
</script> </script>
<style scoped></style> <style scoped></style>

Some files were not shown because too many files have changed in this diff Show More