diff --git a/.env b/.env
new file mode 100644
index 0000000..e3e89cf
--- /dev/null
+++ b/.env
@@ -0,0 +1,2 @@
+VITE_TOKEN_KEY=tokenKey
+VITE_URL_PREFIX=/api
\ No newline at end of file
diff --git a/.env.production b/.env.production
new file mode 100644
index 0000000..e3e89cf
--- /dev/null
+++ b/.env.production
@@ -0,0 +1,2 @@
+VITE_TOKEN_KEY=tokenKey
+VITE_URL_PREFIX=/api
\ No newline at end of file
diff --git a/.env.test b/.env.test
new file mode 100644
index 0000000..e3e89cf
--- /dev/null
+++ b/.env.test
@@ -0,0 +1,2 @@
+VITE_TOKEN_KEY=tokenKey
+VITE_URL_PREFIX=/api
\ No newline at end of file
diff --git a/.eslintignore b/.eslintignore
new file mode 100644
index 0000000..348631b
--- /dev/null
+++ b/.eslintignore
@@ -0,0 +1,15 @@
+
+*.sh
+node_modules
+*.md
+*.woff
+*.ttf
+.vscode
+.idea
+dist
+/public
+/docs
+.husky
+.local
+/bin
+Dockerfile
diff --git a/.eslintrc.js b/.eslintrc.js
new file mode 100644
index 0000000..f7b53bf
--- /dev/null
+++ b/.eslintrc.js
@@ -0,0 +1,72 @@
+module.exports = {
+ 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', '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/require-explicit-emits': 'off',
+ 'vue/html-self-closing': [
+ 'error',
+ {
+ html: {
+ void: 'always',
+ normal: 'never',
+ component: 'always',
+ },
+ svg: 'always',
+ math: 'always',
+ },
+ ],
+ 'vue/multi-word-component-names': 'off',
+ },
+};
diff --git a/.gitignore b/.gitignore
index 53f7466..93d606e 100644
--- a/.gitignore
+++ b/.gitignore
@@ -1,5 +1,24 @@
+# Logs
+logs
+*.log
+npm-debug.log*
+yarn-debug.log*
+yarn-error.log*
+pnpm-debug.log*
+lerna-debug.log*
+
node_modules
-.DS_Store
dist
dist-ssr
-*.local
\ No newline at end of file
+*.local
+.eslintcache
+
+# Editor directories and files
+!.vscode/extensions.json
+.idea
+.DS_Store
+*.suo
+*.ntvs*
+*.njsproj
+*.sln
+*.sw?
diff --git a/.husky/pre-commit b/.husky/pre-commit
new file mode 100755
index 0000000..d506cff
--- /dev/null
+++ b/.husky/pre-commit
@@ -0,0 +1,7 @@
+#!/bin/sh
+. "$(dirname "$0")/_/husky.sh"
+
+[ -n "$CI" ] && exit 0
+
+# Format and submit code according to lintstagedrc.js configuration
+npm run lint:lint-staged
diff --git a/.prettierignore b/.prettierignore
new file mode 100644
index 0000000..f7e39e6
--- /dev/null
+++ b/.prettierignore
@@ -0,0 +1,9 @@
+/dist/*
+.local
+.output.js
+/node_modules/**
+
+**/*.svg
+**/*.sh
+
+/public/*
diff --git a/.stylelintignore b/.stylelintignore
new file mode 100644
index 0000000..0517076
--- /dev/null
+++ b/.stylelintignore
@@ -0,0 +1,3 @@
+/dist/*
+/public/*
+public/*
diff --git a/.vscode/extensions.json b/.vscode/extensions.json
new file mode 100644
index 0000000..3dc5b08
--- /dev/null
+++ b/.vscode/extensions.json
@@ -0,0 +1,3 @@
+{
+ "recommendations": ["johnsoncodehk.volar"]
+}
diff --git a/.vscode/settings.json b/.vscode/settings.json
new file mode 100644
index 0000000..e690e60
--- /dev/null
+++ b/.vscode/settings.json
@@ -0,0 +1,4 @@
+{
+ "i18n-ally.localesPaths": ["src/i18n", "src/i18n/lang"],
+ "cSpell.words": ["consola", "eruda", "mockjs", "nutui", "pinia", "stylelint", "vant", "vite", "vitejs", "vueuse"]
+}
diff --git a/LICENSE b/LICENSE
index 02241f0..486a164 100644
--- a/LICENSE
+++ b/LICENSE
@@ -1,6 +1,6 @@
MIT License
-Copyright (c) 2021 xsf
+Copyright (c) 2020-present, Fast-vue3
Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal
@@ -18,4 +18,4 @@ FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
-SOFTWARE.
+SOFTWARE.
\ No newline at end of file
diff --git a/README.md b/README.md
index 4d8cc8d..6cdf4eb 100644
--- a/README.md
+++ b/README.md
@@ -1,36 +1,421 @@
-# Vue3-Tutorial
-A Tutorial for Vue 3.
+# vue3-h5-template
-- [x] Vue3
-- [x] vue-router
-- [x] Vite
-- [x] setup
-- [x] echarts
+基于 vue3 + vite + nut ui + sass + viewport 适配方案 +axios 封装,构建手机端模板脚手架
-## How to use
-You should clone the repo and install the dependencies, and then npm start.That is all.
+
+### 启动项目
```bash
-$ git clone https://github.com/allan2coder/VUE3-Tutorial.git
-$ cd VUE3-Tutorial
-$ npm install
+npm install
+
+npm run dev
```
-Then launch the project app.
+
+目录
+
+- [√ vite](#)
+- [√ 配置多环境变量](#env)
+- [√ viewport 适配方案](#viewport)
+- [√ nutUI 组件按需加载](#nutUI)
+- [√ Pinia 状态管理](#Pinia)
+- [√ Vue-router4](#router)
+- [√ Axios 封装及接口管理](#axios)
+- [√ vite.config.ts 基础配置](#base)
+- [√ alias](#alias)
+- [√ proxy 跨域](#proxy)
+- [√ Eslint+Pettier+stylelint 统一开发规范 ](#lint)
+
+### ✅ 配置多环境变量
+
+`package.json` 里的 `scripts` 配置 `dev` `dev:test` `dev:prod` ,通过 `--mode xxx` 来执行不同环境
+
+- 通过 `npm run dev` 启动本地环境参数 , 执行 `development`
+- 通过 `npm run dev:test` 启动测试环境参数 , 执行 `test`
+- 通过 `npm run dev:prod` 启动正式环境参数 , 执行 `prod`
+
+```javascript
+"scripts": {
+ "dev": "vite",
+ "dev:test": "vite --mode test",
+ "dev:prod": "vite --mode production",
+}
+```
+
+[▲ 回顶部](#top)
+
+### ✅ viewport 适配方案
+
+不用担心,项目已经配置好了 `viewport` 适配, 下面仅做介绍:
+
+- [postcss-px-to-viewport-8-plugin](https://github.com/xian88888888/postcss-px-to-viewport-8-plugin) 是一款 `postcss` 插件,用于将单位转化为 `vw`, 现在很多浏览器对`vw`的支持都很好。
+
+##### PostCSS 配置
+
+下面提供了一份基本的 `postcss` 配置,可以在此配置的基础上根据项目需求进行修改
+
+```javascript
+// https://github.com/michael-ciniawsky/postcss-load-config
+module.exports = {
+ plugins: {
+ 'postcss-px-to-viewport-8-plugin': {
+ unitToConvert: 'px', // 要转化的单位
+ viewportWidth: 375, // UI设计稿的宽度
+ unitPrecision: 6, // 转换后的精度,即小数点位数
+ propList: ['*'], // 指定转换的css属性的单位,*代表全部css属性的单位都进行转换
+ viewportUnit: 'vw', // 指定需要转换成的视窗单位,默认vw
+ fontViewportUnit: 'vw', // 指定字体需要转换成的视窗单位,默认vw
+ minPixelValue: 1, // 默认值1,小于或等于1px则不进行转换
+ mediaQuery: true, // 是否在媒体查询的css代码中也进行转换,默认false
+ replace: true, // 是否转换后直接更换属性值
+ exclude: [/node_modules/], // 设置忽略文件,用正则做目录名匹配
+ },
+ },
+};
+```
+
+更多详细信息: [vant](https://youzan.github.io/vant/#/zh-CN/quickstart#jin-jie-yong-fa)
+
+**新手必看,老鸟跳过**
+
+很多小伙伴会问我,适配的问题, 因为我们使用的是 Vant UI,所以必须根据 Vant UI 375 的设计规范走,一般我们的设计会将 UI 图上传到蓝湖,我们就可以需要的尺寸了。下面就大搞普及一下 rem。
+
+我们知道 `1rem` 等于 `html` 根元素设定的 `font-size` 的 `px` 值。Vant UI 设置 `rootValue: 37.5` , 你可以看到在 iPhone 6 下看到 ( `1rem 等于 37.5px` ):
+
+```html
+
+```
+
+切换不同的机型,根元素可能会有不同的 `font-size` 。当你写 css px 样式时,会被程序换算成 `rem` 达到适配。
+
+因为我们用了 Vant 的组件,需要按照 `rootValue: 37.5` 来写样式。
+
+举个例子:设计给了你一张 750px \* 1334px 图片,在 iPhone6 上铺满屏幕, 其他机型适配。
+
+- 当`rootValue: 75` , 样式 `width: 750px;height: 1334px;` 图片会撑满 iPhone6 屏幕,这个时候切换其他机型,图片也会跟着撑满。
+- 当`rootValue: 37.5` 的时候,样式 `width: 375px;height: 667px;` 图片会撑满 iPhone6 屏幕。
+
+也就是 iphone 6 下 375px 宽度写 CSS。其他的你就可以根据你设计图,去写对应的样式就可以了。
+
+当然,想要撑满屏幕你可以使用 100%,这里只是举例说明。
+
+```html
+
+
+
+```
+
+[▲ 回顶部](#top)
+
+### ✅ nutUI 组件按需加载
+
+Vite 构建工具,使用 vite-plugin-style-import 实现按需引入。
+
+#### 安装插件
```bash
-$ npm run dev
+npm i vite-plugin-style-import -D
```
-You should see a new browser tap opening and a page of 'index.html' in http://localhost:3000.
+在 `vite.config.ts` 设置
-## How to build the static files
-
-``` bash
-npm run build
+```javascript
+ plugins: [
+ ...
+ createStyleImportPlugin({
+ resolves: [NutuiResolve()],
+ }),
+ ...
+ ],
```
-## Other SPA
-- [React.js](https://github.com/allan2coder/React-SPA) :fire: :fire: :fire:
+#### 使用组件
-## License
-MIT
+项目在 `plugins/nutUI.ts` 下统一管理组件,用哪个引入哪个,无需在页面里重复引用
+
+```javascript
+// 按需全局引入nutUI组件
+import Vue from 'vue';
+import { Button, Cell, CellGroup } from '@nutui/nutui';
+export const nutUiComponents = [Button, Cell, CellGroup];
+
+// 在main.ts文件中引入
+nutUiComponents.forEach((item) => {
+ app.use(item);
+});
+```
+
+[▲ 回顶部](#top)
+
+### ✅ Pinia 状态管理
+
+下一代 vuex,使用极其方便,ts 兼容好
+
+目录结构
+
+```bash
+├── store
+│ ├── modules
+│ │ └── user.js
+│ ├── index.js
+```
+
+使用
+
+```html
+
+```
+
+[▲ 回顶部](#top)
+
+### ✅ Vue-router
+
+本案例采用 `hash` 模式,开发者根据需求修改 `mode` `base`
+
+**注意**:如果你使用了 `history` 模式, `vue.config.js` 中的 `publicPath` 要做对应的**修改**
+
+前往:[vue.config.js 基础配置](#base)
+
+```javascript
+import Vue from 'vue';
+import { createRouter, createWebHistory, Router } from 'vue-router';
+
+Vue.use(Router);
+export const router = [
+ {
+ name: 'root',
+ path: '/',
+ redirect: '/home',
+ component: () => import('@/layout/basic/index.vue'),
+ },
+];
+
+const router: Router = createRouter({
+ history: createWebHistory(),
+ routes: routes,
+});
+
+export default router;
+```
+
+更多:[Vue Router](https://router.vuejs.org/zh/introduction.html)
+
+[▲ 回顶部](#top)
+
+### ✅ Axios 封装及接口管理
+
+`utils/request.js` 封装 axios , 开发者需要根据后台接口做修改。
+
+- `service.interceptors.request.use` 里可以设置请求头,比如设置 `token`
+- `config.hideloading` 是在 api 文件夹下的接口参数里设置,下文会讲
+- `service.interceptors.response.use` 里可以对接口返回数据处理,比如 401 删除本地信息,重新登录
+
+```javascript
+import axios from 'axios';
+import store from '@/store';
+import { Toast } from 'vant';
+// 根据环境不同引入不同api地址
+import { baseApi } from '@/config';
+// create an axios instance
+const service = axios.create({
+ baseURL: baseApi, // url = base api url + request url
+ withCredentials: true, // send cookies when cross-domain requests
+ timeout: 5000, // request timeout
+});
+
+// request 拦截器 request interceptor
+service.interceptors.request.use(
+ (config) => {
+ // 不传递默认开启loading
+ if (!config.hideloading) {
+ // loading
+ Toast.loading({
+ forbidClick: true,
+ });
+ }
+ if (store.getters.token) {
+ config.headers['X-Token'] = '';
+ }
+ return config;
+ },
+ (error) => {
+ // do something with request error
+ console.log(error); // for debug
+ return Promise.reject(error);
+ },
+);
+// respone拦截器
+service.interceptors.response.use(
+ (response) => {
+ Toast.clear();
+ const res = response.data;
+ if (res.status && res.status !== 200) {
+ // 登录超时,重新登录
+ if (res.status === 401) {
+ store.dispatch('FedLogOut').then(() => {
+ location.reload();
+ });
+ }
+ return Promise.reject(res || 'error');
+ } else {
+ return Promise.resolve(res);
+ }
+ },
+ (error) => {
+ Toast.clear();
+ console.log('err' + error); // for debug
+ return Promise.reject(error);
+ },
+);
+export default service;
+```
+
+#### 接口管理
+
+在 `src/api` 文件夹下统一管理接口
+
+- 你可以建立多个模块对接接口, 比如 `home.js` 里是首页的接口这里讲解 `user.js`
+- `url` 接口地址,请求的时候会拼接上 `config` 下的 `baseApi`
+- `method` 请求方法
+- `data` 请求参数 `qs.stringify(params)` 是对数据系列化操作
+- `hideloading` 默认 `false`, 设置为 `true` 后,不显示 loading ui 交互中有些接口不需要让用户感知
+
+```javascript
+import qs from 'qs';
+// axios
+import request from '@/utils/request';
+//user api
+
+// 用户信息
+export function getUserInfo(params) {
+ return request({
+ url: '/user/userinfo',
+ method: 'post',
+ data: qs.stringify(params),
+ hideloading: true, // 隐藏 loading 组件
+ });
+}
+```
+
+#### 如何调用
+
+```javascript
+// 请求接口
+import { getUserInfo } from '@/api/user.js';
+
+const params = {
+ user: 'sunnie',
+};
+getUserInfo(params)
+ .then(() => {})
+ .catch(() => {});
+```
+
+[▲ 回顶部](#top)
+
+### ✅ vite.config.ts 基础配置
+
+如果你的 `Vue Router` 模式是 hash
+
+```javascript
+publicPath: './',
+```
+
+如果你的 `Vue Router` 模式是 history 这里的 publicPath 和你的 `Vue Router` `base` **保持一直**
+
+```javascript
+publicPath: '/app/',
+```
+
+```javascript
+export default function ({ command }: ConfigEnv): UserConfigExport {
+ const isProduction = command === 'build';
+ return {
+ server: {
+ host: '0.0.0.0',
+ },
+ plugins: [
+ vue(),
+ vueJsx(),
+ createStyleImportPlugin({
+ resolves: [NutuiResolve()],
+ }),
+ eruda(),
+ viteMockServe({
+ mockPath: './src/mock',
+ localEnabled: command === 'serve',
+ logger: true,
+ }),
+ ],
+ css: {
+ preprocessorOptions: {
+ scss: {
+ // 配置 nutui 全局 scss 变量
+ additionalData: `@import "@nutui/nutui/dist/styles/variables.scss";`,
+ },
+ },
+ },
+ };
+}
+```
+
+[▲ 回顶部](#top)
+
+### ✅ 配置 alias 别名
+
+```javascript
+resolve: {
+ alias: [{
+ find: 'vue-i18n',
+ replacement: 'vue-i18n/dist/vue-i18n.cjs.js',
+ },
+ // /@/xxxx => src/xxxx
+ {
+ find: /\/@\//,
+ replacement: pathResolve('src') + '/',
+ },
+ // /#/xxxx => types/xxxx
+ {
+ find: /\/#\//,
+ replacement: pathResolve('types') + '/',
+ },
+ ],
+},
+```
+
+[▲ 回顶部](#top)
+
+### ✅ 配置 proxy 跨域
+
+```javascript
+server: {
+ proxy: {
+ '/api': {
+ target: 'https://baidu.com',
+ changeOrigin: true,
+ rewrite: (path) => path.replace(/^\/api/, '')
+ }
+ }
+},
+```
+
+[▲ 回顶部](#top)
+
+### ✅ Eslint+Pettier+stylelint 统一开发规范
+
+根目录下的`.eslintrc.js`、`.stylelint.config.js`、`.prettier.config.js`内置了 lint 规则,帮助你规范地开发代码,有助于提高团队的代码质量和协作性,可以根据团队的规则进行修改
diff --git a/config/constant.ts b/config/constant.ts
new file mode 100644
index 0000000..f47e28e
--- /dev/null
+++ b/config/constant.ts
@@ -0,0 +1 @@
+export const IsReport = process.env.REPORT;
diff --git a/config/vite/plugins/autoImport.ts b/config/vite/plugins/autoImport.ts
new file mode 100644
index 0000000..c55553e
--- /dev/null
+++ b/config/vite/plugins/autoImport.ts
@@ -0,0 +1,21 @@
+/**
+ * @name AutoImportDeps
+ * @description 按需加载,自动引入
+ */
+import AutoImport from 'unplugin-auto-import/vite';
+import { ElementPlusResolver } from 'unplugin-vue-components/resolvers';
+
+export const AutoImportDeps = () => {
+ return AutoImport({
+ dts: 'types/auto-imports.d.ts',
+ imports: [
+ 'vue',
+ 'pinia',
+ 'vue-router',
+ {
+ '@vueuse/core': [],
+ },
+ ],
+ resolvers: [ElementPlusResolver()],
+ });
+};
diff --git a/config/vite/plugins/component.ts b/config/vite/plugins/component.ts
new file mode 100644
index 0000000..c8dbf46
--- /dev/null
+++ b/config/vite/plugins/component.ts
@@ -0,0 +1,20 @@
+/**
+ * @name AutoRegistryComponents
+ * @description 按需加载,自动引入组件
+ */
+import Components from 'unplugin-vue-components/vite';
+import { VueUseComponentsResolver } from 'unplugin-vue-components/resolvers';
+export const AutoRegistryComponents = () => {
+ return Components({
+ // dirs: ['src/components'],
+ extensions: ['vue', 'md'],
+ deep: true,
+ dts: 'types/components.d.ts',
+ directoryAsNamespace: false,
+ globalNamespaces: [],
+ directives: true,
+ include: [/\.vue$/, /\.vue\?vue/, /\.md$/],
+ exclude: [/[\\/]node_modules[\\/]/, /[\\/]\.git[\\/]/, /[\\/]\.nuxt[\\/]/],
+ resolvers: [VueUseComponentsResolver()],
+ });
+};
diff --git a/config/vite/plugins/compress.ts b/config/vite/plugins/compress.ts
new file mode 100644
index 0000000..2914287
--- /dev/null
+++ b/config/vite/plugins/compress.ts
@@ -0,0 +1,18 @@
+/**
+ * @name ConfigCompressPlugin
+ * @description 开启.gz压缩
+ */
+import viteCompression from 'vite-plugin-compression';
+
+export const ConfigCompressPlugin = () => {
+ return viteCompression({
+ verbose: true, // 默认即可
+ disable: false, //开启压缩(不禁用),默认即可
+ deleteOriginFile: false, //删除源文件
+ threshold: 10240, //压缩前最小文件大小
+ algorithm: 'gzip', //压缩算法
+ ext: '.gz', //文件类型
+ });
+
+ return [];
+};
diff --git a/config/vite/plugins/eruda.ts b/config/vite/plugins/eruda.ts
new file mode 100644
index 0000000..112294b
--- /dev/null
+++ b/config/vite/plugins/eruda.ts
@@ -0,0 +1,5 @@
+import eruda from 'vite-plugin-eruda';
+
+export const ConfigEruda = () => {
+ return eruda();
+};
diff --git a/config/vite/plugins/imagemin.ts b/config/vite/plugins/imagemin.ts
new file mode 100644
index 0000000..d8ad39c
--- /dev/null
+++ b/config/vite/plugins/imagemin.ts
@@ -0,0 +1,32 @@
+import viteImagemin from 'vite-plugin-imagemin';
+
+export function ConfigImageminPlugin() {
+ const plugin = viteImagemin({
+ gifsicle: {
+ optimizationLevel: 7,
+ interlaced: false,
+ },
+ mozjpeg: {
+ quality: 20,
+ },
+ optipng: {
+ optimizationLevel: 7,
+ },
+ pngquant: {
+ quality: [0.8, 0.9],
+ speed: 4,
+ },
+ svgo: {
+ plugins: [
+ {
+ name: 'removeViewBox',
+ },
+ {
+ name: 'removeEmptyAttrs',
+ active: false,
+ },
+ ],
+ },
+ });
+ return plugin;
+}
diff --git a/config/vite/plugins/index.ts b/config/vite/plugins/index.ts
new file mode 100644
index 0000000..6d51bd3
--- /dev/null
+++ b/config/vite/plugins/index.ts
@@ -0,0 +1,69 @@
+/**
+ * @name createVitePlugins
+ * @description 封装plugins数组统一调用
+ */
+import type { Plugin } from 'vite';
+import vue from '@vitejs/plugin-vue';
+import vueJsx from '@vitejs/plugin-vue-jsx';
+import vueSetupExtend from 'vite-plugin-vue-setup-extend';
+import { ConfigSvgIconsPlugin } from './svgIcons';
+import { AutoRegistryComponents } from './component';
+import { AutoImportDeps } from './autoImport';
+import { ConfigMockPlugin } from './mock';
+import { ConfigCompressPlugin } from './compress';
+import { ConfigPagesPlugin } from './pages';
+import { ConfigRestartPlugin } from './restart';
+import { ConfigProgressPlugin } from './progress';
+import { ConfigEruda } from './eruda';
+import { ConfigStyleImport } from './styleImport';
+import { ConfigImageminPlugin } from './imagemin';
+import { ConfigVisualizerConfig } from './visualizer';
+
+export function createVitePlugins(isBuild: boolean) {
+ const vitePlugins: (Plugin | Plugin[])[] = [
+ // vue支持
+ vue(),
+ // JSX支持
+ vueJsx(),
+ // setup语法糖组件名支持
+ vueSetupExtend(),
+ ];
+
+ // 自动按需引入组件
+ vitePlugins.push(AutoRegistryComponents());
+
+ // 自动按需引入依赖
+ vitePlugins.push(AutoImportDeps());
+
+ // 自动生成路由
+ vitePlugins.push(ConfigPagesPlugin());
+
+ // 开启.gz压缩 rollup-plugin-gzip
+ vitePlugins.push(ConfigCompressPlugin());
+
+ // 监听配置文件改动重启
+ vitePlugins.push(ConfigRestartPlugin());
+
+ // 构建时显示进度条
+ vitePlugins.push(ConfigProgressPlugin());
+
+ //styleImport
+ vitePlugins.push(ConfigStyleImport());
+
+ // eruda
+ vitePlugins.push(ConfigEruda());
+
+ // rollup-plugin-visualizer
+ vitePlugins.push(ConfigVisualizerConfig());
+ if (isBuild) {
+ // vite-plugin-imagemin
+ vitePlugins.push(ConfigImageminPlugin());
+
+ // vite-plugin-svg-icons
+ vitePlugins.push(ConfigSvgIconsPlugin(isBuild));
+
+ // vite-plugin-mock
+ vitePlugins.push(ConfigMockPlugin(isBuild));
+ }
+ return vitePlugins;
+}
diff --git a/config/vite/plugins/mock.ts b/config/vite/plugins/mock.ts
new file mode 100644
index 0000000..a768bc3
--- /dev/null
+++ b/config/vite/plugins/mock.ts
@@ -0,0 +1,18 @@
+/**
+ * @name ConfigMockPlugin
+ * @description 引入mockjs,本地模拟接口
+ */
+import { viteMockServe } from 'vite-plugin-mock';
+export const ConfigMockPlugin = (isBuild: boolean) => {
+ return viteMockServe({
+ ignore: /^\_/,
+ mockPath: 'mock',
+ localEnabled: !isBuild,
+ prodEnabled: false, //实际开发请关闭,会影响打包体积
+ // https://github.com/anncwb/vite-plugin-mock/issues/9
+ injectCode: `
+ import { setupProdMockServer } from '../mock/_createProdMockServer';
+ setupProdMockServer();
+ `,
+ });
+};
diff --git a/config/vite/plugins/pages.ts b/config/vite/plugins/pages.ts
new file mode 100644
index 0000000..32ec068
--- /dev/null
+++ b/config/vite/plugins/pages.ts
@@ -0,0 +1,13 @@
+/**
+ * @name ConfigPagesPlugin
+ * @description 动态生成路由
+ */
+import Pages from 'vite-plugin-pages';
+export const ConfigPagesPlugin = () => {
+ return Pages({
+ pagesDir: [{ dir: 'src/pages', baseRoute: '' }],
+ extensions: ['vue', 'md'],
+ exclude: ['**/components/*.vue'],
+ nuxtStyle: true,
+ });
+};
diff --git a/config/vite/plugins/progress.ts b/config/vite/plugins/progress.ts
new file mode 100644
index 0000000..8dc2cc3
--- /dev/null
+++ b/config/vite/plugins/progress.ts
@@ -0,0 +1,9 @@
+/**
+ * @name ConfigProgressPlugin
+ * @description 构建显示进度条
+ */
+
+import progress from 'vite-plugin-progress';
+export const ConfigProgressPlugin = () => {
+ return progress() as Plugin;
+};
diff --git a/config/vite/plugins/restart.ts b/config/vite/plugins/restart.ts
new file mode 100644
index 0000000..37ea17f
--- /dev/null
+++ b/config/vite/plugins/restart.ts
@@ -0,0 +1,10 @@
+/**
+ * @name ConfigRestartPlugin
+ * @description 监听配置文件修改自动重启Vite
+ */
+import ViteRestart from 'vite-plugin-restart';
+export const ConfigRestartPlugin = () => {
+ return ViteRestart({
+ restart: ['*.config.[jt]s', '**/config/*.[jt]s'],
+ });
+};
diff --git a/config/vite/plugins/styleImport.ts b/config/vite/plugins/styleImport.ts
new file mode 100644
index 0000000..143978e
--- /dev/null
+++ b/config/vite/plugins/styleImport.ts
@@ -0,0 +1,7 @@
+import { createStyleImportPlugin, NutuiResolve, VantResolve } from 'vite-plugin-style-import';
+
+export const ConfigStyleImport = () => {
+ return createStyleImportPlugin({
+ resolves: [NutuiResolve(), VantResolve()],
+ });
+};
diff --git a/config/vite/plugins/svgIcons.ts b/config/vite/plugins/svgIcons.ts
new file mode 100644
index 0000000..0e92c0a
--- /dev/null
+++ b/config/vite/plugins/svgIcons.ts
@@ -0,0 +1,16 @@
+/**
+ * @name SvgIconsPlugin
+ * @description 加载SVG文件,自动引入
+ */
+import { createSvgIconsPlugin } from 'vite-plugin-svg-icons';
+import path from 'path';
+
+export const ConfigSvgIconsPlugin = (isBuild: boolean) => {
+ return createSvgIconsPlugin({
+ // 指定需要缓存的图标文件夹
+ iconDirs: [path.resolve(process.cwd(), 'src/assets/icons')],
+ // 指定symbolId格式
+ symbolId: 'icon-[dir]-[name]',
+ svgoOptions: isBuild,
+ });
+};
diff --git a/config/vite/plugins/visualizer.ts b/config/vite/plugins/visualizer.ts
new file mode 100644
index 0000000..cbf8a22
--- /dev/null
+++ b/config/vite/plugins/visualizer.ts
@@ -0,0 +1,14 @@
+import visualizer from 'rollup-plugin-visualizer';
+import { IsReport } from '../../constant';
+
+export function ConfigVisualizerConfig() {
+ if (IsReport) {
+ return visualizer({
+ filename: './node_modules/.cache/visualizer/stats.html',
+ open: true,
+ gzipSize: true,
+ brotliSize: true,
+ }) as Plugin;
+ }
+ return [];
+}
diff --git a/config/vite/proxy.ts b/config/vite/proxy.ts
new file mode 100644
index 0000000..f865ff9
--- /dev/null
+++ b/config/vite/proxy.ts
@@ -0,0 +1,20 @@
+import { API_BASE_URL, API_TARGET_URL, MOCK_API_BASE_URL, MOCK_API_TARGET_URL } from '../../config/constant';
+import { ProxyOptions } from 'vite';
+type ProxyTargetList = Record;
+
+const init: ProxyTargetList = {
+ // test
+ [API_BASE_URL]: {
+ target: API_TARGET_URL,
+ changeOrigin: true,
+ rewrite: (path) => path.replace(new RegExp(`^${API_BASE_URL}`), ''),
+ },
+ // mock
+ [MOCK_API_BASE_URL]: {
+ target: MOCK_API_TARGET_URL,
+ changeOrigin: true,
+ rewrite: (path) => path.replace(new RegExp(`^${MOCK_API_BASE_URL}`), '/api'),
+ },
+};
+
+export default init;
diff --git a/index.html b/index.html
index 030a6ff..77290de 100644
--- a/index.html
+++ b/index.html
@@ -3,11 +3,40 @@
-
+
Vite App
-
+
+