因卓诶-原文链接
vue3-vite2-ts-template
托管地址: github-vue3-vite2-ts-template
命令
启动/打包 命令
技术栈:
命令行
通过安装Tool,来可视化地使用模板,因为仓库中的模板大多数都不会全部用到,你可以通过tool去按需引入它们
npm i enjoy-project-tool -g
创建模板
当然,作为模板的伴生工具,我还会继续维护并且持续提出新的feature来减轻我们开发负担
Tool是使用TS开发的,如果你感兴趣可以提pr,这是Tool的仓库
类型文档/组件文档
文档待补充,暂定使用
代码提交
旧版本的husky和新版还是有很多不一样的,所以如果你以前用过husky那么你要在代码提交这里做更多逻辑的话,可以去看看最新的文档。
模板中只拦截了pre_commit这个钩子,目标就是在pre_commit的时候对代码进行lint和自动修复以及美化,而且仅要对暂存区的文件lint,所以使用了lint-staged。这个组合太常见了,有需求的开发者可以再这个上层定义一些有趣的功能提pr。
还有一个需求是校验git commit message的规范,但是对于小团队来讲,校验这个规范没有太大必要,也暂时不会对团队带来好处,所以爱鼓捣的可以去鼓捣哈。
可以推荐团队成员使用 git-commit-plugin-vscode
vscode 开发小指南
推荐使用 Volar 插件进行开发,如果你的 IDE 是 Jetbrains 系列的,那么你可能不太需要这个插件,如果你是 vscode 推荐使用 volar。使用 volar,不仅可以在 vue 开发上和jetbrains 的表现一致,还可以得到更完善 vue3 的支持,甚至非常新/在草案的语法糖都能够快速享受到。
下载volar地址
此模板对于vscode有天然的支持,如果你使用vscode,就能使用模板自带的vscode配置,比如说保存自动lint&fix&prettier或者其他有意思的功能。
模板中自带了若干个vscode的code-snippets,snippets将会持续更新,它和模板深度贴合,可以帮助你摆脱繁琐的开发。下面就一一描述几个snippets的作用:
初始化@types/model/api的提示工具,自动声明命名空间以及导出
初始化model下的api类,自动引入与之匹配的type类型声明文件以及其他可能用到的依赖
初始化model下的cache类,自动引入与之匹配的type类型声明文件以及其他可能用到的依赖
初始化控制器类
初始化vue页面/组件
AntdV 开发小指南
传统的 antdv 的按需加载,都会使用 babel-plugin-import 这个插件进行按需分析然后自动引入,但是 antdv 中有很多嵌套的父子组件:
由于内部设计原因,无法使用这个插件进行按需导入。最主要的是我们已经使用了vite,本身就带有按需导入,我们只需要处理他们的css的按需引入即可。所以使用了2个插件:
第一个插件主要帮助我们自动识别模板中用到的组件,实现自动引入,也就是说我们使用antdv这样的组件库的时候,不需要全量引入,甚至不需要手动的import就可以自动实现按需引入,如图:
而且脚手架内置了按需引入css的逻辑,所以antdv本身的设计原因导致引入css问题开发者也不需要担心。第二个插件主要是辅助第一个插件做按需引入css逻辑的。第一个插件做的按需引入css有些许问题,比如说antdv里面有很多api调用的组件,比如message,通过message方法调用一个组件,这个时候css不生效,就需要使用第二个插件进行处理。
对于message这样的api组件的css不生效的原因很简单,第一个插件仅仅是解析template用到的组件然后自动引入css,但是无法处理import进来的api组件,所以需要第二个插件做处理。
开发指南
这一块根据自身团队成员的习惯会逐步调整,所以这里的介绍会经常更改。
这套微不足道的架构足以应对中小APP,也是非常简单的,主要就是mvc+ts风格。如果你阅读完整个模板文档之后,你会发现很多东西都做了模块化,把业务划分开了,这也是目前团队开发没有注意到的一点,自身开发完爽是爽了,另外一个人维护就要惨了。各种配置,api都找不到,组件/组件参数也找不到,可能为了快速开发,都会去复制老项目和其他页面的代码;这虽然也是一种“复用”,但是总归来说并不是标准的。所以只有将业务划分开,才能快速定位具体核心代码,才能快速复用。
类型
src/@types
像大部分工程一样,把能抽离的type都尽量都抽离到了@types这一层,这一层也暂时根据需求划分了以下几个内容:
里面最重度使用的应该是model,我们在model模型中根据业务定义了很多ts,比如user.ts:
namespace TUserApiModel { type ReqLogin = { captcha: string; password: string; username: string; uuid: string; }; type ResLogin = Promise< ActionResult<{ token: string; }> >;}export default TUserApiModel;
这两个就代表了model里面api层(后面会详细说明model里面的api),使用Req和Res作为前缀也就是请求和响应的类型,那么我们定义好之后,在整个工程中我就可以这样使用类型:
那么同理,types文件夹中像store,hook这样的,也是根据业务划分,去定义类型的,这里就不再过多阐述了。
模型
src/model
目前model分为2个含义:
前端大部分的数据来源都包含到了,api模型定义了不同业务的api方法,比如user.ts:
import useRequest from '../../hook/useRequest';export default class UserApiModel { async login(params: TUserModel.ReqLogin): TUserModel.ResLogin { return await useRequest({ url: `${params}`, method: 'get', options: { authApi: true } }); }}
useRequest是我们自定义实现的hook函数,我们通过这个hook可以发起请求,那么你可以看到在这个类中定义了login这个方法,入参类型就是TUserModel.ReqLogin, 返回类型就是TUserModel.ResLogin,这个类型都是我们在@types定义的。
再比如说我们搭配kurimudb做了缓存的模块化,最常用的缓存插件也预装好了,我们可以在model里面去写这样一段代码:
/model/cache/user.ts
import { Models } from 'kurimudb';import { LocalStorageDriver } from 'kurimudb-driver-localstorage';import { CookieDriver } from 'kurimudb-driver-cookie';export class UserLocalStorage extends Models.keyValue { constructor() { super({ name: 'user', driver: LocalStorageDriver }); }}export class UserCookie extends Models.keyValue { constructor() { super({ name: 'user', driver: CookieDriver }); }}
我们在这里定义了2个kurimudb类,一个是localstorage一个是cookie,我们可以在这里新增一些方法或者直接导出给controller用,因为即便你不新增方法也可以使用kurimudb内置的函数。
我们拥有kurimudb这样的库可以解决存储模块化的问题,我们不用关心这个缓存的key是否被使用过,只需要设置好唯一的name值,它就能给我们提供一组方便调用的api。另外kurimudb还有sessionstorage和indexDB的插件,如果业务需要可以快速的安装,然后声明一个新的类导出即可使用。
控制器
src/controller
在模板默认自带了一个user.ts例子,我们在上一个model中说明了apiModel和cacheModel,这里的controller就直接引入它们。并且在controller暴露入口。
import UserApiModel from '../model/api/user';import { UserLocalStorage, UserCookie } from '../model/cache/user';export default class UserController { private localStorageModel: UserLocalStorage; private cookieModel: UserCookie; private apiModel: UserApiModel; constructor() { this.apiModel = new UserApiModel(); this.localStorageModel = new UserLocalStorage(); this.cookieModel = new UserCookie(); } async login(req: TUserModel.ReqLogin): TUserModel.ResLogin { return await this.apiModel.login(req); }}
控制器我们还可以对api/cache获取的数据做处理,比如说,后端返回的数据格式前端不便直接展示,我们应该在controller需要做一层转译,比如像这样:
transform(): { text: string; value: string }[] { const data = { '0': '小明', '1': '小红' }; let _arr = []; let key: keyof typeof data; for (key in data) { _arr.push({ text: data[key], value: key }); } return _arr; }
视图(.vue)
以vue来举例,我们如何在视图优雅的调用controller?并且如何使用@types定义的类型来巩固我们的组件?
import TUserApiModel from '../../@types/model/api/user';const login = async (params: TUserModel.ReqLogin) => { await userController.login(params);};// 调用login函数login({ captcha: "", password: "", username: "", uuid: ""})
当调用login函数时候,提供了与ReqLogin不符合的数据结构,是会出现报错的。同理,我们调用cache也是一样,需要在controller把cache封装一层暴露给vue即可。
环境变量
可以根据业务需要,建立业务相关的 env 环境(模式)。 vite-模式文档
以下是根目录默认提供了 3 个环境文件,对应了本地,测试,生产环境
内容示例: 根据业务需要进行配置
VITE_APP_API=VITE_APP_SECRET=
那么同理,如果业务需要额外增加新的自定义环境变量,则需要在 src/vite-env.d.ts 中重新定义类型:
/// interface ImportMetaEnv { VITE_APP_API: string; VITE_APP_SECRET: string; // 新的环境变量的定义写这里}
Mock
使用vite-plugin-mock
来做本地开发的mock,模板暂时没有内置生产环境的mock。
// vite.config.tsviteMockServe({ localEnabled: true //是否开启本地的mock功能}),
定义mock api:
// /mock/user.tsimport { MockMethod } from 'vite-plugin-mock';export default [ { url: '/api/get', method: 'get', response: (res: any) => { return { code: 0, data: { name: 'this is mock name' } }; } }] as MockMethod[];