3

最近,在维护一个 RN 项目时,发现存在一些问题。当我们开始使用 RN 之前,肯定会做一些技术调研,比如技术难度、社区活跃度、小伙伴们的了解程度等,其中不可忽视的是有无现成成熟的 组件库 使用,这将使项目开发周期和效率有所提高。

当我们做好技术选型,选定好 组件库 后,随着项目的开发和不断迭代,可能会出现这么一些问题:

  1. 只使用了 组件库 中几个组件而已;
  2. 组件库中的组件需要做一些修改,但是修改起来却比较麻烦;
  3. 往往自定义的组件已经有十几个了;

那么这个时候,可以自定义组件,将第三方组件库替换掉,提取自己的组件发布到 npm 上。今天,我们就来聊聊如何 搭建RN组件库。主要还是学习如何开发一个组件库这么一个过程。过程很重要

提前考虑的问题

当我们决定开发一个 RN组件库 的时候,有几个问题需要提前考虑下:

  1. 为什么要做组件库?
  2. RN组件库目录结构大概长啥样?
  3. 如何开发、预览、验证我们写的组件?
  4. 如何发布组件库?

问题一:为什么要做组件库

除了开头说的三点原因外,我觉得最主要的原因在于一个有趣的现象:明明设计师有一份设计规范,但对于同一组件,不同开发人员依然会要重新写一遍。原因就在于没有开发将设计规范开发落地。设计规范是一张图纸,它不是组件,应该有实打实的代码落地,这样设计规范才能发挥它最大的作用。

问题二:RN组件库目录结构长啥样

因为,我们自定义的组件基本都是纯 js 的文件,很少有涉及到原生的功能(主要是不会0_0)。

因此,我们的目录组织结构,只需要有个 components 文件夹、一个入口文件index.js(将组件统一导出)、一个package.json文件,就可以了。当然,如果你有 icon assets 或者一些其他的目录也能添加进来。

这里需要多说一点,我们只用关心最终上传 npm 的目录,而不用关心与预览相关的目录结构。比如 有两种方式,一种是当做一个RN项目开发,通过 npx react-native init xxx 来初始化项目。一种是 expo,不用搭建 RN 环境,可以参考我写的这篇Expo 搭建 React-native 项目

我用的是第一种,所以 我的目录大致如下:

.
├── README.md
├── app.json
├── android/
├── ios/
├── gulpfile.js
├── index.js
├── package.json
├── dist
│   ├── README.MD
│   ├── index.js
│   ├── package.json
│   └── src
│       ├── components
│       ├── icons
│       ├── themes
│       └── utils
└── src
    ├── App.js
    ├── component-path
    ├── components
    ├── icons
    ├── routes
    ├── themes
    ├── utils
    └── views

dist 目录 就是我最终用来发布 npm 的目录。dist/components 下的组件,是从 src/components 拷贝而来,其他类似目录,同理。至于其他文件和目录怎么来的,后面会讲到。

问题三:如何开发、预览、验证我们写的组件

在开发组件的过程中,需要实时预览组件的效果,那么组件在预览页面的引入方式可能是这样的:

// 预览页面可能位于 src/views/ButtonDemo.js
import { Button } from  '../components'

组件开发完,发布后,想要看线上组件的实际效果,引入可能会变成这样:

import { Button } from  'react-native-unit-zjp'

如果,你开发了有一些组件,有一些预览页面,那么,你在开发过程 和 发布后,为了预览组件,可能需要频繁修改组件路径。而且,还有一个问题,就是在开发完组件,预发布的时候,无法提前预览即将发到线上的组件效果。(我碰到过一个问题,就是本地开发,引入组件看效果的时候,没有任何问题;发布到线上,引入的时候,会报警告)

其实,我们可以有一个统一修改组件引入路径的地方,解决这些问题。

src/component-path 下 添加 index.js 文件:

  

export * from '../components'; // 本地调试
// export * from '../../dist'; // 发布前测试包
// export * from '../../node_modules/react-native-unit-zjp'; // npm 线上包

有了这个开关文件后,预览页面的组件就可以这么引入了:

import { Button } from '../component-path'

在组件开发,预发布,发布后 这几个阶段,只需要切换这一个路径就够了。

上面文件中之所以 要加 '../../node_modules',是因为当 dist 目录下 的 package.json 存在时,直接从 react-native-unit-zjp 引入组件,会指向 dist 目录。

从这点可以得出,如果想要以特定名称引入某文件时,不想写长长的路径的话,可以在该目录下,新建一个 pacakage.json 然后指定它的 name 就可以了

问题四:如何发布组件库

发布 npm 包,其实很简单,就一句命令行的事 npm publish,当然,得从拥有一个 npm 账号开始。这里就不细说了,可以看看官网,或者网上其他教程,很详细,一分钟教你发布npm包。这里只讨论,基于这个项目,怎么发布。

基于上面的思考,dist 成为了我们的组件库发布目录了。这里之所以,单独弄个 dist 目录,只是为了将组件库本身所用到的依赖,与整个项目所用到的依赖区分开。如果,不想这么考虑,完全可以不用 dist 目录,直接在 package.jsonfiles 字段配置需要发布的文件 及 目录。

如果想要 dist 目录,应该有以下这样的流程:

  1. 提交代码至 git,然后拷贝相关目录 到 dist(可用 gulp,下面会讲)
  2. cd dist
  3. npm version xxx
  4. npm publish

关于 npm 中的版本号,这里多说一点。一般使用三位数来描述,以点来分割,例如:1.0.0

  • 主版本号:当你做了不兼容的 API 修改
  • 次版本号:当你做了向下兼容的功能性新增
  • 修订号:当你做了向下兼容的问题修正

通过执行,npm version xxx 来自动更新版本号。(需要将改动提交至 git ,然后再执行此命令。此命令,会自动打上 tag,并提交,需要手动 git push )

  • patch 1.0.0 => 1.0.1
  • minor 1.0.0 => 1.1.0
  • major 1.0.0 => 2.0.0

版本号更新好之后,就可以 npm publish 了。

注意: 首次 publish 只要有个版本号就可以,再次 publish 之前,必须更新版本号,也就是执行 npm version xxx 命令,否者会报错:

You cannot publish over the previously published versions: x.x.x;

解决这四个问题后,我们可以开始创建项目了。

创建 RN组件库项目

上面提到过,我们只关心用于发布的 dist 目录,因此,怎么创建项目,就看个人喜好。如果有现成 RN 开发环境,可以直接初始化一个 RN 项目。如果不想费力搭建 RN 开发环境,可以试试 expo。这里,我就用第一种了。

1、初始化 react-native 项目

npx react-native init xxx

2、开发组件

src/components 目录下开发你的组件。怎么开发组件,这里就不讲了,就当是一个 RN 项目来开发,该建啥目录,缺啥引啥,router 啥的。

组件开发完后,在 src/components 最好有个 导出组件的文件 index.js:

import Button from './Button/Button.js';
export { Button }

3、添加发布文件夹 dist

为了目录清晰,以及有个 单独的 package.json 管理 发布包的版本,因此,决定单独弄个 dist 文件夹。将需要发布的文件及目录拷贝至 dist 目录中。

dist 目录 添加 package.json,执行 npm init 得到如下文件:

{
  "name": "react-native-unit-zjp",
  "version": "0.0.4",
  "description": "## 一套拿来就用的 ReactNative 组件库",
  "main": "index.js",
  "scripts": {
    "test": "echo \\"Error: no test specified\\" && exit 1"
  },
  "keywords": [
    "react-native"
  ],
  "author": "zhangjinpei",
  "license": "ISC",
  "dependencies": {
    "react-native-linear-gradient": "^2.5.6",
    "react-native-root-siblings": "^4.0.6"
  },
  "devDependencies": {},
  "repository": {
    "type": "git",
    "url": "xxx"
  },
  "bugs": {
   "url": "xxx"
  },
  "homepage": "xxx",
  "files": [
    "index.js",
    "README.md",
    "src"
  ]
}

4、拷贝组件到 dist/src 下

为了保持他们之间的引用路径,保持 dist 目录结构,跟 src 目录相似。

为了方便,这里通过 gulp 拷贝文件,配置如下:

const gulp = require('gulp');
const rimraf = require('rimraf');
const { src, dest, task, series} = gulp;

task('clean', (cb) => {
  rimraf('dist/src', cb);
});

task('components', () => {
  return src('src/components/**/*.*')
    .pipe(dest('dist/src/components/'));
});

task('icons', () => {
  return src('src/icons/**/*.*')
    .pipe(dest('dist/src/icons/'));
});

task('themes', () => {
  return src('src/themes/**/*.*')
    .pipe(dest('dist/src/themes/'));
});

task('utils', () => {
  return src('src/utils/**/*.*')
    .pipe(dest('dist/src/utils/'));
});

task('readme', () => {
  return src('./README.md')
    .pipe(dest('dist/'));
});

exports.default = series('clean', 'components', 'icons', 'themes', 'utils', 'readme');

5、导出组件

现在组件有了,还需要有个统一的地方导出组件。在 `dist` 目录添加入口文件,内容如下:

import Theme from './src/themes/Theme'; // 这个是主题配置 如不需要 可以去掉
import * as myUnit from './src/components';

module.exports = {
  Theme,
  ...myUnit
};

这个时候,组件库的架子基本就差不多了,剩下的就是慢慢完善你的组件库了。

这是我的项目 git 项目地址react-native-unit-zjp,或者 在 npm 中搜索 react-native-unit-zjp 目前还在开发中,欢迎提出问题并star。


zhangjinpei
103 声望6 粉丝

做一枚精致的前端er