依托 github+npm 开发和维护前端 lib 模块
想想那种场面,随着前端的代码的不断累积,代码冗长混杂,组件复用困难,即使提取出一些插件,往往都是要通过 script 引入 js 包,往往插件或方法又都暴露在全局,而当你引入的东西越来越多,往往都会一些问题:
1、没有命名空间,会污染全局变量,
2、如果我的插件又依赖别人的插件,别人的插件又依赖别人的插件,如此循环,又会面临处理复杂的依赖逻辑关系。
自从nodejs出现,前端的作用越来越大,解决方案也不断被提出,从利用前端模块化(解决作用域的痛点),再到 npm,webpack 等自动处理复杂的依赖关系(解决依赖关系的痛点),全面的模块化开发打包全流程管理,从青涩一步步走向成熟,这个过程也衍生出了很多规范!
衍生规范:CommonJs(node.js) --> AMD(require.js) --> CMD(seaJS) --> UMD(兼容 CommonJs, AMD, var)
一系列的规范产生和发展,最终形成了今天的格局。随着打包工具的的深度使用,node_modules lib依赖包的开发也变得越来越重要,也让我们探索到底什么样的情况适合作为一个 lib 输出。
例如:
1、与主业务逻辑无关,但又包含很多操作逻辑的富应用,如富文本编辑器之类的
2、可以提取出来的公共 UI 组件库,例如 element,iview,ant-design 等等,这些通用组件都是可以通过传入属性或参数去实例化符合要求的不同形态,这些都是通过lib输出,作为依赖包去安装后使用
3、utils 纯工具库,例如 moment,lodash等等
注:
CommonJs规范: 它是服务端模块化加载规范,由Node推广使用,只能依赖node环境执行。纳尼???这么好的东西,是不是就对传统前端开发来说没用了呢?其实,webpack就是使用这个规范去打包依赖,因为 webpack 本身就是依赖 node 环境开发出来的服务端预处理工具。标志性语法:require('module-name') / module.exports,这个 require 和 AMD 的 require 不同,下面会讲到!// 加载
require("module");
require("../file.js");
// 导出
exports.doStuff = function() {};
module.exports = someValue;
// 导出
define("module", ["dep1", "dep2"], function(d1, d2) {
return someExportedValue;
});
// 加载
require(["module", "../file"], function(module, file) { /* ... */ });
define(function(require, exports, module) {
// 加载
var $ = require('jquery');
var Spinning = require('./spinning');
// 导出
exports.doSomething = ...
module.exports = ...
})
output.libraryTarget
Which format to export the library:
"var" - Export by setting a variable: var Library = xxx (default)
"this" - Export by setting a property of this: this["Library"] = xxx
"commonjs" - Export by setting a property of exports: exports["Library"] = xxx
"commonjs2" - Export by setting module.exports: module.exports = xxx
"amd" - Export to AMD (optionally named)
"umd" - Export to AMD, CommonJS2 or as property in root
// UMD 规范
(function (root, factory) {
if (typeof define === 'function' && define.amd) {
// 兼容AMD
define(['jquery', 'underscore'], factory);
} else if (typeof exports === 'object') {
// 兼容Node, CommonJS
module.exports = factory(require('jquery'), require('underscore'));
} else {
// 兼容浏览器全局变量(root 即 window)
root.returnExports = factory(root.jQuery, root._);
}
}(this, function ($, _) {
// 方法
function a(){}; // 私有方法,因为它没被返回 (见下面)
function b(){}; // 公共方法,因为被返回了
function c(){}; // 公共方法,因为被返回了
// 暴露公共方法
return {
b: b,
c: c
}
}));
node:
Node.js® 是一个基于 Chrome V8 引擎 的 JavaScript 运行时。官网
-
怎么理解:
-
JavaScript 之前只是一种运行在客户端浏览器中的脚本语言,它只能以浏览器 webview 作为宿主执行,而不能像 java 一样直接操作于服务端,而 nodejs 的作用就是可以在服务端执行 JavaScript,让它具备像 java 一样的服务端语言的功能。
-
后端的同事可以理解 node 的 v8 引擎它就类似于 jvm,是一种可以执行 JavaScript 的运行环境。
npm:
NPM 是随同 NodeJS 一起安装的一款非常实用的包管理工具。官方搜索网站
-
主要作用:
-
用于安装和管理依赖包。 npm install 安装,npm run [script] 执行命令行等等
-
后端的同事可以把 npm 想象成一个简易版的 maven
webpack:
本质上,webpack 是一个现代 JavaScript 应用程序的静态模块打包工具。官网
-
主要功能:
-
loader: 用于对模块的源代码进行转换,对 html js css 等进行预处理。loader 可以将文件从不同的语言(如 Es6 转换成 Es5;TypeScript 转换为 JavaScript;将 inline-image 转换为 data URL;less 转换成 css...),loader 甚至允许你直接在 JavaScript 模块中 import CSS 文件!
-
plugin: 插件是 webpack 的支柱功能。插件目的在于解决 loader 无法实现的其他事,例如热更新 Hot Module Replacement(HMR:开发时只需保存代码不用手动刷新直接更新页面)。
-
devServer: 能够用于快速开发应用程序,启动本地 localhost 服务,利用 proxy 反向代理解决接口跨域等
-
后端的同事可以把它想象成 tomcat(不完全准确)。
eslint:
可配置的 JavaScript 和 JSX 代码检测工具。 官网
-
主要功能:
-
方便统一 js 代码规范和写法,可配置规则对代码进行报错或警告提示,利用插件自动修复等。
-
后端同事可以理解为类似于 java CheckStyle 静态代码检查工具。
markdown:
Markdown 是一门比 html 更简单的标记语言,其主要用于书写工程项目 readme 文档。包含:标题,图片,引用,表格,代码块,有序、无序清单等功能 官网 / 基础语法 / 示例
typescript:
是 JavaScript 类型的超集,本质上是对 js 添加了可选的静态类型和基于类的面向对象编程。
-
d.ts:
一个纯粹的类型声明文件,不涉及逻辑。 -
d.ts 作用:
-
一个好用的 library 插件智能提示是不可或缺的。因为 lib 插件本身就是安装在 node_modules 模块中,我们很少去查看它的源码,因此使用时文档和智能提示就显得非常重要
-
d.ts 就经常用于增强 IDE 智能提示。javascript 本身是弱类型,例如参数类型常常不明确,对 IDE 的提示就不是很友好,而 ts 却可以明确定义类型,如果我的代码使用纯 js 书写,却想让他拥有和 ts 一样的类型智能提示,我们就可以书写声明文件 *.d.ts 。官方文档 / *.d.ts 代码示例
jest:
Jest 是 Facebook 的一套开源的 JavaScript 测试框架, 它自动集成了断言、JSDom、覆盖率报告等开发者所需要的所有测试工具,是一款几乎零配置的测试框架。。 官网 / 代码示例,其他常用测试框架还有Mocha
-
使用场景:
-
执行测试脚本,执行完毕后提供精准的测试报告,报错代码覆盖率,有效避免人肉测试导致的场景逻辑遗漏。
-
在项目正在进行的情况下,随时进行单元测试,不用等待整体完工或者有具体可操作的界面的时候再测试。
git:
Git 是一个开源的分布式版本控制系统,可以有效、高速地处理从很小到非常大的项目版本管理
github:
就是本网站。
-
简单介绍:
-
GayHub 是全球最大的同性交友平台。~,其实 GitHub 是一个面向开源及私有软件项目的托管平台,因为只支持 git 作为唯一的版本库格式进行托管,故名 GitHub。
-
不过,为什么说它是交友网站呢?你打开 Github 上一个用户的主页,他的头像 (avatar)、喜欢的人 (following) 、喜欢他的人 (followers)、个人能力 (repositories)、活跃度 (contributions) ,一清二楚...
注:这是旧版,被微软收购后新版已改版
gitlab:
不用介绍了,它是仅次于 GitHub 的同性交友平台。
新建远程仓库并克隆到本地 --> 初始化项目 --> 配置脚手架 --> 组件开发 --> 单元测试 --> 配置声明文件 --> 撰写 readme 文档 --> 打包发布
Create a new repository
创建一个远程仓库
git clone [url]
克隆到本地
npm init
回答一系列问题后,生成 package.json
注意:
-
entry point 非常重要,这是 lib 作为 node_modules 模块化文件的入口
-
package name 有可能会被占用,如果您的包名已经被占用则无法发布。
有两种方法规避:
(1) 直接换名字;
(2) 创建一个带作用域的包名(推荐),例如:@webpack-cli/init(组织),@longlongago2/rm_calendar(用户名) ,创建带作用域的包名 cli 命令如下:npm init --scope=@my-username
or
npm init --scope=@my-org
-
git repository 一般写你的 git 代码托管地址,你如果在 github 创建的远程仓库,这里会自动链接到你的代码库,在 npm 搜索中体现
npx webpack-cli init
回答一系列问题,生成一个基础的 webpack.config.js,然后可以根据自己需要添加各种配置和插件!
webpack 配置注意:
-
关于 lib 打包配置注意:lib 组件中的依赖一定不要打包,否则可能带来严重问题。(使用webpack externals将依赖外部化。npm 会自动下载 lib 模块中的依赖,webpack 会按照 lib 依赖关系自行打包,所以不必在 lib 阶段直接将依赖打包进去,否则如果用户也使用了你 lib 中的依赖,会造成重复打包,可能造成不可预测的报错)
一般目录结构:src lib dist tests example
src: 工程代码目录
lib: 模块化打包目录,模块化引入
dist: 传统打包目录,CDN 引入
tests: 测试脚本
example: 使用示例
jest
npm install --save-dev jest
jest --init
回答一系列问题,创建 jest.config.js
.d.ts: 一般和 js 文件同名同级,或者发布到@types,一般 IDE 会自动识别并智能提示
-
自己定义的库声明文件一般和原始 js 文件同名同级:例如 jquery.js 对应的就是 jquery.d.ts,放在同一级目录。
-
强大的第三方库一般会发布到@types 组织中,以供使用者单独下载:例如 npm install @types/jquery -save 然后你就发现你项目里面多了个 node_modules/@types/jquery 文件夹。
markdown:如何安装、使用代码示例、详细 api,还可以加效果图,FAQ,Changelog 等等
npm publish
注意:
-
发布之前得先登录 npm 账号
-
如果您是再维护新版本,新版本的 package.json version 版本号一定要更改,否则发布不成功!
-
寻找答案:
- github 不仅是一个代码托管平台,它更是一个社区,issues 就可以看做订阅者的帖子,里面会有各种 bug 的解决方案和很多开发者的创意和想法等等,遇到 bug 首先应该去搜索 issues,而不是去百度,实在解决不了还可以对开发者提问。
-
提问:
- 一般有影响力的框架(vue,react...)都会有专门的维护者照看 issues 并进行日常解答和 bug 收集。遇到 bug 或疑问,我们一定要按照维护者提供的提问模板进行有效提问,否则可能会被强制关闭。
-
处理:
-
对于库的管理者来说,为了有效收集 bug 和快速定位问题,我们可以设置 issues 提问模板。
-
默认模版
将在代码库新建目录:.github
在 .github 目录下添加 ISSUE_TEMPLATE.md 文件作为 issues 默认模版。当创建 issue 时,若未建立多模版或选择了 Open a regular issue 时,系统会引用该模版。
-
多模版
将在代码库新建目录:.github/ISSUE_TEMPLATE
该目录下可添加多个 .md 文件作为 issues 模版。当创建 issue 时,系统会展示这些模版供选择。
-
自定义模板
issues 的默认模版和多模版可同时存在,也可以自己定义符合自己需求的模板
-
-
如何创建模板
-
如果你有逛 GitHub 的习惯,应该就已经发现,许多知名的开源项目的 README 上都有着花花绿绿的各式徽章。比如说这个:
那么这些徽章都是干啥的呢? 玩转 github 徽章 / 自己定制徽章
单纯的链接
使用证书
npm 版本
聊天室 Gitter
Travis 使用.travis.yml 配置文件,告诉 Travis CI 怎样对你的项目进行编译或测试,成功徽章会自动展示 passing
Codacy 编程代码自动审查服务平台,可以帮我们分析存在的问题 或者说是 bug,主要包括代码质量、语法规范、功能可用性方面的检查。
订阅数
... 还有很多,喜欢的自己探索自己探索
常用的几种证书
snpm+gitlab,流程和规范大部分是一样的