Files
smart-admin/代码规范/Vue3规范 V2.0.md
2022-12-09 18:52:46 +08:00

705 lines
20 KiB
Markdown
Raw Blame History

This file contains invisible Unicode characters

This file contains invisible Unicode characters that are indistinguishable to humans but may be processed differently by a computer. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

---
title: 2.2、Vue3规范 V2.0
date: 2022-09-07
---
## 阅读须知
<font color="#DC143C">**在阅读《Java规范》之前请一定先阅读上一篇《规范导读 V2.0》!!!**</font> [前往阅读](./basic.md)
## 一、Vue3 基础规范
### 1.1、项目命名
全部采用小写方式, 以中划线分隔。
```javascript
正例`smart-admin`
反例`mall_management-system / mallManagementSystem`
```
### 1.2、目录、文件命名
目录、文件名 均以 小写方式, 以中划线分隔。
```javascript
正例`/head-search/``/shopping-car/``smart-logo.png``role-form.vue`
反例`/headSearch/` `smartLogo.png` `RoleForm.vue`
```
### 1.3、单引号、双引号、分号
- html中、vue的template 中 标签属性 使用 **双引号**
- 所有js中的 字符串 使用 **单引号**
- 所有js中的代码行换行要用 **分号**
## 二、Vue3 组合式 API规范
### 2.1、 使用setup语法糖
- 组件必须使用 `setup` 语法糖
- setup 大法方便简洁
- 全局都要使用setup语法糖
### 2.2、组合式Composition API 规范
组件内必须使用模块化思想,把代码 进行 拆分;
参照 vue3官方文档对于 Composition Api的理解 [更灵活的代码组织](https://cn.vuejs.org/guide/extras/composition-api-faq.html#better-logic-reuse) 组合式Api即 Composition API 解决的是让 相互关联的代码在一起,以更方便的组织代码,故我们的代码格式如下:
<font color="#DC143C">**即:将相关的 变量和代码 写到一起,并使用 行注释 进行分块**</font>
```js
<script setup>
// 各种需要导入
import xxxxx;
import xxxxx;
import xxxxx;
// -------- 定义组件属性和对外暴露的方法、以及抛出的事件 --------
// -------- 表格查询的 变量和方法 --------
// -------- 批量操作的 变量和方法 --------
// -------- 表单的 变量和方法 --------
</script>
```
举例,分成了两个模块,即:
- 模块1显示 、隐藏操作的 变量和方法
- 模块2表单的 变量和方法
<font color="#DC143C">**【以下是正确的例子】**</font>
```js
<script setup>
import { ref, reactive } from 'vue';
import { message } from 'ant-design-vue';
import { SmartLoading } from '/@/components/framework/smart-loading';
import _ from 'lodash';
import { categoryApi } from '/@/api/business/category/category-api';
import { smartSentry } from '/@/lib/smart-sentry';
// emit
const emit = defineEmits('reloadList');
// 组件
const formRef = ref();
// ------------------------------ 显示 、隐藏操作的 变量和方法------------------------------
// 是否展示抽屉
const visible = ref(false);
// 显示
function showModal(categoryType, parentId, rowData) {
Object.assign(form, formDefault);
form.categoryType = categoryType;
form.parentId = parentId;
if (rowData && !_.isEmpty(rowData)) {
Object.assign(form, rowData);
}
visible.value = true;
}
// 隐藏
function onClose() {
Object.assign(form, formDefault);
visible.value = false;
}
// ------------------------------ 表单的 变量和方法 ------------------------------
// 查询表单默认值
const formDefault = {
categoryId: undefined, //分类id
categoryName: '', //分类名字
categoryType: 1, // 分类类型
parentId: undefined, // 父级id
disabledFlag: false, //是否禁用
};
// 查询表单
let form = reactive({ ...formDefault });
// 表单校验规则
const rules = {
categoryName: [{ required: true, message: '请输入分类名称' }],
};
function onSubmit() {
formRef.value
.validate()
.then(async () => {
SmartLoading.show();
try {
if (form.categoryId) {
await categoryApi.updateCategory(form);
} else {
await categoryApi.addCategory(form);
}
message.success(`${form.categoryId ? '修改' : '添加'}成功`);
emit('reloadList', form.parentId);
onClose();
} catch (error) {
smartSentry.captureError(error);
} finally {
SmartLoading.hide();
}
})
.catch((error) => {
console.log('error', error);
message.error('参数验证错误,请仔细填写表单数据!');
});
}
defineExpose({
showModal,
});
</script>
```
<font color="#DC143C">**【错误的例子】**</font>
```js
<script setup>
import { ref, reactive } from 'vue';
import { message } from 'ant-design-vue';
import { SmartLoading } from '/@/components/framework/smart-loading';
import _ from 'lodash';
import { categoryApi } from '/@/api/business/category/category-api';
import { smartSentry } from '/@/lib/smart-sentry';
// ----------------- 定义 所有变量 -----------------
// emit
const emit = defineEmits('reloadList');
// 组件
const formRef = ref();
// 是否展示抽屉
const visible = ref(false);
// 查询表单默认值
const formDefault = {
categoryId: undefined, //分类id
categoryName: '', //分类名字
categoryType: 1, // 分类类型
parentId: undefined, // 父级id
disabledFlag: false, //是否禁用
};
// 查询表单
let form = reactive({ ...formDefault });
// 表单校验规则
const rules = {
categoryName: [{ required: true, message: '请输入分类名称' }],
};
// ----------------- 定义 所有方法 -----------------
// 显示
function showModal(categoryType, parentId, rowData) {
Object.assign(form, formDefault);
form.categoryType = categoryType;
form.parentId = parentId;
if (rowData && !_.isEmpty(rowData)) {
Object.assign(form, rowData);
}
visible.value = true;
}
// 隐藏
function onClose() {
Object.assign(form, formDefault);
visible.value = false;
}
function onSubmit() {
formRef.value
.validate()
.then(async () => {
SmartLoading.show();
try {
if (form.categoryId) {
await categoryApi.updateCategory(form);
} else {
await categoryApi.addCategory(form);
}
message.success(`${form.categoryId ? '修改' : '添加'}成功`);
emit('reloadList', form.parentId);
onClose();
} catch (error) {
smartSentry.captureError(error);
} finally {
SmartLoading.hide();
}
})
.catch((error) => {
console.log('error', error);
message.error('参数验证错误,请仔细填写表单数据!');
});
}
defineExpose({
showModal,
});
</script>
```
<font color="#DC143C">** vue 官方希望是 模块化的组织代码,你看记得官方这个图吗?**</font>
<table>
<tr>
<td><img src="http://img.zhuoluodada.cn/smart-admin-v2/standard/composition-api.png"/></td>
</tr>
</table>
**第一种写法所有变量写到了一起就像vue2的 data区域所有方法写到一起就像vue2中的 methods区域这样下来其实和vue2没什么区别也就语法稍微变了一下那么就与composition api的初衷违背了**,下面引用官方的话
>处理相同逻辑关注点的代码被强制拆分在了不同的选项中,位于文件的不同部分。在一个几百行的大组件中,要读懂代码中的一个逻辑关注点,需要在文件中反复上下滚动,这并不理想。另外,如果我们想要将一个逻辑关注点抽取重构到一个可复用的工具函数中,需要从文件的多个不同部分找到所需的正确片段。
**SmartAdmin中的写法才真正做到了vue3组合式composition api的意图每一块业务的变量与方法写到一起比如上面代码中的`显示与隐藏`的变量和方法放到了一起,`表单`的变量和方法放到了一起**,下面引用官方的话
> 现在与同一个逻辑关注点相关的代码被归为了一组:我们无需再为了一个逻辑关注点在不同的选项块间来回滚动切换。此外,我们现在可以很轻松地将这一组代码移动到一个外部文件中,不再需要为了抽象而重新组织代码,大大降低了重构成本,这在长期维护的大型项目中非常关键。
### 2.3 模板引用变量Ref
对于vue3中的模板引用ref即 ref 是作为一个特殊的 attribute
```html
<input ref="inputRef">
```
我们要求:
- 使用 ref方法参数为空 进行声明变量
- 变量必须以 `Ref`为结尾
- template中的ref 也必须以 `Ref` 为结尾
比如上面的例子,我们声明如下
```js
const inputRef = ref();
```
### 2.4 变量和方法的注释
在使用Composition Api进行代码编写时我们有效的组织了代码但是由于Composition Api变量和方法会写到一起这时候注释就变得很有必要
要求:
- 变量必须都加上注释
- 方法必须加上注释
比如
```js
// 查询 公告 默认值
const queryFormState = {
noticeTypeId: undefined, //分类
keywords: '', //标题、作者、来源
documentNumber: '', //文号
createUserId: undefined, //创建人
deletedFlag: undefined, //删除标识
createTimeBegin: null, //创建-开始时间
createTimeEnd: null, //创建-截止时间
publishTimeBegin: null, //发布-开始时间
publishTimeEnd: null, //发布-截止时间
pageNum: 1,
pageSize: PAGE_SIZE,
};
// 查询 公告 请求表单
const queryForm = reactive({ ...queryFormState });
```
## 三、Vue3 组件规范
### 3.1、 组件文件名
组件文件名应该为 pascal-case 格式
正例:
```javascript
components
|- my-component.vue
```
反例:
```javascript
components
|- myComponent.vue
|- MyComponent.vue
```
### 3.2、 父子组件文件名
和父组件紧密耦合的子组件应该以父组件名作为前缀命名
正例:
```javascript
components
|- todo-list.vue
|- todo-list-item.vue
|- todo-list-item-button.vue
|- user-profile-options.vue 完整单词
```
反例:
```javascript
components
|- TodoList.vue
|- TodoItem.vue
|- TodoButton.vue
|- UProfOpts.vue 使用了缩写
```
### 3.3、 组件属性
组件属性较多,应该主动换行。
正例:
```html
<MyComponent foo="a" bar="b" baz="c"
foo="a" bar="b" baz="c"
foo="a" bar="b" baz="c"
/>
```
反例:
```html
<MyComponent foo="a" bar="b" baz="c" foo="a" bar="b" baz="c" foo="a" bar="b" baz="c" foo="a" bar="b" baz="c"/>
```
### 3.4、 模板中表达式
组件模板应该只包含简单的表达式,复杂的表达式则应该重构为计算属性或方法。复杂表达式会让你的模板变得不那么声明式。我们应该尽量描述应该出现的是什么,而非如何计算那个值。而且计算属性和方法使得代码可以重用。
正例:
```js
<template>
<p>{{ normalizedFullName }}</p>
</template>
// 复杂表达式已经移入一个计算属性
computed: {
normalizedFullName: function () {
return this.fullName.split(' ').map(function (word) {
return word[0].toUpperCase() + word.slice(1)
}).join(' ')
}
}
```
反例:
```js
<template>
<p>
{{
fullName.split(' ').map(function (word) {
return word[0].toUpperCase() + word.slice(1)
}).join(' ')
}}
</p>
</template>
```
### 3.5、 标签顺序
单文件组件应该总是让标签顺序保持为 `<template> 、<script>、 <style>`
正例:
```java
<template>...</template>
<script>...</script>
<style>...</style>
```
反例:
```
<template>...</template>
<style>...</style>
<script>...</script>
```
## 四、Vue Router 规范
### 4.1、 页面传参
页面跳转,例如 A 页面跳转到 B 页面,需要将 A 页面的数据传递到 B 页面,推荐使用 路由参数进行传参,即 `{query:param}`
正例:
```js
let id = ' 123';
this.$router.push({ name: 'userCenter', query: { id: id } });
```
### 4.2、 path 和 name 命名规范
- path`kebab-case`命名规范尽量与vue文件的目录结构保持一致因为目录、文件名都是`kebab-case`,这样很方便找到对应的文件)
- path 必须以 / 开头即使是children里的path也要以 / 开头。如下示例
- 经常有这样的场景某个页面有问题要立刻找到这个vue文件如果不用以/开头path为parent和children组成的可能经常需要在router文件里搜索多次才能找到而如果以/开头,则能立刻搜索到对应的组件
- name 命名规范采用`KebabCase`命名规范且和component组件名保持一致因为要保持keep-alive特性keep-alive按照component的name进行缓存所以两者必须高度保持一致
```js
// 动态加载
export const reload = [
{
path: '/reload',
name: 'reload',
component: Main,
meta: {
title: '动态加载',
icon: 'icon iconfont'
},
children: [
{
path: '/reload/smart-reload-list',
name: 'SmartReloadList',
meta: {
title: 'SmartReload',
childrenPoints: [
{
title: '查询',
name: 'smart-reload-search'
},
{
title: '执行reload',
name: 'smart-reload-update'
},
{
title: '查看执行结果',
name: 'smart-reload-result'
}
]
},
component: () =>
import('@/views/reload/smart-reload/smart-reload-list.vue')
}
]
}
];
```
## 五、 Vue 项目规范
### 5.1、 目录规范
```
src 源码目录
|-- api 所有api接口
|-- assets 静态资源images, icons, styles等
|-- components 公用组件
|-- config 配置信息
|-- constants 常量信息项目所有Enum, 全局常量等
|-- directives 自定义指令
|-- i18n 国际化
|-- lib 外部引用的插件存放及修改文件
|-- mock 模拟接口,临时存放
|-- plugins 插件,全局使用
|-- router 路由,统一管理
|-- store vuex, 统一管理
|-- theme 自定义样式主题
|-- utils 工具类
|-- views 视图目录
| |-- role role模块名
| |-- |-- role-list.vue role列表页面
| |-- |-- role-add.vue role新建页面
| |-- |-- role-update.vue role更新页面
| |-- |-- index.less role模块样式
| |-- |-- components role模块通用组件文件夹
| |-- employee employee模块
```
### 5.2、 api 目录
- api文件要以api为结尾比如 `employee-api.js``login-api.js`,方便查找
- api文件必须导出对象必须以`Api`为结尾,如:`employeeApi``noticeApi`
- api中以一个对象将方法包裹
- api中的注释必须和后端 swagger 文档保持一致,同时保留后端作者
正例:
前端: `department-api.js`
```js
import { getRequest, postRequest } from '/@/lib/axios';
export const departmentApi = {
/**
* @description: 查询部门列表 @author 卓大
* @param {*}
* @return {*}
*/
queryAllDepartment: () => {
return getRequest('/department/listAll');
},
/**
* @description: 查询部门树形列表 @author 卓大
* @param {*}
* @return {*}
*/
queryDepartmentTree: () => {
return getRequest('/department/treeList');
},
/**
* @description: 添加部门 @author 卓大
* @param {*}
* @return {*}
*/
addDepartment: (param) => {
return postRequest('/department/add', param);
},
/**
* @description: 更新部门信息 @author 卓大
* @param {*}
* @return {*}
*/
updateDepartment: (param) => {
return postRequest('/department/update', param);
}
};
```
### 5.3、 assets 目录
assets 为静态资源,里面存放 images, styles, icons 等静态资源,静态资源命名格式为 kebab-case
```
|assets
|-- icons
|-- images
| |-- background-color.png
| |-- upload-header.png
|-- styles
```
### 5.4、 components 目录
此目录应按照组件进行目录划分,目录命名为 kebab-case一个组件必须一个单独的目录
目的:
- 一个组件一个目录是为了将来组件的扩展,因为这是整个项目公用的组件
- 组件入口必须为 index.vue原因也是因为这是整个项目公用的组件
举例如下:
```
|components
|-- error-log
| |-- index.vue
| |-- index.less
|-- markdown-editor
| |-- index.vue
| |-- index.js
|-- kebab-case
```
### 5.5、 constants 目录
此目录存放项目所有常量和枚举,如果常量在 vue 中使用,请使用 `src/plugin/smart-enums-plugin.js`插件,也可以使用我们写的第三方库: [vue-enum 插件](https://www.npmjs.com/package/vue-enum)但是还不支持vue3等待更新吧
具体要求:
- 常量文件要以 `const` 为结尾,比如`login-const.js``file-const.js`
- 变量要:大写下划线,比如 `LOGIN_RESULT_ENUM``LOGIN_SUCCESS``LOGIN_FAIL`
- 如果是 枚举,变量必须以 `ENUM`为结尾,如:`LOGIN_RESULT_ENUM``CODE_FRONT_COMPONENT_ENUM`
目录结构:
```
|constants
|-- index-const.js
|-- role-const.js
|-- employee-const.js
```
例子: employee-const.js
```js
export const EMPLOYEE_STATUS = {
NORMAL: {
value: 1,
desc: '正常'
},
DISABLED: {
value: 1,
desc: '禁用'
},
DELETED: {
value: 2,
desc: '已删除'
}
};
export const EMPLOYEE_ACCOUNT_TYPE = {
QQ: {
value: 1,
desc: 'QQ登录'
},
WECHAT: {
value: 2,
desc: '微信登录'
},
DINGDING: {
value: 3,
desc: '钉钉登录'
},
USERNAME: {
value: 4,
desc: '用户名密码登录'
}
};
export default {
EMPLOYEE_STATUS,
EMPLOYEE_ACCOUNT_TYPE
};
```
### 5.6、 router 与 store 目录
这两个目录一定要将业务进行拆分,不能放到一个 js 文件里。
`router` 尽量按照 views 中的结构保持一致
`store` 按照业务进行拆分不同的 js 文件
### 5.7、 views 目录
目录要求,按照模块划分,其中具体文件名要求如下:
- 如果是列表页面要以list为结尾`role-list.vue``cache-list.vue`
- 如果是 表单页面,要以 form为结尾`role-form.vue``notice-add-form.vue`
- 如果是 modal弹窗要以 modal为结尾如 表单弹窗 `role-form-modal.vue`,详情 `role-detail-modal.vue`
- 如果是 drawer 抽屉页面,要同上以 `Drawer`为结尾
```js
|-- views 视图目录
| |-- role role模块名
| | |-- role-list.vue role列表页面
| | |-- role-add-form.vue role新建页面
| | |-- role-update-form-modal.vue role更新页面
| | |-- index.less role模块样式
| | |-- components role模块通用组件文件夹
| | | |-- role-title-modal.vue role弹出框组件
| |-- employee employee模块
| |-- behavior-log 行为日志log模块
| |-- code-generator 代码生成器模块
```
## 联系我们
[1024创新实验室-主任:卓大](https://zhuoda.vip),混迹于各个技术圈,研究过计算机,熟悉点 java略懂点前端。
[1024创新实验室河南·洛阳](https://1024lab.net) 致力于成为中原领先、国内一流的技术团队,以技术创新为驱动,合作各类项目。
<table>
<tr>
<td><img src="http://img.zhuoluodada.cn/wechat/zhuoda-wechat.jpg"/></td>
<td><img src="http://img.zhuoluodada.cn/wechat/xiaozhen-gzh.jpg"/></td>
<td><img src="http://img.zhuoluodada.cn/wechat/zhuoda-wechat-money-v1.jpg"/></td>
</tr>
<tr>
<td style="text-align:center">加 主任 “卓大” 微信 <br> 拉你入群,一起学习</td>
<td style="text-align:center">关注 “小镇程序员” <br> 分享代码与生活、技术与赚钱</td>
<td style="text-align:center">请 “1024创新实验室” 喝咖啡 <br> 支持我们的开源与分享 </td>
</tr>
</table>