进阶
组件库按需引入
在目前,所有的组件会被打包进一个文件,组件库是一骨碌加载完所有组件,同时也会打包和加载多余的代码。对于小项目这样没有问题,但是当组件库越来越庞大、丰富,特别是像我们带业务逻辑的非js库,代码量会更大,如果不管不顾的一通加载完所有资源,后期肯定会带来业务方面的体验问题。
所以首要的问题是实现源代码的按需引入,而按需引入的前提是实现源码包按独立组件分割和的拆分打包。
代码分拆
单个组件独立构建打包的思路就是给打包工具提供多个独立的入口,根据入口收集其所依赖的资源。一个组件入口产出一个文件
webpack
使用 webpack 配置多入口的方式来按模块拆分打包,每一个模块作为一个入口,与此同时产出对应的文件。
webpack 的配置比较简单,不展开.
实际构建 library 使用 webpack 有不小的缺点, 应为 webpack 产出后的文件中带有一层包裹代码,这种情况下在碎片化的组件库中反而会使打包体积变大,无法起到‘瘦身’的效果。如下的 webpack 包裹代码:
/* 1 *//***/ (function(module, __webpack_exports__, __webpack_require__) { 'use strict'; /* unused harmony export square */ /* harmony export (immutable) */ __webpack_exports__['a'] = cube; function square(x) { return x * x; } function cube(x) { return x * x * x; }});
哪些额外的代码看着有点不那么清爽.
rollup
使用 来打包 library,rollup相较于 webpack 在打包体积上会跟小,更加适合 .
rollup 的特点:
- Tree Shaking: 自动移除未使用的代码, 输出更小的文件
- Scope Hoisting: 所有模块构建在一个函数内, 执行效率更高
- Config 文件支持通过 ESM 模块格式书写
- 可以一次输出多种格式:
-
- 模块规范: IIFE, AMD, CJS, UMD, ESM
- Development 与 production 版本:
.js
,.min.js
是驴是马拉出来溜溜~~
全局安装 rollup
npm install --global rollup// orcnpm install --global rollup
rollup 的迭代比较快,这里稍微留意一下 rollup 的版本,部分插件可能不兼容
添加rollup.config.js
在项目根目录下创建 rollup.config.js 文件:
// rollup.config.jsexport default { input: 'packages/index.js', output: { file: 'lib/app.all.js', format: 'cjs' }};
- input:构建入口
- format:编译打包格式
- file:编译后输出目录
就这么简单,执行以下命令开始将装个 packages 构建构建为一个单文件
rollup -c rollup.config.js
添加 rollup 多文件构建
Rollup 配置文件是一个标准 ES6 模块,默认到处一个对象,也可以到处一个对象用来构建多个模块
// rollup.config.jsexport default [{ input: 'packages/a.js', output: { file: 'lib/app.a.js', format: 'cjs' }},{ input: 'packages/b.js', output: { file: 'lib/app.b.js', format: 'cjs' }}];
packages 目录为组件库源码,相关模块不固定,不适宜写死。对于这个问题通过扫描目录获取所有模块,修改 rollup.config.js :
// rollup.config.jsconst fs = require('fs-extra');const path = require('path');const packages = {};const dir = path.join(__dirname, '../packages');const files = fs.readdirSync(dir);files.forEach(file => { const absolutePath = path.join(dir, file); if (isDir(absolutePath)) { packages[file] =`packages/${file}/index.js`; }});function createRollupConfig (file, name) { const config = { input: file, output: { file: `lib/index.js` : `lib/${name}/index.js`, format: 'es', name: name, sourcemap: true } } return config}const buildPackages = []for (let name in packages) { const file = packages[name] buildPackages.push(createRollupConfig(file, name))}export default buildPackages;
这里打包文件的格式我们使用 es
,es
是指ES6
.这个时候开始构建会报错,因为rollup还不能识别组件库中的 vue 样板代码、语法,同时我们的组件库并不是纯粹的js library, 也需要处理业务组件内引用的样式和图片、字体等。
仅仅是使用 rollup 还不能实现我们的目的,需要借助一系列 rollup 插件来完成
处理vue
.vue
文件的编译需要使用rollup-plugin-vue2
插件:
npm rollup-plugin-vue2 --save-dev
处理样式
样式处理可以使用 rollup-plugin-css-only
插件。由于不喜欢笨拙的css,习惯了scss语法,就直接使用 scss,但其配置就相对要复杂一点。
scss样式处理可以使用rollup-plugin-scss
插件,为了灵活的处理样式我使用
图片处理
html中引入的图片在组件库部署后就无法正常访问了,这里使用 rollup-plugin-url
插件将内嵌的图片编译为 base64 再直接放入 js 文件中。
对于组件库中有较多大尺寸的图片建议直接将图片放入图片服务器,然后通过url 引入,避免打包文件过大的问题.
加入 rollup 插件后的配置:
// rollup.config.jsconst fs = require('fs-extra');const path = require('path');import vue from 'rollup-plugin-vue2'import postcss from 'rollup-plugin-postcss'import postcssScss from 'postcss-scss'import autoprefixer from 'autoprefixer'import base64 from 'postcss-base64'import url from 'rollup-plugin-url'import progress from 'rollup-plugin-progress'import filesize from 'rollup-plugin-filesize';function isDir(dir) { return fs.lstatSync(dir).isDirectory();}const packages = {};const dir = path.join(__dirname, '../packages');const files = fs.readdirSync(dir);files.forEach(file => { const absolutePath = path.join(dir, file); if (isDir(absolutePath)) { packages[file] =`packages/${file}/index.js`; }});function createRollupConfig (file, name) { const config = { input: file, output: { file: `lib/${name}/index.js`, format: 'es', name: name, sourcemap: true }, plugins: [ vue(), postcss({ extract: true, parser: postcssScss, plugins: [ base64({ extensions: ['.png', '.jpeg'], root: './packages/', }), autoprefixer({ add: true }), ] }), url({ limit: 10 * 1024, //include: ['.svg'] }), progress(), filesize() ] } return config}const buildPackages = []for (let name in packages) { const file = packages[name] buildPackages.push(createRollupConfig(file, name))}export default buildPackages;
到此可以运行 rollup -c rollup.config.js
打包,实现源代码按依赖关系和目录进行分拆打包:
lib ├─message │ index.js | index.css | index.js.map ├─pay │ index.js | index.css | index.js.map ├─share │ index.js | index.css | index.js.map
打包后的 packages/pay/index.js > lib/pay/index.js :
// lib/pay/index.jsvar logo = "";var message = {render: function(){var _vm=this;var _h=_vm.$createElement;var _c=_vm._self._c||_h;return _c('div',{staticClass:"message"},[_c('el-row',{staticClass:"message-test"},[_c('el-col',{staticClass:"message-row",attrs:{"span":12}},[_c('p',{staticClass:"text"},[_vm._v("message "+_vm._s(_vm.message)+" -- "+_vm._s(_vm.count))])]),_vm._v(" "),_c('el-col',{attrs:{"span":6}},[_c('p',[_vm._v("\n css backgroud image base64\n ")]),_vm._v(" "),_c('div',{staticClass:"tops"}),_vm._v(" "),_c('hr'),_vm._v(" "),_c('p',[_vm._v("\n html img 内嵌 image base64\n ")]),_vm._v(" "),_c('img',{staticStyle:{"width":"100px"},attrs:{"src":_vm.imgUrl}})])],1)],1)},staticRenderFns: [], name: 'x-message', props: { message: String, }, data: function () { return { count: '2233hahaha', imgUrl: logo } }, created: function () { console.log('logo:', this.imgUrl); }};message.install = function (Vue) { Vue.component(message.name, message);};export default message;//# sourceMappingURL=index.js.map
构建整个组件库
在支持按需引入的同时,如果还需支持完整引入整个组件库。则直接在 rollup 配置里加入一个完整的组件库入口
rollup.config.js 最终配置
// rollup.config.jsconst fs = require('fs-extra');const path = require('path');const pkg = require("../package.json")import vue from 'rollup-plugin-vue2'import postcss from 'rollup-plugin-postcss'import postcssScss from 'postcss-scss'import autoprefixer from 'autoprefixer'import base64 from 'postcss-base64'import url from 'rollup-plugin-url'import progress from 'rollup-plugin-progress'import filesize from 'rollup-plugin-filesize';function isDir(dir) { return fs.lstatSync(dir).isDirectory();}const packages = {};const dir = path.join(__dirname, '../packages');const files = fs.readdirSync(dir);files.forEach(file => { const absolutePath = path.join(dir, file); if (isDir(absolutePath)) { packages[file] =`packages/${file}/index.js`; }});const allScript = `${pkg.name}.all`packages[allScript] = `packages/index.js`;function createRollupConfig (file, name) { const config = { input: file, output: { file: name === allScript ? `lib/index.js` : `lib/${name}/index.js`, format: 'es', name: name, sourcemap: true }, plugins: [ vue(), postcss({ extract: true, parser: postcssScss, plugins: [ base64({ extensions: ['.png', '.jpeg'], root: './packages/', }), autoprefixer({ add: true }), ] }), url({ limit: 10 * 1024, //include: ['.svg'] }), progress(), filesize() ] } return config}const buildPackages = []for (let name in packages) { const file = packages[name] buildPackages.push(createRollupConfig(file, name))}export default buildPackages;
产出的构建目录:
lib ├─message │ index.js | index.css | index.js.map ├─pay │ index.js | index.css | index.js.map ├─share │ index.js | index.css | index.js.map index.js //完整组件库,包含所有组件 index.css index.js.map
到这里构建部分完成,下一步,将构建后的lib目录发布到 npm:
- 修改
package.json
version
字段与上次不一样(如: 0.1.2),否则会提交失败 - 修改
package.json
main
字段为: lib/index.js - 发布:执行 npm publish
按需引入组件
组件库发布后,我们转入业务项目
中npm
安装组件库,如:
npm i qw-ui
安装完成后,在项目node_modules
文件夹下可以找到名为_qw-ui@0.1.1@qw-ui
,即我们刚才发布的组件库.
-
完整引入
当我们想一次引入整个项目,而非单独引入每次组件时: 修改
src/main.js
,全局引入整个qw-ui
,如:import Vue from 'vue'import App from './App.vue'import ElementUI from 'element-ui';import 'element-ui/lib/theme-chalk/index.css';import qwui from 'qw-ui' // 全局引入整个组件库import 'qw-ui/lib/index.css' // 全局载入样式Vue.config.productionTip = falseVue.use(ElementUI)Vue.use(qwui)new Vue({ render: h => h(App),}).$mount('#app')
-
按需引入
按需引入需要借助 ,我们可以只引入需要的组件,以达到减小项目体积的目的.
首先
npm install babel-plugin-import --save-dev
,然后再项目根目录上新建文件.babelrc
.vue cli3
直接修改babel.config.js
文件:// babel.config.jsmodule.exports = { presets: [ '@vue/app' ], plugins: [["import", { "libraryName": "qw-ui", "customName": (name) => { return `../lib/${name}/index`; }, "style": (name) => { return `${name}.css`; } }]]}
// src/main.jsimport Vue from 'vue'import App from './App.vue'import ElementUI from 'element-ui';import 'element-ui/lib/theme-chalk/index.css';import { message } from 'qw-ui' // 按需引入 message 组件Vue.config.productionTip = falseVue.use(ElementUI)Vue.use(message)new Vue({ render: h => h(App),}).$mount('#app')
这时候已经启用了 ,插件会帮我们将
import { message } from 'qw-ui'
转换成import message from 'qw-ui/lib/message'
的写法。同时自动注入组件样式。~ 运行一下项目~~
私有npm
业务性组件库一般只适合于公司内部,组件或多或少的也涉及到代码安全和商业风险,所以将打包后的组件库发布到私有npm
而不是发布到公网上的npm官网相对要安全很多.
私有 npm 仓库可以让我们使用组件就像 npm 官方仓库里的包一样方便。
私有 npm 仓库有以下一些特性:
- 私有包托管在内部服务器或者单独的服务器上;
- 可以同步整个官方仓库,也可以只同步需要的;
- 下载的时候,可以让公共包走公共仓库,私有包走私有仓库;
- 可以缓存下载过的包;
- 对于下载,发布,有对应的权限管理。
私有npm的搭建有多种方式,最简单的使用 git 仓库作为私有仓库.
快速搭建和部署私有的 npm 包管理服务也可以使用
对权限、安全性、稳定性有更高要求的可以使用 , cnpmjs.org 服务的搭建需要配合数据库使用.
完!