vue 开发规范




  • 项目运行指南(#项目运行指南)
  • 开发本地环境(#开发本地环境)
  • 开发相关插件/工具(#开发相关插件工具)
  • 开发规范(#开发规范)
    • vue(#vue)
      • 【数据流向】(#数据流向)
      • 【慎用全局注册】(#慎用全局注册)
      • 【组件名称】(#组件名称)
      • 【组件中的 CSS】(#组件中的-css)
      • 【统一标签顺序】(#统一标签顺序)
      • 【其它注意事项】(#其它注意事项)
      • 【 !!!其它则严格遵守 vue 官方风格指南】(#a-target_blank-hrefhttpscnvuejsorgv2style-guide其它则严格遵守-vue-官方风格指南a)
    • vue-router(#vue-router)
    • vuex(#vuex)
    • 模块复用(#模块复用)
    • 其它杂项(#其它杂项)
    • 代码注释(#代码注释)
    • 工程目录结构(#工程目录结构)
    • 前端部署(#前端部署)
    • 给 UI 的建议(#给-ui-的建议)



项目运行指南

  • 安装/更新依赖包:npm install
    • 说明:进入正式开发前需要提交 package-lock.json,正式开发后慎用 npm update
  • 运行:
    • 启动为 dev 环境:npm run servenpm start
    • 打包为 stage 环境:npm run build:stage
    • 打包为 prod 环境:npm run build
    • 检查并修复源码:npm run lint
    • 运行单元测试:npm run test:unit
    • 启用静态资源服务:npm run dist
    • 版本号操作:npm version major|minor|patch
      • 版本号格式说明:major(主版本号).minor(次版本号).patch(修订号)

开发本地环境

  • 在 VSCode 中给予独立的工作区,而非文件夹,以获得最佳的智能感知
  • 新建 .env.development.local 来重写部分环境变量,如:
    • 模拟数据:VUE_APP_MOCK = true
    • 接口服务:DEV_PROXY_TARGET_API = http://10.25.73.159:8081
    • ...

开发相关插件/工具

  • VSCode 相关插件
    • 必要插件
      • ESLint
      • Vetur
      • Prettier - Code formatter
      • path Autocomplete
    • 推荐插件
      • stylelint
      • vscode-element-helper (element-ui 专用)
      • Debugger for Chrome
      • GitLens -- Git supercharged
  • Chrome 相关插件
    • 必要插件
      • vue-devtools
    • 推荐插件
      • json-formatter
  • 其它工具
    • 推荐:PostmanPostwoman

开发规范

vue

【数据流向】

  • 单个组件的数据流

    props、data/$store/$route、computed (由前面派生)
      ↓
    template/render
      ↓
    用户交互事件、初始化的异步回调
      ↓
    data/$store/$route
    
  • 组件间的数据流

    • 父向子传递用 props
    • 子向父传递用 vue 内置的自定义事件,即 this.$emit
    • 父子双向传递用 <a target="_blank" href="https://cn.vuejs.org/v2/guide/components-custom-events.html">v-model</a> 或 <a target="_blank" href="https://cn.vuejs.org/v2/guide/components-custom-events.html">.sync</a>
    • 跨越传递用 vuex(慎用 EventBus)
    • 紧密耦合的祖孙间传递也可以考虑用父组件作为中间运输层
    • 紧密耦合的兄弟间传递也可以考虑用父组件作为中转运输层

【慎用全局注册】

  • 组件、混入 ... 应使用局部注册

    局部注册可保持清晰的依赖关系,并且 IDE 智能感知更为友好

【组件名称】

  • 名称大小写

    <script>
      import MyComponent from '@/components/MyComponent.vue' // 文件名使用 PascalCase 命名法
      export default {
        name: 'ComponentName', // 必须有 name
        components: { MyComponent },
      }
    </script>
    
    <template>
      <div>
        <!-- 在 template 中一律使用 kebab-case 方式调用 -->
        <my-component />
        <el-input />
      </div>
    </template>
    
  • 使用前缀

    • 非业务通用组件使用 Base 前缀
    • <a href="#hash_Ex">扩展/包装第三方开源组件或内部公共库组件 (不建议使用高阶组件)</a> 使用 Ex 前缀
    • 单例组件使用 The 前缀

【组件中的 CSS】

  • 使用 <a target="_blank" href="https://vue-loader.vuejs.org/zh/guide/css-modules.html">CSS Modules</a>,基于如下考虑:

    • 不让外部进行样式重写,避免强耦合 (可通过 props 来处理内部样式的变化)
    • 放心使用简短且语义强的 class 名,无需多余的命名空间
  • 使用方式

    • 语法

      /* 默认为局部 */
      .xxx {
      }
      
      /* 从局部转到全局 */
      :global(.yyy) {
      }
      // or
      :global {
      }
      
      /* 从全局转到局部 */
      :local(.zzz) {
      }
      // or
      :local {
      }
      
    • 单个组件专属

      <style lang="less" module></style>
      
    • 多个组件共用 (*.module.less)

      import style from './style.module.less'
      

【统一标签顺序】

  • script --> template --> style,并使用空行分隔

【其它注意事项】

  • 慎用 this.$refs、this.$parent、this.$root、provide/inject
    • this.$refs 一般用在第三方开源组件或内部公共库组件或非常稳定的组件,以调用显式声明的方法
    • 在万不得已的情况下需要暴露方法给外部调用时需要加 pub 前缀,如:this.$refs.pubFocus()
  • 尽量不要在 watch 中直接变更数据,易造成死循环。数据变更应该交给用户交互事件或初始化的异步回调
  • 组件中的 data 及 vuex 中的 state 应该可序列化,即不要存 undefined、function 等

【 <a target="_blank" href="https://cn.vuejs.org/v2/style-guide/">!!!其它则严格遵守 vue 官方风格指南</a>】


vue-router

  • url 定义准则

    • path 对应视图变化,query 对应数据变化,hash 对应滚动条位置

    • path 使用 kebab-case 命名法,并且尽量与组件名相匹配(即一眼看到 path 就能迅速找到对应的组件)

      路由 path:/project-list
        ↓
      路由组件:@/views/ProjectList.vue | @/views/ProjectList/index.vue
      
  • 命名路由的 name 值使用 kebab-case 命名法,并且嵌套时要带命名空间(使用简写)

    export const routes = {
      path: '/user-center',
      name: 'user-center',
      // ...
      children: [
        {
          path: 'base-info',
          name: 'uc-base-info', // 带命名空间 uc-
          // ...
        },
      ],
    }
    
  • 当组件依赖 $route 作为核心数据时,要使用<a target="_blank" href="https://router.vuejs.org/zh/guide/essentials/passing-props.html">路由组件传参</a>,与 $route 解耦,也使得依赖更为显式清晰

    export const routes = {
      path: '/project-detail',
      props: ({ query: { id } }) => ({ id }),
      // ...
    }
    
  • 视图跳转尽量使用声明式(特别是 PC 端)

    <router-link :to="path | { name, ... }">使用声明式</router-link>
    <a @click="$router.push(...)">使用命令式</a>
    

vuex

  • 需要由 vuex 管理的数据

    • 组件间共享的响应式数据
    • 组件间需要跨越传递的数据
  • getter、mutation、action、module 使用驼峰命名法

  • module 应避免嵌套,尽量扁平化

  • module 应该启用命名空间 namespaced: true


模块复用

  • 避免重复造轮子,多使用成熟的现成工具/类库/组件,如:lodash、qs、url-parse、date-fns/format 等

  • 模块设计原则:

    • 高内聚低耦合、可扩展
    • 不要去改变模块输入的数据 (引用类型),如:函数参数、组件 prop
    • 模块的入参为可选 Boolean 时,默认值应设计为 false
      • 如:hasToolbar?=false 或 noToolbar?=false
  • 方法接口的设计

    // 参数类型与个数要保持稳定
    // 建议参数不要超过3个,且预留一个 options 对象,以提高扩展性
    // 方法尽量纯净 (纯函数思想)
    export function myMethod1(a, options) {} // 当必选参数只有一个时
    export function myMethod2(a, b, options) {} // 当必选参数只有两个时
    export function myMethod3(options) {} // 当必选参数有两个以上时
    export function myMethod4(options) {} // 当所有参数都是可选时
    
    // 有时为了提高灵活性,参数类型可以是两重,一重是期望值,另一重是返回期望值的函数 (可带参)
    export function myMethod5(a) {
      a = typeof a === 'function' ? a() : a
    }
    
  • <span id="hash_Ex">扩展/包装第三方开源组件或内部公共库组件 (不建议使用高阶组件)</span>

    • 使用 extends 混入 (相关命名需要加 ex 前缀,防止覆盖)
    • 使用<a target="_blank" href="https://cn.vuejs.org/v2/guide/render-function.html">函数式组件</a>包装

其它杂项

  • IDE 统一使用 VSCode,并统一使用相关插件及配置

  • js 变量声明尽量使用 const

  • js 变量或对象属性使用驼峰命名法

  • js 私有变量或对象私有属性使用 _ 前缀 (注意: <a target="_blank" href="https://cn.vuejs.org/v2/style-guide">vue 组件属性不要使用 _ 前缀</a>)

    // 表明该变量仅在 createId 方法中使用 (与 createId 方法紧挨着)
    let _count = 0
    const createId = () => `${Date.now()}_${++_count}`
    
    // 适时使用立即执行函数可以简洁作用域及保护私有变量
    const createId = (() => {
      let count = 0
      return () => `${Date.now()}_${++count}`
    })()
    
  • 导入模块时不要省略后缀(js 除外),利于 IDE 感知

  • 导入当前目录以外的模块时,建议使用'@'别名

    // js
    import XxxXxx from '@/components/XxxXxx.vue'
    
    <!-- template -->
    <img src="@/assets/logo.png" />
    
    /* style */
    @import '~@/styles/vars.less';
    .xxx {
      background: url('~@/assets/logo.png');
    }
    
  • 严格遵守 ESLint 语法校验,警告级别的也要处理 (暂时用不到的代码可以先注释掉)

  • css

    • 全局 class 使用 g- 前缀
    • CSS 选择器应避免深嵌套,尽量的扁平化
    • 关键选择器 (最右边) 避免使用通配符 *

代码注释

  • 文件头部注释

    • 脚本文件、样式文件

      /**
       * 说明
       * @author 作者
       */
      
    • vue 文件

      <!-- 说明 -->
      <!-- @author 作者 -->
      
  • js 注释 (结合 <a target="_blank" href="https://jsdoc.app/">JSDoc 注释标准</a>,帮助 IDE 智能感知)

    • 注释格式

      /**
       * 文件头部、大的区块、JSDoc
       */
      
      /* 一般的区块 */
      
      // 小的区块、行
      
    • <a target="_blank" href="https://jsdoc.app/howto-es2015-modules.html">ES 2015 Modules</a>

      /**
       * 使用 param 表示函数形参
       * 使用 returns 表示函数返回值
       * @param {类型} data
       * @param {object} [options] 可选参数
       * @param {类型} options.xxx
       * @param {类型 =} options.yyy 可选属性
       * @returns {类型}
       */
      export function myMethod(data, options) {}
      
      /**
       * 使用 type 进行类型断言
       * @type {import('vue-router').RouteConfig[]}
       */
      const routes = []
      
      /**
       * 使用 typedef 定义类型,方便多处使用(命名时需要首字母大写)
       * @typedef {routes[0]} RouteConfig
       * @param {(meta: object, route: RouteConfig) => boolean} filterCallback
       * @returns {RouteConfig[]}
       */
      export const filterMapRoutes = function(filterCallback) {}
      
      /**
       * 类型参考:https://www.tslang.cn/docs/handbook/basic-types.html
       *
       * 基本
       * @type {boolean}
       * @type {number}
       * @type {string}
       * @type {1 | 2 | 3}
       * @type {'a' | 'b' | 'c'}
       *
       * 数组
       * @type {Array}
       * @type {string[]}
       *
       * 函数
       * @type {Function}
       * @type {(data) => void}
       * @type {(data: Array) => void | boolean}
       *
       * 对象
       * @type {object}
       *
       * 联合
       * @type {number | string}
       * @type {boolean | (() => boolean)}
       *
       * 导入 ts 类型
       * @type {import('xxx').yyy}
       *
       * 从现有的 js 变量或 ts 类型进行推导
       * @type {Parameters<fn>} 取函数形参的类型
       * @type {Parameters<fn>[0]} 取函数第一个形参的类型
       * @type {ReturnType<fn>} 取函数返回值的类型
       * @type {obj['xxx']} 取指定属性值的类型(不能使用点语法)
       * ...
       */
      
    • <a target="_blank" href="https://jsdoc.app/howto-es2015-classes.html">ES 2015 Classes</a>

    • 待完成或待优化的地方

      /* TODO: 说明 */
      
  • css 注释

    • 全局样式需要写注释

      /* 说明 */
      .g-class1 {
      }
      
      /* 说明 */
      .g-class2 {
      }
      
  • vue template 注释

    • 适当使用注释与空行

      <!-- 说明 -->
      <div>block1</div>
      
      <!-- 说明 -->
      <div>block2</div>
      

工程目录结构

|-- .env.development ------------ dev 环境变量
|-- .env.development.local ------ dev 本地环境变量 (被 git 忽略,需手动新建,用来重写部分环境变量)
|-- .env.production-stage ------- stage 环境变量
|-- .env.production ------------- prod 环境变量
|-- .env.test
|-- .vscode --------------------- 统一 VSCode 插件及配置
|-- static-server.js ------------ 静态资源服务 (node 运行),通常用于预览/检查打包结果,或者临时给其他人员启用前端服务
|-- docs ------------------------ 开发文档
|   |-- README.html ------------- 由 ../README.md 手动生成 (使用 VSCode 插件 Markdown Preview Enhanced)
|   |-- xxx.md
|   |-- xxx.html
|-- public
|   |-- favicon.ico
|   |-- index.html
|   |-- libs -------------------- 不支持模块化加载的第三方 ES5 类库/模块 (只能通过全局变量引用)
|-- src
    |-- main.js
    |-- App.vue
    |-- libs -------------------- 支持模块化加载但是无法通过 npm 安装的第三方 ES5 类库/模块
    |-- assets
    |-- styles
    |   |-- global.less
    |   |-- reset.less
    |   |-- vars.less ----------- less 全局变量/函数 (webpack 自动注入)
    |   |-- xxx.less
    |-- scripts
    |   |-- utils --------------- 通用方法
    |   |-- constants ----------- 常量 (多使用 Object.freeze)
    |   |-- xxx.js
    |   |-- http ---------------- axios 实例
    |       |-- index.js
    |       |-- http.js
    |       |-- createAxios.js
    |       |-- xxx.js
    |-- injects ----------------- vue 全局注册 (慎用)
    |   |-- index.js
    |   |-- $xxx.js
    |   |-- v-xxx.js
    |   |-- mixin_xxx.js
    |   |-- xxx.js
    |-- element-ui
    |   |-- index.js
    |   |-- rewrite ------------- 主题样式复写
    |       |-- index.less
    |       |-- xxx.less
    |-- vant
    |   |-- index.js
    |   |-- vars.less ----------- 内置变量复写
    |   |-- rewrite ------------- 主题样式复写
    |       |-- index.less
    |       |-- xxx.less
    |-- router
    |   |-- index.js
    |   |-- routes.js
    |   |-- registerInterceptor.js
    |-- store
    |   |-- index.js
    |   |-- root.js
    |   |-- xxx.js
    |-- api
    |   |-- xxx.js
    |   |-- mock ---------------- 模拟数据
    |       |-- index.js
    |       |-- createMock.js
    |       |-- xxx.js
    |-- components
    |   |-- BaseXxx.vue --------- 非业务通用组件
    |   |-- TheXxx.vue ---------- 单例组件
    |   |-- ExXxx.vue ----------- 扩展/包装第三方开源组件或内部公共库组件
    |   |-- XxxXxx.vue
    |   |-- ComponentExamples --- 非单例公共组件需要在这里写示例
    |   |   |-- index.vue
    |   |   |-- XxxXxx.vue
    |   |-- SvgIcon ------------- svg-sprite 图标组件
    |   |   |-- index.vue
    |   |   |-- icons
    |   |-- directives ---------- 可复用的自定义指令(局部注册)
    |   |   |-- xxx.js
    |   |-- mixins -------------- 可复用的混入(局部注册)
    |       |-- xxx.js
    |-- views
        |-- Xxx.vue
        |-- Xxx ----------------- 除了 api 和 vuex,其它的专属模块要内聚在同一目录下
            |-- index.vue
            |-- Xxx.vue --------- 相关页面/子页面/子路由
            |-- xxx.js
            |-- xxx.module.less
            |-- components ------ 存放私有组件

前端部署

  • 跨域处理

    • 使用代理或 <a target="_blank" href="https://www.baidu.com/s?wd=cors跨域">CORS</a>
  • history 模式<a target="_blank" href="https://router.vuejs.org/zh/guide/essentials/history-mode.html">路由处理</a>

    • 如果 url 匹配的不是静态资源 (不带后缀的),则返回 /index.html 页面
  • 客户端缓存处理 (配置响应头)

    • 静态资源

      • 不缓存 /index.html

        Cache-Control: no-store
        
      • 强缓存 /static-hash/**/*

        Cache-Control: public,max-age=31536000
        
      • <a target="_blank" href="https://www.baidu.com/s?wd=http协商缓存">协商缓存</a> (默认)

        Cache-Control: no-cache
        Etag: xxx
        Last-Modified: xxx
        
    • XHR (解决 IE 缓存问题)

      Cache-Control: no-cache
      
  • gzip 压缩

    • 静态资源:启用 gzip 压缩 (除了像素型图片)

    • XHR:发给客户端的响应数据超过指定阀值时应启用 gzip 压缩


给 UI 的建议

  • 对于中后台项目,在画 UI 界面时,建议参考前端已选型的开源组件库,并推荐使用开源组件库提供的制图元件/模板,如:<a target="_blank" href="http://element-cn.eleme.io/#/zh-CN/resource">element-ui</a>
  • 对于 H5 项目,如:<a target="_blank" href="https://youzan.github.io/vant/#/zh-CN/design">vant</a>
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,539评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,911评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,337评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,723评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,795评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,762评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,742评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,508评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,954评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,247评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,404评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,104评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,736评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,352评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,557评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,371评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,292评论 2 352

推荐阅读更多精彩内容

  • 本文档为前端vue 开发规范 ·规范目的 ·命名规范 ·结构化规范 ·注释规范 ·编码规范 ·CSS 规范 规范目...
    近朱者炽阅读 799评论 0 3
  • 由于所在公司前端代码较不规范,近期应公司领导要求,整理出了一份公司内部的前端开发规范标准。这里参考了一些文章,并对...
    追寻1989阅读 1,161评论 0 3
  • 前端通用规范文档,猛戳这里 一.react一般规范 1.基本结构(使用mobx) ├── build ...
    lemonzoey阅读 3,347评论 2 4
  • 为了前端代码形成统一的风格,制定该代码规范(该规范部分参照VUE官网介绍的风格指南以及常用的规范) 1.文件夹命名...
    朋_朋阅读 599评论 0 1
  • Vue 开发规范目录及说明 规范目的 命名规范 结构化规范注 释规范 编码规范 CSS 规范 规范目的 为提高团队...
    凌枝阅读 5,271评论 1 49