找回密码
 立即注册

QQ登录

只需一步,快速开始

断天涯大虾
社区贡献组   /  发表于:2017-3-20 10:54  /   查看:5019  /  回复:0
本帖最后由 断天涯大虾 于 2017-3-20 10:59 编辑

最近有个项目使用了 Vue.js ,本来一直使用的是 PHPStorm 来进行开发,可是遇到了很多问题。
后来,果断放弃收费的 PHPStorm ,改用 vscode (Visual Stdio Code)

当然 vscode 对 vue 也不是原生支持的,今天来扒一扒如何配置 vscode 以便优雅地编辑 vue 文件

先来扒一扒使用 PHPStorm 遇到的问题:
  • vue文件虽然可以通过插件来解决高亮问题,但是 <script> 标签中的 ES6 代码的识别老是出问题,箭头函数有的时候能正确识别,有的时候会报错
  • 无法正确识别 vue 文件中的 jsx 语法
  • 无法正确识别和高亮 vue 文件 <style> 标签中使用的 less 语法
  • vue文件中 <template> 部分使用了大量的自定义标签(自定义组件)和自定义属性,会报一堆 warning
  • 经常性卡死
  • webpack实时编译的错误不能直接展示在代码编辑器内,还得自己到控制台中查看


如何安装 vscode
很简单,传送门: 官网下载安装

第一步,要支持 vue 文件的基本语法高亮
这里,我试过好3个插件: vue , VueHelper 和 vetur ,最终选择使用 vetur 。
安装插件: Ctrl + P 然后输入 ext install vetur 然后回车点安装即可。
p.s: vscode 的插件安装比 PHPStorm 的插件安装更快捷方便,安装完成后还不用重启整个程序,只要重新加载下工作区窗口就可以了。

安装完 vetur 后还需要加上这样一段配置下:
  1. "emmet.syntaxProfiles": {
  2.   "vue-html": "html",
  3.   "vue": "html"
  4. }
复制代码

这时可以打开一个vue文件试试,注意下右下角状态栏是否正确识别为 vue 类型:
如果被识别为 text 或 html ,则记得要点击切换下。

第二步,要支持 vue 文件的 ESLint
可能还有人会问为什么要 ESLint ?没有 lint 的代码虽然也可能可以正确运行,但是 lint 作为编译前的一道检测成本更小,而且更快。此外, ESLint 还有很多规范是帮助我们写出更加优雅而不容易出错的代码的。

jshint 本来也是个不错的选择,但是 ESLint 对 jsx 的支持让我还是选择了 ESLint.
安装插件: Ctrl + P 然后输入 ext install eslint 然后回车点安装即可。

ESLint 不是安装后就可以用的,还需要一些环境和配置:
首先,需要全局的 ESLint , 如果没有安装可以使用 npm install -g eslint 来安装。
其次,vue文件是类 HTML 的文件,为了支持对 vue 文件的 ESLint ,需要 eslint-plugin-html 这个插件。可以使用 npm install -g eslint-plugin-html 来安装

接着,安装了 HTML 插件后,还需要在 vscode 中配置下 ESLint:
  1. "eslint.validate": [
  2.         "javascript",
  3.         "javascriptreact",
  4.         "html",
  5.         "vue"
  6.     ],
  7.     "eslint.options": {
  8.         "plugins": ["html"]
  9.     },
复制代码

最后,别忘了在项目根目录下创建 .eslintrc.json , 如果还没创建,还可以使用下面快捷命令来创建:
这样一来 vue 中写的 js 代码也能正确地被 lint 了。

要是不小心少个括号之类的都可以有对应的报错:

多余 import 也都能报错:
还是蛮智能的。

第三步,配置构建任务
vue 项目的构建我选择用 webpack ,不过,并不是直接使用命令行下的 webpack 而是使用了 webpack 的 API 写的 node 脚本。 脚本主要有两个,一个是 build/bin/build.js 另一个是 build/bin/watch.js 分别是单次构建和实时构建。

于是乎,对应 vscode 中的 tasks 也是有两个: build 和 watch ,简单配置如下:
  1. {
  2.     // See https://go.microsoft.com/fwlink/?LinkId=733558
  3.     // for the documentation about the tasks.json format
  4.     // use `Ctrl+P` and type `task` + SPACE + <taskName> to run a task
  5.     "version": "0.1.0",
  6.     "tasks": [
  7.         {
  8.             "taskName": "build",
  9.             "echoCommand": true,
  10.             "command": "node",
  11.             "args": [
  12.                 "build/bin/build.js"
  13.             ],
  14.             "suppressTaskName": true,
  15.             "isBuildCommand": true
  16.         },
  17.         {
  18.             "taskName": "watch",
  19.             "echoCommand": true,
  20.             "command": "node",
  21.             "args": [
  22.                 "build/bin/watch.js"
  23.             ],
  24.             "suppressTaskName": true,
  25.             "isBackground": true
  26.         }
  27.     ]
  28. }
复制代码
这样配置好后,按 Ctrl + Shift + B 即可开始单次构建。 不过单次构建比较慢(要10秒+),一般我都用实时构建: Ctrl + P 然后输入 task watch <回车> 即可开始实时构建。实时构建除了第一次比较慢,其他时候还是非常快的,一般1秒内就可以构建好。

最后,webpack 构建错误提示
webpack 构建失败后一般都会有错误提示,会显示在输出窗口中:

为啥是彩色的? 因为装了 Output Colorizer 这个插件。
当然,这样还是不够方便 -- 实时构建是后台运行的,“输出”窗口一般也都是在后台,每次保存完文件还得点开岂不麻烦。

要是能做到像 ESLint 一样直接把错误标到编辑器上面就好了。真的可以吗?翻了下 vscode 的文档,发现有神奇的 problemMatcher -- 可以对任务输出进行解析,解析出的问题会显示在“问题”窗口中,如果还有文件名行号和列号,则会在源代码编辑窗口中对应的位置标出来。

先放个最终效果:

在这个文件的第32行,import 了一个不存在的模块,这样的错误在 ESLint 中当然是检查不出来的,然而在 webpack 的实时构建中会报错:

这个事情的困难在于两点:
  • 如何通过 problemMatcher 把这个错误给抓出来?
  • 如何找到错误对应的行号?(如果可能的话,还有列号)


webpack的错误输出格式并不是完全统一的,而且有些还没有行号 -- 一方面可能是 webpack 的 bug ,另一方面 vue 文件在构建的时候会拆成 template, script 和 style 三个方面进行构建,报错的行号可能对不上。

最终我的解决方案是对 webpack 的错误重新格式化输出,然后匹配:
首先,重新格式化输出需要 format-webpack-stats-errors-warnings 这个包(偶新写的)
  1. npm install --save-dev format-webpack-stats-errors-warnings
复制代码

然后,到 build/bin/build.js 和 build/bin/watch.js 中在 webpack 构建完成的回调函数中增加这个格式化后的输出:
更多使用介绍见 github

最后,在 .vscode/tasks.json 中,每个任务下添加 problemWatcher :
  1. // ...
  2. {
  3.     "taskName": "build",
  4.     // ...
  5.     // build 任务的:
  6.     "problemMatcher": {
  7.         "owner": "webpack",
  8.         "fileLocation": [
  9.             "relative",
  10.             "${workspaceRoot}"
  11.         ],
  12.         "pattern": {
  13.             "regexp": "^!>(\\w+): (\\S+)?:(\\d+),(\\d+)(?:~(?:(\\d+),)?(\\d+))?: (.*)[        DISCUZ_CODE_4        ]quot;,
  14.             "severity": 1,
  15.             "file": 2,
  16.             "line": 3,
  17.             "column": 4,
  18.             "endLine": 5,
  19.             "endColumn": 6,
  20.             "message": 7
  21.         }
  22.     }
  23. }

  24. {
  25.     "taskName": "watch",
  26.     // ...
  27.     // watch 任务的:
  28.     "problemMatcher": {
  29.         "owner": "webpack",
  30.         "fileLocation": [
  31.             "relative",
  32.             "${workspaceRoot}"
  33.         ],
  34.         "pattern": {
  35.             "regexp": "^!>(\\w+): (\\S+)?:(\\d+),(\\d+)(?:~(?:(\\d+),)?(\\d+))?: (.*)[        DISCUZ_CODE_4        ]quot;,
  36.             "severity": 1,
  37.             "file": 2,
  38.             "line": 3,
  39.             "column": 4,
  40.             "endLine": 5,
  41.             "endColumn": 6,
  42.             "message": 7
  43.         },
  44.         "watching": {
  45.             "activeOnStart": true,
  46.             "beginsPattern": "^\\s*Webpack begin run",
  47.             "endsPattern": "^\\s*Build complete at"
  48.         }
  49.     }
  50.     // ...
  51. }
  52. // ...
复制代码

注:在 watch 任务中,为了匹配何时开始和何时结束,我在 webpack 构建的 run 和 watch 时增加了一个 console.log('Webpack begin run') 的打印,而在构建完成后增加了一个 console.log("Build complete at ..") 的打印。

OK,终于基本搞定了 vscode ,可以愉快地开发 vue 应用了。

   
关于葡萄城:全球最大的控件提供商,世界领先的企业应用定制工具、企业报表和商业智能解决方案提供商,为超过75%的全球财富500强企业提供服务。

0 个回复

您需要登录后才可以回帖 登录 | 立即注册
返回顶部