mirror of
https://github.com/sunniejs/vue-h5-template.git
synced 2025-04-05 19:42:08 +08:00
docs: 完善部分文档
This commit is contained in:
parent
d23e5bacb1
commit
ceff802428
2
.env.production
Normal file
2
.env.production
Normal file
@ -0,0 +1,2 @@
|
||||
VITE_TOKEN_KEY=tokenKey
|
||||
VITE_URL_PREFIX=/api
|
439
README.md
439
README.md
@ -1,11 +1,438 @@
|
||||
# Vue 3 + Typescript + Vite
|
||||
# vue-h5-template
|
||||
|
||||
This template should help get you started developing with Vue 3 and Typescript in Vite. The template uses Vue 3 `<script setup>` SFCs, check out the [script setup docs](https://v3.vuejs.org/api/sfc-script-setup.html#sfc-script-setup) to learn more.
|
||||
基于 vue3 + vite + nut ui + sass + viewport 适配方案 +axios 封装,构建手机端模板脚手架
|
||||
|
||||
## Recommended IDE Setup
|
||||
掘金: [移动端适配方案](https://juejin.cn/post/7018433228591595550)
|
||||
|
||||
- [VSCode](https://code.visualstudio.com/) + [Volar](https://marketplace.visualstudio.com/items?itemName=johnsoncodehk.volar)
|
||||
<p>
|
||||
<img src="./public/screen.png" width="320" style="display:inline;">
|
||||
</p>
|
||||
|
||||
## Type Support For `.vue` Imports in TS
|
||||
### Node 版本要求
|
||||
|
||||
Since TypeScript cannot handle type information for `.vue` imports, they are shimmed to be a generic Vue component type by default. In most cases this is fine if you don't really care about component prop types outside of templates. However, if you wish to get actual prop types in `.vue` imports (for example to get props validation when using manual `h(...)` calls), you can enable Volar's `.vue` type support plugin by running `Volar: Switch TS Plugin on/off` from VSCode command palette.
|
||||
推荐你使用 nodejs 14+版本,毕竟技术日新月异。你可以使用 [nvm](https://github.com/nvm-sh/nvm) 或 [nvm-windows](https://github.com/coreybutler/nvm-windows) 在同一台电脑中管理多个 Node 版本。
|
||||
|
||||
本示例 Node.js 14.19.0
|
||||
|
||||
### 启动项目
|
||||
|
||||
```bash
|
||||
|
||||
git clone https://github.com/sunniejs/vue-h5-template.git
|
||||
|
||||
cd vue-h5-template
|
||||
|
||||
npm install
|
||||
|
||||
npm run serve
|
||||
```
|
||||
|
||||
<span id="top">目录</span>
|
||||
|
||||
- √ Vue-cli4
|
||||
- [√ 配置多环境变量](#env)
|
||||
- [√ viewport 适配方案](#viewport)
|
||||
- [√ nutUI 组件按需加载](#nutUI)
|
||||
- [√ Pinia 状态管理](#Pinia)
|
||||
- [√ Vue-router4](#router)
|
||||
- [√ Axios 封装及接口管理](#axios)
|
||||
- [√ vite.config.ts 基础配置](#base)
|
||||
- [√ ts 支持](#alias)
|
||||
- [√ Eslint+Pettier+stylelint 统一开发规范 ](#pettier)
|
||||
|
||||
### <span id="env">✅ 配置多环境变量 </span>
|
||||
|
||||
`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)
|
||||
|
||||
### <span id="viewport">✅ viewport 适配方案 </span>
|
||||
|
||||
不用担心,项目已经配置好了 `viewport` 适配, 下面仅做介绍:
|
||||
|
||||
Vant 中的样式默认使用`px`作为单位,如果需要使用`rem`单位,推荐使用以下两个工具:
|
||||
|
||||
- [postcss-px-to-viewport-8-plugin](https://github.com/xian88888888/postcss-px-to-viewport-8-plugin) 是一款 `postcss` 插件,用于将单位转化为 `vw`
|
||||
|
||||
##### PostCSS 配置
|
||||
|
||||
下面提供了一份基本的 `postcss` 配置,可以在此配置的基础上根据项目需求进行修改
|
||||
|
||||
```javascript
|
||||
// https://github.com/michael-ciniawsky/postcss-load-config
|
||||
module.exports = {
|
||||
plugins: {
|
||||
autoprefixer: {
|
||||
overrideBrowserslist: ['Android 4.1', 'iOS 7.1', 'Chrome > 31', 'ff > 31', 'ie >= 8'],
|
||||
},
|
||||
'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
|
||||
<html data-dpr="1" style="font-size: 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
|
||||
<img class="image" src="https://www.sunniejs.cn/static/weapp/logo.png" />
|
||||
|
||||
<style>
|
||||
/* rootValue: 75 */
|
||||
.image {
|
||||
width: 750px;
|
||||
height: 1334px;
|
||||
}
|
||||
/* rootValue: 37.5 */
|
||||
.image {
|
||||
width: 375px;
|
||||
height: 667px;
|
||||
}
|
||||
</style>
|
||||
```
|
||||
|
||||
[▲ 回顶部](#top)
|
||||
|
||||
### <span id="nutUI">✅ nutUI 组件按需加载 </span>
|
||||
|
||||
Vite 构建工具,使用 vite-plugin-style-import 实现按需引入。
|
||||
|
||||
#### 安装插件
|
||||
|
||||
```bash
|
||||
npm i vite-plugin-style-import -D
|
||||
```
|
||||
|
||||
在`vite.config.ts` 设置
|
||||
|
||||
```javascript
|
||||
plugins: [
|
||||
...
|
||||
createStyleImportPlugin({
|
||||
resolves: [NutuiResolve()],
|
||||
}),
|
||||
...
|
||||
],
|
||||
```
|
||||
|
||||
#### 使用组件
|
||||
|
||||
项目在 `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)
|
||||
|
||||
### <span id="Pinia">✅ Pinia 状态管理</span>
|
||||
|
||||
下一代 vuex,使用极其方便,ts 兼容好
|
||||
|
||||
目录结构
|
||||
|
||||
```bash
|
||||
├── store
|
||||
│ ├── modules
|
||||
│ │ └── user.js
|
||||
│ ├── index.js
|
||||
```
|
||||
|
||||
使用
|
||||
|
||||
```html
|
||||
<script lang="ts" setup>
|
||||
import { useUserStore } from '@/store/modules/user';
|
||||
const userStore = useUserStore();
|
||||
userStore.login();
|
||||
</script>
|
||||
```
|
||||
|
||||
[▲ 回顶部](#top)
|
||||
|
||||
### <span id="router">✅ Vue-router </span>
|
||||
|
||||
本案例采用 `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('@/components/Basic/index.vue'),
|
||||
},
|
||||
];
|
||||
|
||||
const router: Router = createRouter({
|
||||
history: createWebHistory(),
|
||||
routes: routes,
|
||||
});
|
||||
|
||||
export default router;
|
||||
```
|
||||
|
||||
更多:[Vue Router](https://router.vuejs.org/zh/introduction.html)
|
||||
|
||||
[▲ 回顶部](#top)
|
||||
|
||||
### <span id="axios">✅ Axios 封装及接口管理</span>
|
||||
|
||||
`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)
|
||||
|
||||
### <span id="base">✅ vite.config.ts 基础配置 </span>
|
||||
|
||||
如果你的 `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)
|
||||
|
||||
### <span id="alias">✅ 配置 alias 别名 </span>
|
||||
|
||||
```javascript
|
||||
resolve: {
|
||||
alias: {
|
||||
'@': pathResolve('./src'),
|
||||
},
|
||||
},
|
||||
```
|
||||
|
||||
[▲ 回顶部](#top)
|
||||
|
||||
### <span id="proxy">✅ 配置 proxy 跨域 </span>
|
||||
|
||||
```javascript
|
||||
server: {
|
||||
proxy: {
|
||||
'/api': {
|
||||
target: 'https://baidu.com',
|
||||
changeOrigin: true,
|
||||
rewrite: (path) => path.replace(/^\/api/, '')
|
||||
}
|
||||
}
|
||||
},
|
||||
```
|
||||
|
||||
[▲ 回顶部](#top)
|
||||
|
||||
<h3>文档在拼命完善中。。。</h3>
|
||||
|
||||
# 关于我
|
||||
|
||||
<p>
|
||||
<img src="./public/account.jpg" width="256" style="display:inline;">
|
||||
</p>
|
||||
|
||||
扫描添加下方的微信并备注加交流群,交流学习,及时获取代码最新动态。
|
||||
|
||||
<p>
|
||||
<img src="./public/group.png" width="256" style="display:inline;">
|
||||
</p>
|
||||
|
||||
如果对你有帮助送我一颗珍贵的小星星(づ ̄3 ̄)づ╭❤~
|
||||
|
@ -3,6 +3,8 @@
|
||||
"version": "1.0.0",
|
||||
"scripts": {
|
||||
"dev": "vite",
|
||||
"dev:test": "vite --mode test",
|
||||
"dev:prod": "vite --mode production",
|
||||
"build": "vue-tsc --noEmit && vite build",
|
||||
"preview": "vite preview",
|
||||
"lint:eslint": "eslint --cache --max-warnings 0 \"{src,mock}/**/*.{vue,ts,tsx}\" --fix",
|
||||
|
BIN
public/account.jpg
Normal file
BIN
public/account.jpg
Normal file
Binary file not shown.
After Width: | Height: | Size: 39 KiB |
BIN
public/group.png
Normal file
BIN
public/group.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 48 KiB |
BIN
public/screen.png
Normal file
BIN
public/screen.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 19 KiB |
@ -1,12 +1,9 @@
|
||||
<template>
|
||||
<nut-navbar :left-show="false" :title="$t('tabbar.home')" />
|
||||
<header class="header">
|
||||
<img width="30" src="https://v3.cn.vuejs.org/logo.png" />
|
||||
<h4 class="intro-title"> Fast-Vue3 </h4>
|
||||
</header>
|
||||
<header class="header"> <img src="https://www.sunniejs.cn/static/weapp/logo.png" /><span> VUE H5开发模板</span> </header>
|
||||
<p class="intro-header">
|
||||
{{ $t('introduction') }}
|
||||
<a href="https://github.com/study-vue3/fast-vue3">
|
||||
<a href="https://github.com/sunniejs/vue-h5-template.git">
|
||||
<nut-icon class="github-icon" name="github" />
|
||||
</a>
|
||||
</p>
|
||||
|
@ -1,4 +1,3 @@
|
||||
import legacy from '@vitejs/plugin-legacy';
|
||||
import vue from '@vitejs/plugin-vue';
|
||||
import vueJsx from '@vitejs/plugin-vue-jsx';
|
||||
import { resolve } from 'path';
|
||||
@ -23,9 +22,6 @@ export default function ({ command }: ConfigEnv): UserConfigExport {
|
||||
createStyleImportPlugin({
|
||||
resolves: [NutuiResolve()],
|
||||
}),
|
||||
legacy({
|
||||
targets: ['defaults', 'not IE 11'],
|
||||
}),
|
||||
eruda(),
|
||||
viteMockServe({
|
||||
mockPath: './src/mock',
|
||||
|
Loading…
x
Reference in New Issue
Block a user