摘要:從到完美,寫一個庫庫前端組件庫之前講了很多關(guān)于項目工程化前端架構(gòu)前端構(gòu)建等方面的技術(shù),這次說說怎么寫一個完美的第三方庫。使用導(dǎo)出模塊,就可以在使用這個庫的項目中構(gòu)建時使用功能。
從 1 到完美,寫一個 js 庫、node 庫、前端組件庫
之前講了很多關(guān)于項目工程化、前端架構(gòu)、前端構(gòu)建等方面的技術(shù),這次說說怎么寫一個完美的第三方庫。
1. 選擇合適的規(guī)范來寫代碼js 模塊化的發(fā)展大致有這樣一個過程 iife => commonjs/amd => es6,而在這幾個規(guī)范中:
iife: js 原生支持,但一般不會直接使用這種規(guī)范寫代碼
amd: requirejs 定義的加載規(guī)范,但隨著構(gòu)建工具的出現(xiàn),便一般不會用這種規(guī)范寫代碼
commonjs: node 的模塊加載規(guī)范,一般會用這種規(guī)范寫 node 程序
es6: ECMAScript2015 定義的模塊加載規(guī)范,但到目前為止,幾乎所有的 js 運(yùn)行環(huán)境都不支持,包括瀏覽器、node(包括 electron、nw.js)、React Native 等
針對原生不支持任何規(guī)范的運(yùn)行環(huán)境程序(如瀏覽器、React Native),建議使用 es6 規(guī)范來寫代碼,然后由工具轉(zhuǎn)換成原生 js 能夠運(yùn)行的。
而針對 node 程序,可以直接用 commonjs 規(guī)范來寫,也可由 es6 規(guī)范來寫,然后用工具轉(zhuǎn)化成 commonjs 規(guī)范。
所以,總的來說,都可以使用 es6 規(guī)范來寫代碼,然后用工具轉(zhuǎn)換成其他規(guī)范,而且 es6 的代碼可以使用 tree-shaking 功能。
參考:
IIFE(Immediately-invoked function expression)
requirejs、electron、nw.js、react-native
Tree shaking
2. 選擇合適的構(gòu)建工具對于前端項目來說,因為有靜態(tài)資源(如圖片、字體等)加載與按需加載的需求,所以使用 webpack 是不二選擇,但對于第三方庫來說,其實還有更好的選擇:rollup。
可以查看 webpack 之外的另一種選擇:rollup 了解 webpack 與 rollup 之間各自的差異與優(yōu)勢。
webpack 在打包成第三方庫的時候只能導(dǎo)出 amd/commonjs/umd,而 rollup 能夠?qū)С?amd/commonjs/umd/es6。使用 rollup 導(dǎo)出 es6 模塊,就可以在使用這個庫的項目中構(gòu)建時使用 tree-shaking 功能。
對于有樣式文件(css、less、scss)、靜態(tài)資源文件(圖片、字體)的前端組件來說,可以使用 rollup-plugin-postcss 插件配合 rollup 處理樣式文件與靜態(tài)資源文件。
參考:
webpack、rollup、rollup-plugin-postcss
webpack 之外的另一種選擇:rollup
UMD (Universal Module Definition)
tree-shaking
webpack 如何優(yōu)雅的使用tree-shaking(搖樹優(yōu)化)
3. 定好目錄結(jié)構(gòu)一般庫項目的目錄:
|-- / # 項目根目錄 |-- src/ # 源代碼目錄 |-- lib/(dist/) # 發(fā)布文件目錄 |-- test/ # 測試文件目錄 |-- ... # 更多其他目錄
如果是多包項目(一個項目里有多個 npm packages,比如 babel):
|-- / # 項目根目錄 |-- packages/ # packages 目錄 |-- pkg1/ # package1 目錄 |-- src/ # 源代碼目錄 |-- lib/(dist/) # 發(fā)布文件目錄 |-- pkg2/ # package2 目錄 |-- src/ # 源代碼目錄 |-- lib/(dist/) # 發(fā)布文件目錄 |-- ...
后面會詳細(xì)講解多包項目。
4. 搭建一個好的腳手架不管是應(yīng)用項目還是第三方庫項目,都需要搭建一個好的腳手架,來幫助我們更好的編寫代碼、構(gòu)建項目等。
可以查看 搭建自己的前端腳手架 了解一些基本的腳手架文件與工具。
比如:
.editorconfig: 用這個文件來統(tǒng)一不同編輯器的一些配置,比如 tab 轉(zhuǎn) 2 個空格、自動插入空尾行、去掉行尾的空格等,http://editorconfig.org
eslint、stylelint、prettier: 規(guī)范化代碼風(fēng)格、優(yōu)化代碼格式等
husky、lint-staged: 在 git 提交之前對代碼進(jìn)行審查,否則不予提交
.travis.yml: 一個很棒的持續(xù)集成服務(wù),https://www.travis-ci.org/
詳細(xì)的文件、工具與配置,參考 搭建自己的前端腳手架。
另外,針對開源的第三方庫,還可以有:
LICENSE: 協(xié)議文件
CONTRIBUTING.md: 項目代碼參與者
codecov.yml: 測試覆蓋率配置文件
.github: github 上的一些自定義配置,比如 issue 模板、pr 模板等
/docs: 文檔目錄
/examples: 使用示例目錄
/scripts: 腳本目錄
加上 rollup 的配置文件 rollup.config.js:
rollup.config.js如果是 node 程序,把 es6 規(guī)范轉(zhuǎn)化成 commonjs 規(guī)范:
export default { input: "src/index.js", output: { file: "lib/index.js", format: "cjs", }, };
如果是前端庫,還需要轉(zhuǎn) es6+ 到 es5、導(dǎo)出不同規(guī)范的文件(es6/commonjs/amd/umd):
import babel from "rollup-plugin-babel"; import postcss from "rollup-plugin-postcss"; export default [ { file: "lib/cjs.js", format: "cjs", }, { file: "lib/m.js", format: "esm", }, { file: "lib/umd.js", format: "umd", name: "Name", }, { file: "lib/amd.js", format: "amd", }, ].map(output => ({ input: "src/index.js", output, plugins: [ babel({ presets: ["@babel/preset-env"], }), postcss({ extract: !0 }), // 構(gòu)建樣式文件時需要這個插件 ], }));.gitignore
一般來說,我們并不希望把發(fā)布文件放到 git 的版本控制之中,而只是發(fā)布到倉庫而已,所以:
# .gitignore .DS_Store node_modules bower_components /coverage *.log .idea .vscode .eslintcache package-lock.json /lib # 把 lib 排除在外 /packages/*/lib # 多包項目package.json
{ ... # node 項目 "main": "lib/index.js", # 前端項目 "main": "lib/cjs.js", # commonjs 規(guī)范文件 "module": "lib/m.js", # es6 規(guī)范文件 "umd:main": "lib/umd.js", # umd 規(guī)范文件 "amd:main": "lib/amd.js", # amd 規(guī)范文件 "files": [ # 發(fā)布時只發(fā)布 lib 目錄下文件 "lib" ], "scripts": { ... "build": "rollup -c", # 構(gòu)建發(fā)布文件 "prepublishOnly": "npm run build", # npm publish 之前先 npm run build "pretest": "npm run build", # npm run test 之前先 npm run build }, ... }
在實際項目中,構(gòu)建工具(如 webpack)會首先找這個包中的 module 字段對應(yīng)的 es6 規(guī)范文件,并使用 tree-shaking;如果不存在,然后找 main 字段對應(yīng)的文件。
有些構(gòu)建工具可能也會用 amd 規(guī)范文件與 umd 規(guī)范文件。
參考:
搭建自己的前端腳手架
5. 構(gòu)建多包項目如果一個項目很大,需要分割成多個 npm 包進(jìn)行管理,但這些包仍然在一個項目里,并且這些包可能有相互依賴關(guān)系,這個時候就比較難以管理和開發(fā)了。
為了方便的管理多包項目,lerna 便應(yīng)運(yùn)而生,babel、create-react-app、jest、lila 等都是用 lerna 來管理多個包的。
英文不好的童鞋,可以參考 使用lerna管理大型前端項目,了解 lerna 的一些基本用法。
lerna 一般目錄文件結(jié)構(gòu)my-lerna-repo/ package.json packages/ package-1/ package.json package-2/ package.json安裝 lerna,初始化項目
# 安裝 npm i -g lerna
# 初始化 git init lerna-repo && cd lerna-repo lerna init
# 初始化后的目錄及文件 lerna-repo/ packages/ package.json lerna.json配置文件 lerna.json
{ "version": "0.5.2", # 當(dāng)前版本號 "packages": [ "packages/*" ], "command": { "publish": { # 發(fā)布配置 "ignoreChanges": [ # 哪些文件變動不會引發(fā)發(fā)布新版本 "*.md", "*.json", "*.txt", "test/**", "example/**", "package.json" ] }, "bootstrap": { "npmClient": "cnpm" # lerna bootstrap 時使用哪個 npm 客戶端 } }, "npmClientArgs": [ # npm 客戶端 運(yùn)行時的參數(shù) "--no-package-lock" ] }常用命令 lerna publish: 發(fā)布所有有更新的包
在默認(rèn)的固定模式(Fixed mode)下,這個命令會檢查 packages 目錄下哪些包的文件有更新(lerna.json 中 command.publish.ignoreChanges 除外),然后把 lerna.json 中的 version 與有更新的包中 package.json 的 version 字段更新到一個新的版本號上,最后把這些有更新的包都發(fā)布到遠(yuǎn)程倉庫上。
lerna bootstrap: 啟動建立包相互之間的 node_modules 鏈接這個命令會根據(jù)各個包下 package.json 里面的 dependencies 和 devDependencies 配置,使用 symlink 在各個包的 node_modules 下面建立引用關(guān)系。這樣就解決了相互之間有依賴而無法聯(lián)調(diào)的問題。
lerna changed: 查看哪些包有更新,可以發(fā)布一個新的版本 lerna diff [package?]: 查看包都更新了些什么 lerna run [script]: 使用 npm 運(yùn)行每個包下面的 [script]參考:
lerna
使用lerna管理大型前端項目
6. 示例單個包的 node 項目可以參考我的項目:sclean
單個包的前端項目可以參考我的項目:see-fetch
多個包的項目可以參考我的項目:lila
后續(xù)更多博客,查看 https://github.com/senntyou/blogs
作者:深予之 (@senntyou)
版權(quán)聲明:自由轉(zhuǎn)載-非商用-非衍生-保持署名(創(chuàng)意共享3.0許可證)
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/108699.html
摘要:從到完美,寫一個庫庫前端組件庫之前講了很多關(guān)于項目工程化前端架構(gòu)前端構(gòu)建等方面的技術(shù),這次說說怎么寫一個完美的第三方庫。使用導(dǎo)出模塊,就可以在使用這個庫的項目中構(gòu)建時使用功能。 從 1 到完美,寫一個 js 庫、node 庫、前端組件庫 之前講了很多關(guān)于項目工程化、前端架構(gòu)、前端構(gòu)建等方面的技術(shù),這次說說怎么寫一個完美的第三方庫。 1. 選擇合適的規(guī)范來寫代碼 js 模塊化的發(fā)展大致有...
摘要:從到完美,用和寫一個在年開源了后,緊接著在年就又開源了,就此打開了用和前端技術(shù)寫原生之路。對應(yīng)來看,程序只有兩個部分,和,而部分則分為組件和。所以,在端的純庫無在中同樣適用,如等。應(yīng)用實例便是使用開發(fā)的一個日記。 從 1 到完美,用 js 和 react-native 寫一個 APP facebook 在 2013 年開源了 react 后,緊接著在 2015 年就又開源了 react...
摘要:從到再到搭建編寫構(gòu)建一個前端項目選擇現(xiàn)成的項目模板還是自己搭建項目骨架搭建一個前端項目的方式有兩種選擇現(xiàn)成的項目模板自己搭建項目骨架。使用版本控制系統(tǒng)管理源代碼項目搭建好后,需要一個版本控制系統(tǒng)來管理源代碼。 從 0 到 1 再到 100, 搭建、編寫、構(gòu)建一個前端項目 1. 選擇現(xiàn)成的項目模板還是自己搭建項目骨架 搭建一個前端項目的方式有兩種:選擇現(xiàn)成的項目模板、自己搭建項目骨架。 ...
摘要:從到再到搭建編寫構(gòu)建一個前端項目選擇現(xiàn)成的項目模板還是自己搭建項目骨架搭建一個前端項目的方式有兩種選擇現(xiàn)成的項目模板自己搭建項目骨架。使用版本控制系統(tǒng)管理源代碼項目搭建好后,需要一個版本控制系統(tǒng)來管理源代碼。 從 0 到 1 再到 100, 搭建、編寫、構(gòu)建一個前端項目 1. 選擇現(xiàn)成的項目模板還是自己搭建項目骨架 搭建一個前端項目的方式有兩種:選擇現(xiàn)成的項目模板、自己搭建項目骨架。 ...
摘要:從到再到搭建編寫構(gòu)建一個前端項目選擇現(xiàn)成的項目模板還是自己搭建項目骨架搭建一個前端項目的方式有兩種選擇現(xiàn)成的項目模板自己搭建項目骨架。使用版本控制系統(tǒng)管理源代碼項目搭建好后,需要一個版本控制系統(tǒng)來管理源代碼。 從 0 到 1 再到 100, 搭建、編寫、構(gòu)建一個前端項目 1. 選擇現(xiàn)成的項目模板還是自己搭建項目骨架 搭建一個前端項目的方式有兩種:選擇現(xiàn)成的項目模板、自己搭建項目骨架。 ...
閱讀 3991·2021-11-18 13:21
閱讀 4802·2021-09-27 14:01
閱讀 3120·2019-08-30 15:53
閱讀 2396·2019-08-30 15:43
閱讀 1741·2019-08-30 13:10
閱讀 1521·2019-08-29 18:39
閱讀 897·2019-08-29 15:05
閱讀 3350·2019-08-29 14:14