联系客服
客服二维码

联系客服获取更多资料

微信号:LingLab1

客服电话:010-82185409

意见反馈
关注我们
关注公众号

关注公众号

linglab语言实验室

回到顶部
Vue项目接入Eslint与+Prettier+Husky自动格式化代码

424 阅读 2020-07-25 09:52:02 上传

以下文章来源于 西语语言学工作坊

简介

前端项目,主要有两种选择ESLintTSLintTSLint仅针对TS代码,因此如果采用TSLint规范TS代码,JS代码需要采用其他工具。而ESLint不仅能规范js代码,通过配置解析器,也能规范TS代码。此外由于性能问题,TypeScript 官方决定全面采用ESLint

Eslint的主要功能包含代码格式的校验,代码质量的校验,JS规范,如用===而不是==判断相等、用驼峰命名变量而不是用下划线。而 Prettier 是美丽的意思,只是代码格式的校验(并格式化代码),不会对代码质量进行校验,如单行代码长度、tab 长度、空格、逗号表达式等问题。在实际项目中,eslint可以检测出代码问题,并标红,但是并不会自动格式化,需要手动格式化,接入Prettier并配置(需要分webstormvscode)可以进行自动化。但是需要考虑到的是,prettiereslint的规则有可能冲突,因此需要考虑到当冲突时,需要解决冲突,以谁的标准为准,正常是以prettier为准。

起步

如果是新的项目,用vue官方的脚手架拉取项目时,在下面的步骤中选择Eslint+Prettier就好了,我们可以发现,官方也是集成了Eslint+Prettier的,因此还是比较靠谱的:

安装 Eslint 依赖

// .eslintrc.js
module.exports = {
  roottrue,
  parserOptions: {
    // 定义ESLint的解析器
    parser: "babel-eslint",
    sourceType"module",
  },
  // 指定代码的运行环境
  env: {
    browsertrue,
    nodetrue,
    es6true,
  },
  extends: [
    //继承 vue 的标准特性
    "plugin:vue/essential",
    "eslint:recommended",
  ],
  // 自定义eslint规则,严格按照StandardJS
  rules: {
    "vue/max-attributes-per-line": [
      2,
      {
        singleline10,
        multiline: {
          max1,
          allowFirstLinefalse,
        },
      },
    ],
    "vue/singleline-html-element-content-newline""off",
    "vue/multiline-html-element-content-newline""off",
    "vue/name-property-casing": ["error""PascalCase"],
    "vue/no-v-html""off",
    // 两个空格缩进
    indent: [
      2,
      2,
      {
        SwitchCase1,
      },
    ],
    // 单引号
    quotes: [
      2,
      "single",
      {
        avoidEscapetrue,
        allowTemplateLiteralstrue,
      },
    ],
    // 未使用的变量
    "no-unused-vars": [
      2,
      {
        vars"all",
        args"after-used",
      },
    ],
    // 关键字前后空格
    "keyword-spacing": [
      2,
      {
        beforetrue,
        aftertrue,
      },
    ],
    // function关键字和函数名后面的空格
    "space-before-function-paren": [2"never"],
    // 除了null,其他用===而不是==
    eqeqeq: ["error""always", { null"ignore" }],
    // 字符串拼接操作符直接用空格
    "space-infix-ops"2,
    // 逗号前面不用空格,逗号后面用空格
    "comma-spacing": [
      2,
      {
        beforefalse,
        aftertrue,
      },
    ],
    // else必须和反花括号一行
    "brace-style": [
      2,
      "1tbs",
      {
        allowSingleLinetrue,
      },
    ],
    // 多行 if 语句的的括号不能省
    curly: [2"multi-line"],
    // 使用浏览器全局变量时加上 window. 前缀
    "no-undef"2,
    // 不允许有连续多行空行
    "no-multiple-empty-lines": [
      2,
      {
        max1,
      },
    ],
    // 换行符在运算符的位置
    "operator-linebreak": [
      2,
      "after",
      {
        overrides: {
          "?""before",
          ":""before",
        },
      },
    ],
    // 条件语句中赋值语句
    "no-cond-assign"2,
    // 单行代码块两边加空格
    "block-spacing": [2"always"],
    // 对属性名强制使用驼峰
    camelcase: [
      0,
      {
        properties"always",
      },
    ],
    // 不允许有多余的行末逗号
    "comma-dangle": [2"never"],
    // 始终将逗号置于行末
    "comma-style": [2"last"],
    // 点号操作符须与属性需在同一行
    "dot-location": [2"property"],
    // 函数调用时标识符与括号间不留间隔
    "func-call-spacing": ["error""never"],
    // 键值对当中冒号与值之间要留空白
    "key-spacing": [
      2,
      {
        beforeColonfalse,
        afterColontrue,
      },
    ],
    // 构造函数要以大写字母开头, 但调用大写字母开头的函数不一定需要new
    "new-cap": [
      2,
      {
        newIsCaptrue,
        capIsNewfalse,
      },
    ],
    // 无参的构造函数调用时要带上括号
    "new-parens"2,
    // 对象中定义了存值器,一定要对应的定义取值器
    "accessor-pairs"2,
    // 子类的构造器中一定要调用 super
    "constructor-super"2,
    // 使用数组字面量而不是构造器
    "no-array-constructor""error",
    // 避免使用 arguments.callee 和 arguments.caller
    "no-caller"2,
    // 避免对类名重新赋值
    "no-class-assign"2,
    // 避免修改使用 const 声明的变量
    "no-const-assign"2,
    // 正则中不要使用控制符
    "no-control-regex""error",
    // 不要对变量使用 delete 操作。
    "no-delete-var"2,
    // 不要定义冗余的函数参数
    "no-dupe-args"2,
    // 类中不要定义冗余的属性
    "no-dupe-class-members"2,
    // 对象字面量中不要定义重复的属性
    "no-dupe-keys"2,
    // switch 语句中不要定义重复的 case 分支
    "no-duplicate-case"2,
    // 同一模块有多个导入时一次性写完
    "no-duplicate-imports""error",
    // 正则中不要使用空字符
    "no-empty-character-class"2,
    // 不要解构空值
    "no-empty-pattern"2,
    //
    "no-eval"2,
    "no-ex-assign"2,
    "no-extend-native"2,
    "no-extra-bind"2,
    "no-extra-boolean-cast"2,
    "no-extra-parens": [2"functions"],
    "no-fallthrough"2,
    "no-floating-decimal"2,
    "no-func-assign"2,
    "no-implied-eval"2,
    "no-inner-declarations": [2"functions"],
    "no-invalid-regexp"2,
    "no-irregular-whitespace"2,
    "no-iterator"2,
    "no-label-var"2,
    "no-labels": [
      2,
      {
        allowLoopfalse,
        allowSwitchfalse,
      },
    ],
    "no-lone-blocks"2,
    "no-mixed-spaces-and-tabs"2,
    "no-multi-spaces"2,
    "no-multi-str"2,
    "no-new-func""error",
    "no-new-object"2,
    "no-new-require"2,
    "no-new-symbol"2,
    "no-new-wrappers"2,
    "no-obj-calls"2,
    "no-octal"2,
    "no-octal-escape"2,
    "no-path-concat"2,
    "no-proto"2,
    "no-redeclare"2,
    "no-regex-spaces"2,
    "no-return-assign": [2"except-parens"],
    "no-self-assign"2,
    "no-self-compare"2,
    "no-sequences"2,
    "no-shadow-restricted-names"2,
    "no-sparse-arrays"2,
    "no-template-curly-in-string""error",
    "no-this-before-super"2,
    "no-throw-literal"2,
    "no-trailing-spaces"2,
    "no-undef-init"2,
    "no-unmodified-loop-condition"2,
    "no-unneeded-ternary": [
      2,
      {
        defaultAssignmentfalse,
      },
    ],
    "no-unreachable"2,
    "no-unsafe-finally"2,
    "no-unsafe-negation""error",
    "no-useless-call"2,
    "no-useless-computed-key"2,
    "no-useless-escape"0,
    "no-useless-rename"2,
    "no-whitespace-before-property"2,
    "no-with"2,
    "padded-blocks": [2"never"],
    "rest-spread-spacing": ["error""never"],
    "semi-spacing": [
      2,
      {
        beforefalse,
        aftertrue,
      },
    ],
    "space-before-blocks": [2"always"],
    "space-in-parens": [2"never"],
    "space-unary-ops": [
      2,
      {
        wordstrue,
        nonwordsfalse,
      },
    ],
    "spaced-comment": [
      2,
      "always",
      {
        markers: [
          "global",
          "globals",
          "eslint",
          "eslint-disable",
          "*package",
          "!",
          ",",
        ],
      },
    ],
    "template-curly-spacing": [2"never"],
    "use-isnan"2,
    "valid-typeof"2,
    "wrap-iife": [2"any"],
    "yield-star-spacing": [2"both"],
    yoda: [2"never"],
    // 分号
    semi: [2"never"],
    "no-unexpected-multiline"2,
    "arrow-spacing": [
      2,
      {
        beforetrue,
        aftertrue,
      },
    ],
    "eol-last"2,
    "generator-star-spacing": [
      2,
      {
        beforetrue,
        aftertrue,
      },
    ],
    "handle-callback-err": [2"^(err|error)$"],
    "jsx-quotes": [2"prefer-single"],
    "no-array-constructor"2,
    "no-console""off",
    "no-native-reassign"2,
    "no-negated-in-lhs"2,
    "no-shadow-restricted-names"2,
    "no-spaced-func"2,
    "no-useless-constructor"2,
    "one-var": [
      2,
      {
        initialized"never",
      },
    ],
    "prefer-const"2,
    "no-debugger": process.env.NODE_ENV === "production" ? 2 : 0,
    "object-curly-spacing": [
      2,
      "always",
      {
        objectsInObjectsfalse,
      },
    ],
    "array-bracket-spacing": [2"never"],
  },
  //当使用第三方的SDK时,eslint会报找不到,可以加入到globals,取消对这个的检查
  globals: {
    fengmaptrue,
  },
};

如果有需要忽略的文件也可以在 .eslintignore 文件中进行配置:

npm i -D prettier eslint-plugin-prettier eslint-config-prettier prettier-eslint-cli
  • prettierprettier插件的核心代码

  • eslint-plugin-prettier:将prettier作为ESLint规范来使用

  • eslint-config-prettier:解决ESLint中的样式规范和prettier中样式规范的冲突,以prettier的样式规范为准,使ESLint中的样式规范自动失效

  • prettier-eslint-cli:prettier-eslint-cli 允许你对多个文件用prettier-eslint进行格式化。

Prettier 配置

在项目的根目录下创建.prettierrc.js文件并配置prettier代码检查规则:

// .eslintrc.js
module.exports = {
  // 其他配置。。。
  extends: [
    //继承 vue 的标准特性
    "plugin:vue/essential",
    "eslint:recommended",
    //避免与 prettier 冲突
    "plugin:prettier/recommended",
  ],
  // 其他配置不变。。。
};

开发工具配置

公司的开发工具可能有VscodeWebstorm。因此这里分开配置,并自行找到对应的方案。

VScode 配置

首先需要安装EslintPrettierVetur插件。如下图所示:

然后打开setting将下面的代码黏贴进去,这样在 VScode 中保存(ctrl+s)的时候就会进行自动格式化:

npm install husky lint-staged --save-dev
  • husky: 在项目中添加git钩子,在 git 各个生命周期(姑且这样称呼吧)中执行一些自定义操作。我们这里主要是用在 git提交之前执行 linter 操作,不通过则提交无效。

  • lint-staged: 简而言之,就是只针对 git 提交的文件进行一些操作,而非整个项目的所有文件。我们这里主要是用在 git 提交之前进行 linter 时只针对提交的文件,以进行渐进式的重构。

// lint-staged.config.js
module.exports = {
  "src/**/*.{js,vue}": ["eslint --fix""git add"],
  "src/**/*.{vue,html,css,scss,sass}": ["stylelint --fix""git add"],
  "src/**/*.{js,vue,html,css,scss,sass}": [
    "prettier-eslint --write",
    "git add",
  ],
};

总结

基本上,这些工具初次配置起来还是非常麻烦的,但这是一件一劳永逸的事情,所以还是值得花时间去做的。


点赞
收藏
表情
图片
附件