pnpm+monorepo+changesets+dumi 构建React组件库
前言
搭建组件库,是每个前端或早或晚需要接触的内容。在此之前,我都是使用lerna来管理多包项目。但了解到pnpm workspace之后,我发现使用pnpm+monorepo+changesets方式来管理多包项目更为方便。
组件库除了第三方包的开发外,还需提供组件库的使用文档,这才是一个完整的组件库项目。因此这里我选择使用dumi搭建React的组件库
1. 初始化
新建一个文件夹并运行
npx create-dumi
有三个模块可供选择,这里选择第二项react Library
Static Site # 用于构建网站
React Library # 用于构建组件库,有组件例子
Theme Package # 主题包开发脚手架,用于开发主题包
2. 定义pnpm工作空间
新建pnpm-workspace.yaml
packages:
- 'packages/**'
- '!**/test/**'
- '!**/demos/**'
- '!**/es/**'
- '!**/*.md'
3. monorepo项目目录结构调整
1.删除根目录下src,新建packages目录
2.在packages文件夹下新建components和hooks,分别存放组件和hook,目录结构为
└── components
├── src
├── .fatherrc.ts
└── package.json
└── README.md
└── hooks
├── src
├── .fatherrc.ts
└── package.json
└── README.md
.fatherrc.ts
import { defineConfig } from 'father';
export default defineConfig({
platform: 'browser',
esm: {
transformer: 'babel',
},
});
package.json
{
"name": "@wjcao/components",
"version": "1.0.0",
"private": false,
"description": "A react library developed with dumi",
"homepage": "https://github.com/cwjbjy/components#readme",
"bugs": {
"url": "https://github.com/cwjbjy/components/issues"
},
"repository": {
"type": "git",
"url": "https://github.com/cwjbjy/components"
},
"license": "MIT",
"author": "cwj <595842570@qq.com>",
"main": "./dist/esm/index.js",
"module": "./dist/esm/index.js",
"typings": "./dist/esm/index.d.ts",
"files": [
"dist"
],
"scripts": {
"build": "father build"
},
"dependencies": {
"styled-components": "^6.1.11"
},
"devDependencies": {
"@types/styled-components": "^5.1.34",
"antd": "^5.19.0"
},
"peerDependencies": {
"antd": ">=5.11.5",
"react": ">=16.9.0",
"react-dom": ">=16.9.0"
},
"publishConfig": {
"access": "public"
}
}
private,publishConfig:设置为非私有包,面向公众(私有包npm会收费)
main,module:指向打包后的入口文件
typings:指向打包后的TS声明文件
files:需要上传npm的文件夹(package.json与README.md会默认上传)
peerDependencies:平级依赖,即使用该包不需要额外下载react,使用项目本身的react包即可
4. 修改全局样式
dumi的默认样式可能不满足需求,可在.dumi下新建global.less
::-webkit-scrollbar {
width: 10px;
height: 10px;
&-thumb {
background-color: rgba(0, 0, 0, 0.12);
border: 2px solid transparent;
border-radius: 8px;
background-clip: padding-box;
&:hover {
background-color: rgba(0, 0, 0, 0.24);
}
}
&-track {
background-color: transparent;
}
}
.dumi-default-doc-layout {
.dumi-default-header-content {
max-width: 2000px;
}
> .dumi-default-doc-layout-mobile-bar + main {
max-width: 2000px;
// margin: unset;
.markdown {
font-size: 15px;
}
}
.dumi-default-sidebar {
top: 76px;
width: 256px;
overflow: overlay;
dl > dd {
padding: 6px 0;
& > a {
font-size: 15px;
}
}
}
}
5. 配置dumi
dumi默认为单项目管理。多项目管理需额外配置。
修改.dumirc.ts
import { defineConfig } from 'dumi';
import { join } from 'path';
const basePath = '/docs/';
export default defineConfig({
themeConfig: {
name: '前端文档',
//头部菜单栏
nav: [
{
title: '规范',
link: '/spec',
},
{
title: 'components',
link: '/components',
},
{
title: 'hooks',
link: '/hooks',
},
],
prefersColor: { default: 'light', switch: true }, //主题色
socialLinks: {
github: 'https://github.com/cwjbjy?tab=repositories',
},
},
outputPath: 'docs-dist', //打包后文档的包名
base: basePath,
publicPath: basePath,
hash: true, //文档包名是否生成hash,防止浏览器缓存
//解析目录
resolve: {
docDirs: ['docs'], //配置 Markdown 文档的解析目录
atomDirs: [
//配置子项目(例如组件、函数、工具等)Markdown 的解析目录。
{ type: 'components', dir: 'packages/components/src' },
{ type: 'hooks', dir: 'packages/hooks/src' },
// { type: 'tools', dir: 'packages/tools/src' },
],
},
//别名:dumi 2不再感知 monorepo,需要手动配置包名到 src 的 alias。
alias: {
'@wjcao/components': join(__dirname, 'packages/components/src'),
'@wjcao/hooks': join(__dirname, 'packages/hooks/src'),
// '@wjcao/utils': join(__dirname, 'packages/tools/src'),
},
});
6. 配置Eslint
1. 在根目录安装
pnpm add -D eslint-plugin-import -w
2. 修改.eslintrc.js
module.exports = {
extends: require.resolve('@umijs/lint/dist/config/eslint'),
plugins: ['react-hooks', 'import'],
ignorePatterns: ['dist', 'node_modules', '.eslintrc.js'],
rules: {
'react-hooks/rules-of-hooks': 'error',
'react-hooks/exhaustive-deps': 'warn',
'@typescript-eslint/no-unused-vars': 'warn',
'@typescript-eslint/no-unused-expressions': 'off',
'@typescript-eslint/no-use-before-define': 'off',
'@typescript-eslint/ban-types': 'warn',
'react/no-array-index-key': 'off',
'@typescript-eslint/no-shadow': 'off',
'@typescript-eslint/consistent-type-imports': 'off',
'@typescript-eslint/method-signature-style': 'off',
'@typescript-eslint/no-invalid-this': 'off',
'import/no-duplicates': 'error',
'no-redeclare': 'off',
'no-console': [
'warn',
{
allow: ['warn', 'error'],
},
],
'no-debugger': 'warn',
'import/order': [
'error',
{
groups: [
'builtin',
'external',
'internal',
['parent', 'sibling', 'index'],
'object',
'type',
'unknown',
],
pathGroups: [
{
pattern: 'react*',
group: 'builtin',
patternOptions: {
noComment: false,
},
position: 'before',
},
{
pattern: '*/**/*.less',
group: 'type',
position: 'after',
},
{
pattern: '*/**/*.css',
group: 'type',
position: 'after',
},
{
pattern: '@components/**',
group: 'external',
position: 'after',
},
{
pattern: '@hooks/**',
group: 'external',
position: 'after',
},
],
pathGroupsExcludedImportTypes: [],
'newlines-between': 'always',
alphabetize: {
order: 'asc',
caseInsensitive: true,
},
},
],
},
overrides: [
{
files: ['*.ts', '*.tsx'],
rules: {
'no-undef': 'off',
},
},
],
};
7. 配置.gitignore
node_modules
dist
docs-dist
.dumi/tmp
.dumi/tmp-test
.dumi/tmp-production
.DS_Store
8. 配置tsconfig.json
{
"compilerOptions": {
"target": "esnext",
"module": "esnext",
"moduleResolution": "node",
"jsx": "react-jsx",
"esModuleInterop": true,
"strict": true,
"skipLibCheck": true,
"stripInternal": true,
"forceConsistentCasingInFileNames": true,
"resolveJsonModule": true,
"declaration": true,
"baseUrl": ".",
"paths": {
"@@/*": [".dumi/tmp/*"],
"@wjcao/components/*": ["./packages/components/src/*"],
"@wjcao/hooks/*": ["./packages/hooks/src/*"]
}
},
"include": [".dumirc.ts", "packages/**/*"],
"exclude": ["node_modules", "dist", "docs-dist"]
}
9. 打docs包
运行 pnpm run docs:build,会生成docs-dist文件
运行 pnpm run docs:preview,可预览
10. 打包
1. 全部打包
pnpm -r build
2. 部分打包
pnpm run build -F @wjcao/components
-F 后面的参数为子包package.json的name值
11. 配置changesets
pnpm+monorepo负责管理包的依赖,changesets 负责管理包的version和生成changelog
1. 安装
pnpm add -D @changesets/cli -w
2. 初始化
pnpm changeset init
只需要将access的值restricted,改为public(开源)
config.json其他配置:
指令 | 功能 |
---|---|
commit | 不要让 changeset 在 publish 的时候帮我们做 git add |
linked | 配置哪些包要共享版本 |
access | 公私有安全设定,内网建议 restricted ,开源使用 public |
baseBranch | 项目主分支 |
updateInternalDependencies | 确保某包依赖的包发生 upgrade,该包也要发生 version upgrade 的衡量单位(量级) |
ignore | 不需要变动 version 的包 |
3. 生成变更记录
pnpm changeset
在生成时,会提示是否生成 major 模式,即大版本变更1.2.3的1;不选择,直接回车,会提示是否生成minor模式,即中版本变更1.2.3的2;不选中直接回车,默认选则patch模式,即小版本变更1.2.3的3
4. 消耗变更记录并更新版本
pnpm changeset version
5. git记录中会生成一个tag
6. 推送tag
git push origin @wjcao/components@1.0.1
11. 发包
- 首先注意components/package.json中是否有公共包的配置
"private": false,
"publishConfig": {
"access": "public"
},
-
提交git,并保存
-
如果以前改过npm的镜像地址,比如使用了淘宝的镜像,就先改回来
npm config set registry=<https://registry.npmjs.org>
-
登录npm:
npm login
(如果没有首先注册,并且验证邮箱) -
访问npm:www.npmjs.com/org/create 创建命名空间,例如:wjcao
-
运行:
pnpm changeset publish
-
在其他react项目中安装:
pnpm add @wjcao/components
12. 发布npm包时可能遇到的报错信息与解决方案
邮箱未验证
npm ERR! publish Failed PUT 403 npm ERR! code E403 npm ERR! you must verify your email before
publishing a new package: <https://www.npmjs.com/email-edit> : your-package
去邮箱验证
没有权限发布
npm ERR! publish Failed PUT 403 npm ERR! code E403 npm ERR! You do not have permission to
publish "your-package". Are you logged in as the correct user? : your-package
包和别人的包重名了。修改包名
需要登录
npm ERR! code ENEEDAUTH npm ERR! need auth auth required for publishing npm ERR! need auth You need to authorize this machine using `npm adduser`
使用 npm login 登录
只有管理员才有权限发布
npm ERR! publish Failed PUT 403 npm ERR! code E403 npm ERR! \[no\_perms] Private mode enable,
only admin can publish this module \[no\_perms] Private mode enable, only admin can publish
this module: your-package
源设置成第三方源,比如设置了淘宝镜像。只要把源设为默认的即可
npm config set registry registry.npmjs.org
包名过于类似
npm ERR! publish Failed PUT 403 npm ERR! code E403 npm ERR! Package name too similar to
existing packages; try renaming your package to '@hopgoldy/auto-git' and publishing with 'npm
publish --access=public' instead : your-package
无法发布私有包
npm ERR! publish Failed PUT 402 npm ERR! code E402 npm ERR! You must sign up for private
packages :
这个当你的包名为@your-name/your-package时才会出现,原因是当包名以@your-name开头时,npm publish会默认发布为私有包,但是 npm 的私有包需要付费,所以需要添加如下参数进行发布:
npm publish --access public
或者在package.json中添加
"publishConfig": {
"access": "public"
},
包的命名空间
lerna ERR! E404 Scope not found
点击链接创建包的命名空间:www.npmjs.com/org/create
结尾
本篇只提供了配置过程,如果对配置不清楚,或有问题的,可在评论区留言
具体的React库开发如果有需要,也可在评论区留言,如果看到有需求的小伙伴,后续我会继续更相关文章
转载自:https://juejin.cn/post/7387610960159129612