剑客
关注科技互联网

Element 一套优雅的 Vue 2 组件库是如何开发的

Element 一套优雅的 Vue 2 组件库是如何开发的

今年的杭 JSConf 大会上,受到gridcontrol 作者现场开源项目的感染,我们也在现场宣布开源这套基于 Vue 2 开发的组件库 —— Element。上场前五分钟才建的空仓库,到晚上我们真正推代码上去已经收(pian)到了 100 多 star,而且仅仅三天时间就获得了 1k star。这个项目其实早在 Vue 2 进入 beta 时就开始开发了,一直到八月底才接近尾声。初期也遇到一些棘手的问题,很庆幸都找到了解决方案。在这里整理一些解决方案分享给大家。

如何管理多个独立的组件项目 –lerna

最初制定的目标是有一个主项目管理所有的组件弄成一个包,然后每一个组件都是单独一个包。这样用户可以安装所有组件也可以只安装自己需要的组件。于是我们最直接的做法就是一个组件建一个项目,到后面组件越来越多管理起来也越加复杂,每一次升级主仓库就要更新一堆依赖组件的版本号。而且开发起来也不方便。

后来看到 babel 的仓库的目录结构很是奇特。一个 packages 文件夹下有所有的 babel 官方插件,直到发现他们用了一个叫 lerna 的工具。可以让你在主项目下管理多个子项目,从而解决了多个包互相依赖,且发布时需要手动维护多个包的问题。

所以我们重构了目录结构,所有插件单独一个项目放在 packages 目录下,可单独打包发布;同时最外面的 src 目录下的入口文件引入所有组件,打包发布的主项目就是包含了所有的组件。从而就解决了多个子项目管理的问题。

element/
  package.json
  packages/
    component-a/
      package.json
    component-b/
      package.json

如何解决定制多套主题的问题

组件库一般都会自带一套主题,也可能会有多套主题可供选择,当然也要满足用户自定义的需求。所以 Vue 推荐的定义 scope 的样式就不可行了,同时也不能把样式写在组件里。那么如何写样式同时单独发布的组件如何引用样式文件也是我们要解决的问题。

为了方便用户覆盖样式,我们采用 BEM 风格来写 CSS,这样的好处是类名基本都是一级,少数才会有嵌套情况,这样很容易的就可以直接覆盖掉原有样式。我们使用了自家开发的 postcss-salad PostCSS 插件来写样式。集成了多个实用的 PostCSS 插件同时也支持 BEM 风格。

并且样式文件目录也作为单独一个子项目发布,这样引入整个包可以包含样式文件,单独安装的组件可以通过安装主题包的方式引入样式文件。

这么做的好处是方便以后扩展其他主题,或者开发者可以自己定义一套其他 CSS 预处理的版本例如 Less 或 Scss。

文档是如何工作的 — vue-markdown-loader

当初写 Mint UI 时就遇到了要用 Vue 写文档的问题:如何才能在写 Markdown 时也能写 Vue 组件的 Demo。虽然后来并没有在 Mint UI 的文档里写 Demo。最开始在 Element 的内部版本里,找遍了各种 Vue 的 Markdown 相关插件,要么是在 template 里定义 Markdown 格式,要么就是有一个 Markdown 的组件。都不能做到纯粹的写 Markdown 文件,并且还能写 Demo。

后来想到或许可以尝试把 Markdown 文件转成 Vue 组件。毕竟可以在 Markdown 里写 HTML,那么完全可以作为 Vue 的模板。后来就有了 vue-markdown-loader,一个把 Markdown 转成 Vue 组件的 webpack loader,搭配 vue-router 就能搭建一个可以在 Markdown 里写 Vue 代码的文档网站。

如何同时打包多个组件 — cooking

由于前面的设定,最终需要每个组件都单独打包一份并发布。同时每一个组件下面都会有一个对应的配置文件,传统的用 webpack 打包虽然支持传入数组,但是并不能传入多个文件。但是我们有 cooking,一个基于 webpack 但是配置更简单同时使用上也更容易的工具。打包时只需要同时传入多个配置文件,利用 webpack 接受数组配置项的特性,就能同时进行打包。

有了这些工具,让 Element 的开发工作变得更容易且更高效。现在 Element 已经正式开源,并且我们已经放出了文档,同时还放出了设计资源,欢迎各位 Vue 开发者来尝试,也欢迎来做贡献。

————-

项目链接:

分享到:更多 ()

评论 抢沙发

  • 昵称 (必填)
  • 邮箱 (必填)
  • 网址