1 pnpm

1.1 安装与配置

  • 安装
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
// HomeBrew
$ brew install pnpm
// powershell
$env:PNPM_VERSION='7.0.0-beta.2' ; iwr https://get.pnpm.io/install.ps1 -useb | iex
// npm
$ npm install -g pnpm@next-7
// curl
$ curl -fsSL https://get.pnpm.io/install.sh | PNPM_VERSION=7.0.0-beta.2 sh -
// wget
$ wget -qO- https://get.pnpm.io/install.sh | PNPM_VERSION=7.0.0-beta.2 sh -
  • 配置淘宝源
1
$ pnpm config set registry http://registry.npm.taobao.org

1.2 基本使用

1
2
3
$ pnpm install
$ pnpm add -D/-O xxx
$ pnpm remove xxx

2 Vite

2.1 搭建

1
$ pnpm create vite

根据命令提示操作即可

2.2 配置项目

2.2.1 mobx的安装与基本使用

关于 MobX | MobX 中文文档 | MobX 中文网 (mobxjs.com)

  • 安装依赖
1
$ pnpm install mobx mobx-react
  • 创建stores层文件
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
// stores/countStore.ts
import { makeAutoObservable } from "mobx";

class Count {
  constructor() {
    // makeAutoObservable(target, overrides?, options?) 
    // makeAutoObservable与makeObservable类似,不同点是它会默认推断所有属性,比makeObservable更容易维护。
    // 注意:makeAutoObservable不能用于super或subclassed的类。
    makeAutoObservable(this);
  }
  // 实例属性变为observable
  num = 0;
  // 实例方法变为autoAction
  add = () => {
    this.num += 1;
  };
}
// 导出Count类的实例
export default new Count();
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
// stores/index.tsx
import * as React from "react";
import { configure } from "mobx";
import countStore from "./CountStore";

configure({ enforceActions: "always" });

// 要将每个store整合为一个
export const stores = { countStore };
// 创建上下文对象,将全部store文件存入全局上下文环境中,供各组件使用
export const storeContext = React.createContext(stores);
// 封装useStores hook,提供使用上下文环境的接口
export const useStores = () => React.useContext(storeContext);
  • 使用(mobx异步方法的使用见下文)
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
// App.tsx
import { observer } from "mobx-react";
import { useStores } from "./store";

const App = () => {
  const { countStore } = useStores();
  const { num, add } = countStore;
  return (
    <>
      <h2>count:{num}</h2>
      <button onClick={() => add()}>
        +1
      </button>
    </>
  );
};
// observer函数/装饰器可以用来将React组件转变成响应式组件。
// observer是由单独的mobx-react包提供的。写法有两种,第一种,将组件用observer包裹起来;第二种,对组件加装饰器@observer。
export default observer(App);

2.2.2 配置lessantd支持

  • 安装lessantd与按需加载的插件
1
2
$ pnpm add antd @ant-design/icons
$ pnpm add less vite-plugin-imp -D
  • 修改vite.config.ts
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
import { defineConfig } from "vite";
import react from "@vitejs/plugin-react";
import vitePluginImp from "vite-plugin-imp";

// https://vitejs.dev/config/
export default defineConfig({
  plugins: [
    react(),
    vitePluginImp({
      optimize: true,
      libList: [
        {
          libName: "antd",
          libDirectory: "es",
          style: (name) => `antd/es/${name}/style`,
        },
      ],
    }),
  ],
  css: {
    preprocessorOptions: {
      less: {
        javascriptEnabled: true, // 支持内联 JavaScript
      },
    },
  },
});

2.2.3 二次封装axios

  • 安装
1
$ pnpm add axios
  • 封装(src/api/request.ts)
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
import { notification } from "antd";
import axios, { AxiosInstance, AxiosRequestConfig, AxiosResponse } from "axios";

const codeMessage = {
  200: "服务器成功返回请求的数据。",
  201: "新建或修改数据成功。",
  202: "一个请求已经进入后台排队(异步任务)。",
  204: "删除数据成功。",
  400: "发出的请求有错误,服务器没有进行新建或修改数据的操作。",
  401: "用户没有权限(令牌、用户名、密码错误)。",
  403: "用户得到授权,但是访问是被禁止的。",
  404: "发出的请求针对的是不存在的记录,服务器没有进行操作。",
  406: "请求的格式不可得。",
  410: "请求的资源被永久删除,且不会再得到的。",
  422: "当创建一个对象时,发生一个验证错误。",
  500: "服务器发生错误,请检查服务器。",
  502: "网关错误。",
  503: "服务不可用,服务器暂时过载或维护。",
  504: "网关超时。",
};

class MyAxios {
  instance: AxiosInstance;

  constructor(config: AxiosRequestConfig) {
    this.instance = axios.create(config);
    // 全局请求拦截
    this.instance.interceptors.request.use(
      (config) => config,
      (error) => error
    );
    // 全局响应拦截
    this.instance.interceptors.response.use(
      (res) => res.data,
      (error) => {
        // 在响应拦截器中添加统一的错误处理
        const { response, request } = error;
        if (response && response.status) {
          const errorText = codeMessage[response.status] || response.statusText;
          const { status } = response;
          const { responseURL: url } = request;
          notification.open({
            message: `请求错误 ${status}: ${url}`,
            description: errorText,
          });
        } else if (!response) {
          notification.open({
            message: "网络错误",
            description: "您的网络发生异常,无法连接服务器",
          });
        }
        return response;
      }
    );
  }

  // request作为原型方法继承给实例
  request(config: AxiosRequestConfig): Promise<AxiosResponse> {
    return new Promise<AxiosResponse>((resolve, reject) => {
      this.instance
        .request(config)
        .then((res) => {
          resolve(res);
        })
        .catch((err) => {
          reject(err);
        });
    });
  }
}

const myAxios = new MyAxios({
  // 根据环境变量改变前缀
  baseURL: "/api",
  timeout: 0,
  headers: {
    Authorization: `Bearer d38de888-cb73-4cf1-a335-d3f47312f148`,
  },
});

export default myAxios;
  • 搭配mobx使用
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
// store/AsyncStore.ts
import { makeAutoObservable } from "mobx";
import { getList } from "../api/equipAPI";
class AsyncStore {
  constructor() {
    makeAutoObservable(this);
  }
  equips: any[] = [];
  setEquip(data: any[]) {
    this.equips = data;
  }
  // 异步方法可以使用generator,会被mobx解析为flow类型
  *getEquipList(): any {
    const resp = yield getList();
    this.setEquip(resp.data);
  }
}
export default new AsyncStore();

// store/index.ts中注册asyncStore

// view/page2/index.tsx
import { Button } from "antd";
import { observer } from "mobx-react";
import { useStores } from "../../store";

const Page2 = () => {
  const { asyncStore } = useStores();
  // 异步方法不能在store中解构取出
  const { equips } = asyncStore;
  return (
    <>
      <Button
        onClick={() => {
          asyncStore.getEquipList();
        }}
      >
        请求
      </Button>
      <div>{equips.toString()}</div>
    </>
  );
};
export default observer(Page2);

2.2.4 开发环境proxy代理

  • vite.config.ts
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
export default defineConfig({
  server: {
    proxy: {
      "/api": {
        target: "http://10.30.20.203:8080",
        // changeOrigin: true,
        rewrite: (path) => path.replace(/^\/api/, ""),
      },
    },
  },
  ...
}

2.2.5 路由配置

  • 安装react-router-dom
1
$ pnpm add react-router-dom
  • src下创建路由配置文件src/config/routes,利用react路由的懒加载——即对组件进行分割打包成多个chunk来减少一次性加载的资源大小,从而加快首屏渲染速度,提升用户体验
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
import { lazy } from "react";

const routes = [
  {
    path: "/",
    exact: true,
    component: lazy(() => import("../view/home")),
    name: "Home",
  },
  {
    path: "/home",
    component: lazy(() => import("../view/home")),
    name: "Home",
  },
  {
    path: "/page1",
    component: lazy(() => import("../view/page1")),
    name: "Page1",
  },
  ...
];
export default routes;
  • 封装router.tsx作为路由文件入口。由于路由是以懒加载的形式渲染的,所以切换页面时可能会产生延迟,因此使用Suspense组件将路由组件包裹,并在fallback中声明懒加载组件加载完成前做的事,优化整个页面的交互
  • 注意Route中的component属性改为了element
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
import { Suspense } from "react";
import { Route, Routes } from "react-router-dom";
import routes from "./config/routes";

const Router = () => {
  const myRoutes = routes.map((item) => {
    return (
      <Route key={item.path} path={item.path} element={<item.component />} />
    );
  });
  return (
    <Suspense fallback={<div>Loading...</div>}>
      <Routes>{myRoutes}</Routes>
    </Suspense>
  );
};

export default Router;
  • App.tsx,注意使用HashRouterBrowserRouter组件包裹Router
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
import Router from "./router";
import { HashRouter, Link } from "react-router-dom";

function App() {
  return (
    <div className="App">
      <HashRouter>
        <Link to="/">Home Page</Link>
        <Link to="/page1">Page1</Link>
        <Link to="/page2">Page2</Link>
        <Router />
      </HashRouter>
    </div>
  );
}

export default App;

2.2.5 自定义字体配置

  • 将字体文件放在assets/fonts

  • index.html中添加style标签,注册自定义字体族

1
2
3
4
5
6
7
8
9
<style>
    @font-face {
      font-family: CustomFont;
      src: local("CustomFont"),
        url(./assets/fonts/Montserrat-Regular.ttf) format("truetype");
      font-weight: normal;
      font-style: normal;
    }
</style>

2.2.6 eslint与prettier配置

  • 安装eslint与prettier
1
$ pnpm add -D @typescript-eslint/eslint-plugin eslint eslint-plugin-react @typescript-eslint/parser prettier 
  • .eslintrc.js
  1
  2
  3
  4
  5
  6
  7
  8
  9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
 31
 32
 33
 34
 35
 36
 37
 38
 39
 40
 41
 42
 43
 44
 45
 46
 47
 48
 49
 50
 51
 52
 53
 54
 55
 56
 57
 58
 59
 60
 61
 62
 63
 64
 65
 66
 67
 68
 69
 70
 71
 72
 73
 74
 75
 76
 77
 78
 79
 80
 81
 82
 83
 84
 85
 86
 87
 88
 89
 90
 91
 92
 93
 94
 95
 96
 97
 98
 99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
module.exports = {
  // react17以上的版本,使用plugin:react/jsx-runtime
  extends: ['eslint:recommended', 'plugin:react/jsx-runtime'],
  // 代码运行环境
  env: {
    browser: true,
    commonjs: true,
    es6: true,
  },
  parser: '@typescript-eslint/parser',
  parserOptions: {
    ecmaFeatures: {
      jsx: true,
      modules: true,
    },
    sourceType: 'module',
    ecmaVersion: 6,
  },
  plugins: ['react', '@typescript-eslint'],
  settings: {
    'import/resolver': {
      node: {
        extensions: ['.tsx', '.ts', '.js', '.json'],
        alias: [['@', './src']],
      },
    },
  },
  rules: {
    /*
     * 建议规则
     */
    // 必须使用全等
    eqeqeq: 2,
    // 不建议使用consolelog
    'no-console': 1,
    // 不建议使用alert
    'no-alert': 2,
    // 强制对多行注释使用特定风格,块注释或行注释
    // 'multiline-comment-style': 2,
    // 不建议使用不必要的分号
    'no-extra-semi': 2,
    // 不建议使用短符号进行类型转换(禁用隐式转换)
    'no-implicit-coercion': 2,
    // 不建议注释与代码在同一行
    'no-inline-comments': 2,
    // 不建议在三元操作数中间换行
    // 'no-nested-ternary': 2,
    // 不建议将变量初始化为undefined
    'no-undef-init': 2,
    // 不建议在有更简单的可替代的表达式时(||)使用三元操作符
    'no-unneeded-ternary': 2,
    // 不建议变量定义却不使用
    'no-unused-expressions': 2,
    // 不建议使用var
    'no-var': 2,
    // 推荐对象字面量简写语法
    'object-shorthand': 1,
    // 推荐回调函数使用箭头函数形式
    'prefer-arrow-callback': 1,
    // 优先使用对象与数组的解构语法
    'prefer-destructuring': 1,
    // 建议使用模板字符串而非字符串拼接
    'prefer-template': 2,
    // 在注释内容前建议添加空格
    'spaced-comment': 1,

    /*
     * 布局风格
     */
    // 在数组开括号后和闭括号前强制换行
    'array-bracket-newline': ['error', { multiline: true }],
    // 建议数组方括号中使用一致的空格?
    'array-bracket-spacing': 0,
    // 建议多项数据的最后一项保留逗号
    'comma-dangle': ['error', 'always-multiline'],
    // 建议逗号前添加空格
    'comma-spacing': ['error', { before: false, after: true }],
    // 箭头函数前后应有空格
    'arrow-spacing': 2,
    // 强制使用一致的空格,默认逗号放在数组元素、对象属性或变量声明之后
    'comma-style': 2,
    // 不建议在计算属性(obj[foo] || { [foo]: 1 })中使用空格
    'computed-property-spacing': 2,
    // 强制在点号之前或之后换行
    // 'dot-location': ['error', 'object'],
    // 函数调用时,名称与括号之间不建议存在空格
    'func-call-spacing': 2,
    // 在迭代器函数名与*号之间是否存在空格
    /* 'generator-star-spacing': ['error', 'before'], */
    // 缩进为2
    indent: ['error', 2, { SwitchCase: 1 }],
    // jsx元素中的属性使用双引号
    'jsx-quotes': ['error', 'prefer-double'],
    // 对象的key和value之间要有空格
    'key-spacing': ['error', { beforeColon: false, afterColon: true }],
    // 强制在关键字(else this function)前后使用一致的空格
    'keyword-spacing': ['error', { before: true }],
    // 允许注释周围存在空行
    'lines-around-comment': [
      'error',
      // 在对象字面量的开始位置、在块语句的结束位置允许注释
      { allowObjectStart: true, allowBlockStart: true },
    ],
    // 要求调用无参构造函数时带括号
    'new-parens': 'error',
    // 要求方法链中每个调用都有一个换行符
    'newline-per-chained-call': 'error',
    // 空格只允许一个
    'no-multi-spaces': 'error',
    // 不允许出现多个空行
    'no-multiple-empty-lines': 'error',
    // 行尾禁止出现空格
    'no-trailing-spaces': 'error',
    // 属性前(foo. bar)不允许出现空格
    'no-whitespace-before-property': 'error',
    // enforce consistent line breaks after opening and before closing braces
    // 'object-curly-newline': ['error', { multiline: true }],
    // 花括号前后要有空格
    'object-curly-spacing': ['error', 'always'],
    // 当代码块只有一条语句时,省略花括号
    curly: ['error', 'multi-or-nest'],
    // 换行时换行符放在操作符后面(即操作符后换行)
    'operator-linebreak': ['error', 'after'],
    // 建议对象的属性在各自的独立行
    'object-property-newline': [
      'error',
      { allowAllPropertiesOnSameLine: true },
    ],
    // 在剩余和扩展运算符及其表达式之间不需要空格(... a)
    'rest-spread-spacing': ['error', 'never'],
    // 分号前无空格,分号后有空格(for循环)
    'semi-spacing': ['error', { before: false, after: true }],
    // 强制分号出现在行尾
    'semi-style': ['error', 'last'],
    // 要求块语句之前有空格(func() {},try {},if() {})
    'space-before-blocks': 'error',
    // 函数圆括号之前是否存在空格
    'space-before-function-paren': [
      'error',
      { anonymous: 'always', named: 'never', asyncArrow: 'always' },
    ],
    // 禁止圆括号内出现空格
    'space-in-parens': ['error', 'never'],
    // 要求操作符周围有空格(1 === 1)
    'space-infix-ops': 'error',
    // 要求一元操作符(new delete typeof ! ++)在之前或之后存在空格
    'space-unary-ops': [2, { words: true }],
    // 要求包裹正则表达式
    'wrap-regex': 2,
    // Enforce spacing around the * in yield* expressions
    /* 'yield-star-spacing': ['error', 'before'], */
  },
}
  • .prettierrc.js
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
module.exports = {
  // 对象花括号内部前后有空格
  bracketSpacing: true,
  // 缩进 2
  tabWidth: 2,
  // 句尾分号
  semi: false,
  // 使用单引号
  singleQuote: true,
  // 最后一项后的逗号
  trailingComma: 'all',
  // 元素的‘>’单独一行
  bracketSameLine: false,
  arrowParens: 'avoid',
  // jsx中使用双引号
  jsxSingleQuote: false,
  stylelintIntegration: true,
  eslintIntegration: true,
}

2.2.7 husky配置github提交代码规范

1
2
3
4
5
6
7
$ pnpm add husky -D
# 初始化husky,将git hooks交由husky执行
$ npx husky install
husky - Git hooks installed
# 在package.json添加命令
$ npm set-script prepare "husky install"
# 会在每次pnpm install时自动执行,神奇呀!
1
2
3
4
5
6
# 自动格式化lint-staged
$ pnpm add lint-staged -D
$ npx husky add .husky/pre-commit "npx lint-staged"
husky - created .husky/pre-commit
# 提交一下
$ git add .husky/pre-commit
  • 创建.lintstagedrc.json
1
2
3
4
5
6
{
	"*.{js,jsx,ts,tsx}": ["npx prettier --write", "npx eslint --fix"],
  // 暂无stylelint
	"*.{css,less,scss}": ["npx prettier --write", "npx stylelint --fix"],
	"*.{json,md}": ["npx prettier --write"]
}
1
2
3
4
5
6
# 安装anglar的提交规范
$ pnpm add @commitlint/config-conventional @commitlint/cli -D
# 配置commitlint去使用conventional的配置
$ echo "module.exports = { extends: ['@commitlint/config-conventional'] };" > commitlint.config.js
# 添加commit-msg钩子,会在pre-commit之后运行,检查commit message的内容
$ npx husky add .husky/commit-msg 'npx --no -- commitlint --edit $1'

2.2.8 vite多环境构建配置

3 布局