构建配置
配置文件
若希望对 ICE PKG 的能力进行配置,推荐在项目根目录中添加名为 build.config.mts
的配置文件:
import { defineConfig } from '@ice/pkg';
// 使用 defineConfig 工具函数以获得更好的类型提示
export default defineConfig({
// 配置选项
});
注:ICE PKG 支持的配置文件类型包括:
build.config.mts
build.config.mjs
build.config.ts
build.config.js
build.json
完整配置项
entry
- 类型:
string | string[] | { [entryAlias: string]: string }
- 默认值:
'./src/index'
指定构建入口。支持配置单入口或者多个入口。
指定单个入口:
import { defineConfig } from '@ice/pkg';
export default defineConfig({
entry: './src/index',
});
指定多个入口:
import { defineConfig } from '@ice/pkg';
export default defineConfig({
// 数组形式
entry: ['./src/foo', './src/bar'],
// 对象形式,key 值作为 chunk name
entry: {
foo: './src/foo',
bar2: './src/bar'
}
});
alias
- 类型:
Record<string, string>
- 默认值:
{}
比如,将 @
指向 ./src
目录。
import { defineConfig } from '@ice/pkg';
export default defineConfig({
alias: {
'@': './src',
},
});
然后代码里 import '@/foo'
会被改成 import '/path/to/your/project/foo'
。
define
- 类型:
Record<string, string | boolean | number | object | null>
- 默认值:
{ __DEV__: 'true' | 'false', 'process.env.NODE_ENV': '"development"' | '"production"' }
定义编译时环境变量,会在编译时被替换。注意:属性值会经过一次 JSON.stringify()
转换。
例如,希望在代码中注入版本号,用全局变量 __VERSION__
来替代:
import pkg from './package.json' assert { type: 'json' };
import { defineConfig } from '@ice/pkg';
export default defineConfig({
define: {
'__VERSION__': pkg.version,
},
});
在编译时,所有 __VERSION__
都会被替换为项目的版本号。
在 TS 项目中,需要在 typings.d.ts
或其他类型声明文件中,声明 define
所设置的属性,以便通过类型检查,并获得类型提示。比如:
declare const __VERSION__: string
ICE PKG 默认注入了 __DEV__
全局变量,用于标识开发态环境。这个变量在输出一些仅在 development 环境的信息时非常有用。比如,输出在用户开发态才显示的警告信息。
if (__DEV__) {
console.warn('请注意,这可能会产生错误!');
}
实际上,在编译时,__DEV__
会被替换为 process.env.NODE_ENV !== 'production'
。
sourceMaps
- 类型:
boolean | 'inline'
- 默认值:start 阶段默认为
true
,build 阶段默认为false
是否生成 sourcemap,这在代码调试的时候非常有用。
import { defineConfig } from '@ice/pkg';
export default defineConfig({
sourceMaps: true,
});
这会为所有产物额外输出 .js.map
文件。如果你想要 sourcemap 是内联在源码中的,可将选项配置为 inline
:
import { defineConfig } from '@ice/pkg';
export default defineConfig({
sourceMaps: 'inline',
});
jsxRuntime
- 类型:
'automatic' | 'classic'
- 默认值:
'automatic'
设置 JSX 转换的方式,并交给编译工具(SWC)编译处理 JSX 语法。
假设有这样一段 JSX 代码:
import React from 'react';
function App() {
return <h1>Hello World</h1>;
}
当 jsxRuntime
的值是 automatic
,编译结果是:
import {jsx as _jsx} from 'react/jsx-runtime';
function App() {
return _jsx('h1', { children: 'Hello world' });
}
当 jsxRuntime
的值是 classic
,编译结果是:
import React from 'react';
function App() {
return React.createElement('h1', null, 'Hello world');
}
generateTypesForJs
- 类型:
boolean
- 默认值:
false
为 JavaScript 代码生成类型文件。ICE PKG 默认为文件后缀为 .ts
生成类型文件。
如果使用 JSDoc 为 JavaScript 生成了类型注解,该配置会非常有效。详细介绍 工程能力 - 生成类型文件。
plugins
- 类型:
Array<string | [string, any?]>
- 默认值:
[]
ICE PKG 基于 build-scripts 插件系统,配置额外的 ICE PKG 插件,以进行更深度的工程定制。更多内容请参考插件开发。
import { defineConfig } from '@ice/pkg';
export default defineConfig({
plugins: [
// npm 依赖
'@ice/plugin-docusaurus',
// 相对路径
'./customPlugin.mjs',
// 指定插件选项
['@ice/plugin-docusaurus', { title: 'Hello World' }]
],
});
transform
Transform 模式是 ICE PKG 默认的编译模式。
该字段定义 Transform 模式 下额外的配置。默认配置是:
import { defineConfig } from '@ice/pkg';
export default defineConfig({
transform: {
formats: ['esm', 'es2017']
}
});
formats
- 类型:
Array<'cjs' | 'esm' | 'es2017'>
- 默认值:
['esm', 'es2017']
输出的类型。ICE PKG 会默认把产物输出到 esm
(输出 ES module + ES5 产物) 和 es2017
(输出 ES module + ES2017 产物) 两个文件夹。
- esm # ES module + ES5 产物
- es2017 # ES module + ES2017 产物
若想要输出 CommonJS 产物,可如下配置:
import { defineConfig } from '@ice/pkg';
export default defineConfig({
transform: {
formats: ['cjs', 'esm', 'es2017'],
},
});
则输出如下文件夹:
- cjs # CommonJS + ES5 产物
- esm # ES module + ES5 产物
- es2017 # ES module + ES2017 产物
excludes
- 类型:
string | string[]
- 默认值:
undefined
排除无需编译的文件。比如,我们不想编译 src
下的所有测试文件,其中测试文件包含在 __tests__
目录下,或以 *.test.[j|t]s
结尾。
import { defineConfig } from '@ice/pkg';
export default defineConfig({
transfrom: {
excludes: ['**/__tests__/**', '*.test.[j|t]s'],
},
});
excludes
的配置完全遵循 minimatch 写法。
bundle
该字段定义 Bundle 模式 下额外的配置,若开启,默认生成 dist
文件目录。bundle
包含以下配置:
formats
- 类型:
['esm', 'umd', 'cjs', 'es2017']
- 默认值:
['esm', 'es2017']
输出的类型,默认是输出 esm
和 es2017
产物。
- index.esm.es5.production.js # 输出 ES module + es5 产物
- index.esm.es2017.production.js # 输出 ES module + es2017 产物
若只需要产出 umd 规范产物,可配置为:
import { defineConfig } from '@ice/pkg';
export default defineConfig({
bundle: {
formats: ['umd', 'es2017'],
},
});
则输出以下产物:
- index.umd.es5.production.js # 输出 umd + es5 产物
- index.umd.es2017.production.js # 输出 umd + es2017 产物
注意,如果需要打包生成 umd 规范产物,不能够配置多个 entry(入口),否则会报错 Error: Invalid value "umd" for option "output.format" - UMD and IIFE output formats are not supported for code-splitting builds.
cjs 规范产物同理将 formats
配置为 ['cjs', 'es2017']
即可。
Bundle 模式的 formats 如果单独配置 ['es2017']
将不会生效,因为其仅决定产物语法层面规范,而无法决定产物的模块规范。因此其必须与 'esm'
、'umd'
和 'cjs'
中的至少一项搭配配置才能正常生成对应模块规范的 ES2017 产物。
modes
- 类型:
Array<'development' | 'production'>
- 默认值:
['production']
指定输出的产物是否经过压缩。默认情况下输出的产物是压缩过的。
- index.esm.es5.production.js # 输出 ES module + es5 产物
- index.esm.es2017.production.js # 输出 ES module + es2017 产物
增加 'development'
时,会额外输出一份未压缩的的产物,这也意味着用户可以在开发态使用该产物获得更多的开发时信息。在开发 Library 时,这将会非常有作用。
import { defineConfig } from '@ice/pkg';
export default defineConfig({
bundle: {
modes: ['production', 'development'],
},
});
- index.esm.es5.development.js # 输出未压缩产物(ES module + es5)
- index.esm.es5.production.js # 输出压缩产物 (ES module + es5)
- index.esm.es2017.development.js # 输出未压缩产物 (ES module + es2017)
- index.esm.es2017.production.js # 输出未压缩产物 (ES module + es2017)
name
- 类型:
string
- 默认值:
package.name
library 导出的名称,可以通过 window[name]
访问,一般配合打包 umd
产物时使用。默认值为 package.json
配置的 name
字段。
import { defineConfig } from '@ice/pkg';
export default defineConfig({
bundle: {
name: 'ICEPKG'
},
});
externals
- 类型:
boolean | Record<string, string>
- 默认值:
true
默认情况下,bundle 的产物包含所有依赖产物。该选项可修改这一结果。若想要 Bundle 不包含依赖产物,可如下配置:
import { defineConfig } from '@ice/pkg';
export default defineConfig({
bundle: {
externals: true,
},
});
若想要自定义配置 externals,参考如下配置:
import { defineConfig } from '@ice/pkg';
export default defineConfig({
bundle: {
externals: {
react: 'React',
'react-dom': 'ReactDOM',
},
},
});
minify
- 类型:
boolean | { js?: boolean | ((mode: string, command: string) => boolean | { options?: swc.JsMinifyOptions }); css?: boolean | ((mode: string, command: string) => boolean | { options?: cssnano.Options });}
- 默认值:build 阶段且 mode 是
production
时为true
,否则为false
是否压缩 JS 和 CSS 资源。
import { defineConfig } from '@ice/pkg';
export default defineConfig({
bundle: {
// production 产物和 development 产物不压缩
minify: false,
// 修改 JS 和 CSS 压缩参数
minify: {
js: (mode, command) => { options: { /* */ } },
css: (mode, command) => { options: { /* */ } },
}
},
});
polyfill
- 类型:
false | 'entry' | 'usage'
- 默认值:
'usage'
配置处理 polyfill 的逻辑。不同值的含义:
false
: 不引入任何 polyfill'entry'
: 根据配置的 format 值在每个文件开头都引入对应的 polyfill'usage'
: 根据源码中使用到的代码按需引入 polyfill
polyfill
默认值将会在下个 BK 版本改成 false
。推荐组件的 bundle 产物不引入任何 polyfill(也就是设置成 false
),而是使用 CDN 的方式引入 polyfill。
development
此选项已废弃,请使用 modes
- 类型:
boolean
- 默认值:
false
若开启该选项,则会额外输出一份 未压缩的 的产物,这也意味着用户可以在开发态使用该产物获得更多的开发时信息。在开发 Library 时,这将会非常有作用。
import { defineConfig } from '@ice/pkg';
export default defineConfig({
bundle: {
development: true,
},
});
上述配置会输出如下产物:
- index.esm.es5.development.js # 输出未压缩产物(ES module + es5)
- index.esm.es5.production.js # 输出压缩产物 (ES module + es5)
- index.esm.es2017.development.js # 输出未压缩产物 (ES module + es2017)
- index.esm.es2017.production.js # 输出未压缩产物 (ES module + es2017)