組件庫按需引入
在目前,全部的組件會被打包進一個文件,組件庫是一骨碌加載完全部組件,同時也會打包和加載多餘的代碼。對於小項目這樣沒有問題,可是當組件庫愈來愈龐大、豐富,特別是像咱們帶業務邏輯的非js庫,代碼量會更大,若是無論不顧的一通加載完全部資源,後期確定會帶來業務方面的體驗問題。javascript
因此首要的問題是實現源代碼的按需引入,而按需引入的前提是實現源碼包按獨立組件分割和的拆分打包。css
單個組件獨立構建打包的思路就是給打包工具提供多個獨立的入口,根據入口收集其所依賴的資源。一個組件入口產出一個文件html
webpackvue
使用 webpack 配置多入口的方式來按模塊拆分打包,每個模塊做爲一個入口,與此同時產出對應的文件。java
webpack 的配置比較簡單,不展開.node
實際構建 library 使用 webpack 有不小的缺點, 應爲 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; } });
哪些額外的代碼看着有點不那麼清爽.git
rollupgithub
使用 rollup來打包 library,rollup相較於 webpack 在打包體積上會跟小,更加適合 .web
rollup 的特色:
.js
, .min.js
是驢是馬拉出來溜溜~~
npm install --global rollup // or cnpm install --global rollup
rollup 的迭代比較快,這裏稍微留意一下 rollup 的版本,部分插件可能不兼容
在項目根目錄下建立 rollup.config.js 文件:
// rollup.config.js export default { input: 'packages/index.js', output: { file: 'lib/app.all.js', format: 'cjs' } };
就這麼簡單,執行如下命令開始將裝個 packages 構建構建爲一個單文件
rollup -c rollup.config.js
Rollup 配置文件是一個標準 ES6 模塊,默認處處一個對象,也能夠處處一個對象用來構建多個模塊
// rollup.config.js export 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.js const 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
文件的編譯須要使用rollup-plugin-vue2
插件:
npm rollup-plugin-vue2 --save-dev
樣式處理可使用 rollup-plugin-css-only
插件。因爲不喜歡笨拙的css,習慣了scss語法,就直接使用 scss,但其配置就相對要複雜一點。
scss樣式處理可使用rollup-plugin-scss
插件,爲了靈活的處理樣式我使用Postcss
html中引入的圖片在組件庫部署後就沒法正常訪問了,這裏使用 rollup-plugin-url
插件將內嵌的圖片編譯爲 base64 再直接放入 js 文件中。
對於組件庫中有較多大尺寸的圖片建議直接將圖片放入圖片服務器,而後經過url 引入,避免打包文件過大的問題.
加入 rollup 插件後的配置:
// rollup.config.js const 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.js var 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 const 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
安裝組件庫,如:
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 = false Vue.use(ElementUI) Vue.use(qwui) new Vue({ render: h => h(App), }).$mount('#app')
按需引入
按需引入須要藉助 babel-plugin-import,咱們能夠只引入須要的組件,以達到減少項目體積的目的.
首先npm install babel-plugin-import --save-dev
,而後再項目根目錄上新建文件.babelrc
.
vue cli3
直接修改babel.config.js
文件:
// babel.config.js module.exports = { presets: [ '@vue/app' ], plugins: [["import", { "libraryName": "qw-ui", "customName": (name) => { return `../lib/${name}/index`; }, "style": (name) => { return `${name}.css`; } }]] }
// src/main.js import 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 = false Vue.use(ElementUI) Vue.use(message) new Vue({ render: h => h(App), }).$mount('#app')
這時候已經啓用了 babel-plugin-import ,插件會幫咱們將import { message } from 'qw-ui'
轉換成 import message from 'qw-ui/lib/message'
的寫法。同時自動注入組件樣式。
~ 運行一下項目~~
私有npm
業務性組件庫通常只適合於公司內部,組件或多或少的也涉及到代碼安全和商業風險,因此將打包後的組件庫發佈到私有npm
而不是發佈到公網上的npm官網相對要安全不少.
私有 npm 倉庫可讓咱們使用組件就像 npm 官方倉庫裏的包同樣方便。
私有 npm 倉庫有如下一些特性:
私有npm的搭建有多種方式,最簡單的使用 git 倉庫做爲私有倉庫.
快速搭建和部署私有的 npm 包管理服務也可使用 verdaccio
對權限、安全性、穩定性有更高要求的可使用 cnpmjs.org, cnpmjs.org 服務的搭建須要配合數據庫使用.
完!