构建配置
ice.js 支持常用的构建配置项,所有的配置项在 ice.config.mts
中设置。
配置文件
构建配置文件
为了获取良好的类型提示,ice.js 推荐以 ice.config.mts
作为配置文件:
import { defineConfig } from '@ice/app';
export default defineConfig(() => ({
publicPath: '/',
}));
兼容性配置
构建的兼容性配置推荐配置在 .browserslistrc
文件中:
chrome 55
更多配置请参考 browserslist 文档
配置项
alias
- 类型:
Record<string, string | false>
- 默认值:
{ "@": "./src/" }
在 ice.js 默认内置常用的 alias 规则,因此项目大多数时候不需要配置即可更加简单的导入模块了:
-import CustomTips from '../../../components/CustomTips';
+import CustomTips from '@/components/CustomTips';
如果需要配置别名对 import 路径进行映射:
import { defineConfig } from '@ice/app';
export default defineConfig(() => ({
alias: {
pages: './src/pages',
},
}));
crossOriginLoading
- 类型:
false | 'anonymous' | 'use-credentials'
- 默认值:
false
配置
import { defineConfig } from '@ice/app';
export default defineConfig(() => ({
crossOriginLoading: 'anonymous'
}));
指定 webpack 启用 cross-origin 去加载 chunk。
define
- 类型:
Record<string, string | boolean>
- 默认值:
{ 'process.env.NODE_ENV': 'development' | 'production'; 'import.meta.renderer': 'client' | 'server'; 'import.meta.target': string; }
在编译时将代码中的全局变量替换成其他值或者表达式。一般用于区分不同环境以执行不同代码逻辑。
import { defineConfig } from '@ice/app';
export default defineConfig(() => ({
define: {
ASSETS_VERSION: JSON.stringify('0.1.0'),
AGE: '11',
},
}));
在代码中直接使用对应定义的变量:
console.log(ASSETS_VERSION);
// 最终会被编译成:
// console.log('0.1.0');
console.log(AGE);
// 最终会被编译成:
// console.log(11);
注意,在编译时,将会对你设置的 define
替换值进行类似字符串拼接的方式生成新的代码。因此:
- 对于引用数据类型(
function
和object
),需要使用JSON.stringify()
方法处理 - 对于要替换的全局变量是字符串时,需要使用
JSON.stringify()
方法处理或者多添加一对引号(如"'hello world'"
),否则就是一个标识符,有可能跟预期结果不一致的情况
对于运行时变量,ice.js 更加推荐通过环境变量的方式注入。
dataLoader
- 类型:
boolean | { fetcher: { packageName: string; method: string } }
- 默认值
true
是否启用内置的数据预加载能力以及自定义发送者(fetcher)。
publicPath
- 类型:
string
- 默认值:
/
配置 Webpack 的 output.publicPath 属性,仅在运行 build 命令时生效。
devPublicPath
- 类型:
string
- 默认值:
/
同 publicPath 仅在执行 start 时生效。
hash
- 类型:
boolean | string
- 默认值:
false
如果希望构建后的资源带 hash 版本,可以将 hash 设置为 true
,也可以设置为 contenthash
按文件内容生成 hash 值:
import { defineConfig } from '@ice/app';
export default defineConfig(() => ({
hash: 'contenthash',
}));
externals
- 类型:
Record<string, string>
- 默认值:
{}
设置哪些模块不打包,转而通过 <script>
或其他方式引入,比如:
import { defineConfig } from '@ice/app';
export default defineConfig(() => ({
externals: {
react: 'React',
'react-dom': 'ReactDOM',
},
}));
对应在 document.ts
或者页面模版里添加 CDN 文件:
import { Main, Scripts } from 'ice';
function Document() {
return (
<html lang="en">
<body>
<Main />
+ <script crossOrigin="" src="https://unpkg.com/react@18/umd/react.production.min.js"></script>
+ <script crossOrigin="" src="https://unpkg.com/react-dom@18/umd/react-dom.production.min.js"></script>
<Scripts />
</body>
</html>
);
}
export default Document;
outputDir
- 类型:
string
- 默认值:
build
构建产物输出目录,默认为 build
目录
proxy
- 类型:
object
- 默认值:
{}
配置 dev 开发阶段的代理功能。配置项与 Webpack devServer.proxy 保持一致。
import { defineConfig } from '@ice/app';
export default defineConfig(() => ({
proxy: {
'/api': {
target: 'http://jsonplaceholder.typicode.com/',
changeOrigin: true,
pathRewrite: { '^/api' : '' },
},
},
}));
minify
- 类型:
boolean
- 默认值:
true
压缩产物,目前默认仅在 build 阶段生效
dropLogLevel
- 类型:
boolean | DropType[] | DropType
- 默认值:
false
,不移除任何 console 代码
压缩代码时移除 console. 相关代码,配置为true时,移除所有console.相关代码。当想移除部分console代码,例如想要移除console.log和console.error时,可以配置为
import { defineConfig } from '@ice/app';
export default defineConfig(() => ({
dropLog: ['error', 'log'],
}));
也可以根据console等级来进行移除
// console 等级为 trace < debug < log < info < warn < error
// 例如想要移除trace、debug、log时可以像下面这样配置
import { defineConfig } from '@ice/app';
export default defineConfig(() => ({
dropLog: 'log',
}));
compileDependencies
- 类型:
array | boolean
- 默认值:
[]
默认情况下为了保证 dev 开发阶段的体验,node_modules
下文件不会进行编译,而考虑到 build 阶段对代码体积的极致优化以及兼容性保证,将会对 node_modules
下内容也进行编译。
如果 dev 阶段需要额外编译一些依赖,build 阶段下仍然全量编译,可以参考下面的方式在 dev 阶段通过正则追加一些配置:
import { defineConfig } from '@ice/app';
export default defineConfig(() => ({
compileDependencies: process.env.NODE_ENV === 'development' ? [/@alifd\/next/, /need-compile/] : true,
}));
如果 build 阶段仍然需要全量编译,请务必增加环境判断
如果希望 dev 和 build 阶段均编译 node_modules
,可以设置为 true
import { defineConfig } from '@ice/app';
export default defineConfig(() => ({
compileDependencies: true,
}));
如果明确知道哪些依赖需要进行编译也可以通过正则方式进行设置:(对 dev 和 build 同时生效)
import { defineConfig } from '@ice/app';
export default defineConfig(() => ({
compileDependencies: [/@alifd\/next/, /need-compile/],
}));
postcss
- 类型:
ProcessOptions & { plugins?: (string | [string, Record<string, any>?])[] };
- 默认值:
{}
用于添加 postcss 自定义配置。示例如下:
import { defineConfig } from '@ice/app';
export default defineConfig(() => ({
postcss: {
plugins: [
'postcss-px-to-viewport-8-plugin',
{
// ...
},
],
syntax: 'sugarss',
}
}));
ice.js 内置的 postcss 配置是:
{
"plugins": [
["postcss-nested"],
["postcss-preset-env", {
"stage": 3,
"autoprefixer": {
"flexbox": "no-2009",
},
"features": {
"custom-properties": false,
},
}],
["postcss-plugin-rpx2vw"],
],
}
如果需要完全重写 postcss 配置或修改内置的 postcss 配置,需要在项目根目录下新增 postcss.config.js
文件并加入配置,工程上会清空内置的 postcss 配置。
module.exports = {
plugins: [
[
'postcss-preset-env',
// 修改 postcss-preset-env 的选项
{
stage: 2,
}
]
],
}
polyfill
- 类型:
'usage' | 'entry' | false
- 默认值:
false
框架提供了多种 polyfill 的方式,开发者可以按实际情况选择对应的设置:
usage
按开发者使用的语法自动引入对应的polyfill
,适用于node_modules
也进行编译的场景(一定程度上影响编译效率以及三方依赖二次编译造成的代码冗余)entry
自动引入 browser(浏览器)需要兼容的polyfill
,适用于node_modules
依赖不进行编译的场景(可能存在大量未被使用的polyfill
被引入)
如果面向现代浏览器进行开发,大量 ES 语法均不需要引入 Polyfill,我们推荐不开启
polyfill
配置。如果你的代码或者三方依赖要求兼容到 IE 11 等浏览器,可以选择主动引入指定语法的 polyfill 或者开启polyfill
配置。
transform
- 类型:
(code:string, id: string) => string | {code: string; map?: SourceMap | null;}
- 默认值:
undefined
通过 transform
配置实现代码的转化:
import { defineConfig } from '@ice/app';
import { transformSync } from '@babel/core';
export default defineConfig(() => ({
transform: (originalCode, id) => {
if (!id.includes('node_modules')) {
// 借助 babel 编译
const { code, map } = transformSync(originalCode, {
plugins: ['transform-decorators-legacy'],
});
return { code, map };
}
},
}));
ice.js 内置通过
swc
提升编译体验,如果在transform
配置上过多依赖 babel 等工具将可以能造成编译性能瓶颈
ssr
- 类型:
boolean
- 默认值:
false
是否开启 SSR 能力,更多 SSR 相关内容参考 SSR 文档。
ssg
- 类型:
boolean
- 默认值:
true
是否开启 SSG 能力,更多 SSG 相关内容参考 SSG 文档。
server
- 类型:
{ format: 'esm' | 'cjs'; bundle: boolean; ignores: IgnorePattern[]; externals: string[]; onDemand: boolean; }
- 默认值:
{ format: 'esm', bundle: false, ignores: [], externals: [], onDemand: false }
SSR / SSG 产物标准,推荐以 ESM 标准进行执行,如果希望打包成一个 cjs 模块,可以进行如下设置:
import { defineConfig } from '@ice/app';
export default defineConfig(() => ({
server: {
format: 'cjs',
bundle: true,
},
}));
可以通过 ignores
参数,为 SSR / SSG 产物过滤指定文件:
import { defineConfig } from '@ice/app';
export default defineConfig(() => ({
server: {
ignores: [{
resourceRegExp: /^\.\/locale$/,
contextRegExp: /moment$/,
}]
},
}));
其中:
- resourceRegExp 对应文件的匹配路径
- contextRegExp (可选)对应文件内容的匹配规则
通过 externals
参数,可以在构建 Server 端产物时 external 指定内容:
import { defineConfig } from '@ice/app';
export default defineConfig(() => ({
server: {
externals: ['react', 'react-dom']
},
}));
通过 onDemand
参数,可以在执行 Server 端产物时,按需构建所需的问题,并且提供体验良好的模块热更新服务:
import { defineConfig } from '@ice/app';
export default defineConfig(() => ({
server: {
onDemand: true,
format: 'esm',
},
}));
routes
- 类型:
{ ignoreFiles: string[]; defineRoutes: (route: DefineRouteFunction) => void }
- 默认值:
{}
ignoreFiles
用于忽略 src/pages
下的文件被处理成路由模块,使用 glob 表达式(minimatch)对文件路径匹配。
import { defineConfig } from '@ice/app';
export default defineConfig(() => ({
routes: {
// 忽略 src/pages 下所有 components 目录
ignoreFiles: ['**/components/**'],
},
}));
defineRoutes
对于约定式路由不满足的场景,可以通过以下方式自定义路由地址。
import { defineConfig } from '@ice/app';
export default defineConfig(() => ({
routes: {
defineRoutes: (route) => {
// 将 /about-me 路由访问内容指定为 about.tsx
// 第一个参数是路由地址
// 第二个参数是页面组件的相对地址(前面不能带 `/`),相对于 `src/pages` 目录
route('/about-me', 'about.tsx');
// 嵌套路由的场景需要使用第三个 callback 参数来定义嵌套路由
route('/', 'layout.tsx', () => {
route('/product', 'products.tsx');
});
},
},
}));
同一个路由组件只能分配一条路由规则,即同时执行以下语句时,仅生效后执行的逻辑。
route('/about-me', 'about.tsx');
route('/about-you', 'about.tsx');
config
对于简单的自定义场景,通过 defineRoutes
可以快速在约定式路由的基础上进行自定义。但对于大量自定义或者原配置式路由的升级项目,支持以 config
的字段指定路由信息:
import { defineConfig } from '@ice/app';
export default defineConfig({
routes: {
config: [
{
path: 'rewrite',
// 从 src/page 开始计算路径,并且需要写后缀。
component: 'sales/layout.tsx',
children: [
{
path: '/favorites',
component: 'sales/favorites.tsx',
},
{
path: 'overview',
component: 'sales/overview.tsx',
},
{
path: 'recommends',
component: 'sales/recommends.tsx',
},
],
},
{
path: '/',
component: 'index.tsx',
},
],
},
});
sourceMap
- 类型:
boolean | string
- 默认值:
development
模式:默认为 'cheap-module-source-map',支持通过false
关闭,不支持设置为其他枚举值。production
模式:默认false
。
splitChunks @deprecated
不再建议使用,能力由 codeSplitting 替代
默认会根据模块体积自动拆分 chunks,有可能会出现多个 bundle。如果不希望打包产物出现过多 bundle ,可设置成 false
。
codeSplitting
- 类型:
boolean | 'vendors' | 'page' | 'chunks' | 'page-vendors'
- 默认值:
true
框架内置了三种分包策略分别为 chunks
(默认策略,无需额外设置),page
和 vendors
。
vendors
策略:将异步 chunks 里的三方依赖统一打入到 vendor.js 中,避免重复,在依赖不变的情况下有效利用缓存。缺陷是如果项目过大会导致单文件尺寸过大。page
策略:所有路由级别组件按需加载,如果需保留原splitChunks: false
的效果,配置该策略 。page-vendors
策略:在page
策略的基础上,将异步 chunks 里的三方依赖统一打入到 vendor.js 中,以达到有效利用缓存的结果。chunks
策略:在路由级别组件按需加载的基础上,根据模块体积大小自动拆分 chunks,为框架默认推荐策略。
如果存在特殊场景期望关闭分包能力,可以设置成 false
。
syntaxFeatures
- 类型:
{ exportDefaultFrom: boolean; functionBind: boolean; }
- 默认值:
undefined
ice.js 内置了大量 ES 语法支持,便于开发者进行编码。对于 proposal-export-default-from 和 proposal-bind-operator 由于其提案进度较慢,我们并不推荐使用。如果希望支持该语法,可以主动配置 syntaxFeatures
进行启用。
tsChecker
- 类型:
boolean
- 默认值:
false
默认关闭 TypeScript 类型检测,如需开启配置为 true
即可。
eslint
- 类型:
boolean | object
- 默认值:
undefined
配置说明:
false
:不检测 eslint 错误true
:将 eslint 错误展示在预览页面上object
: 仅 Webpack 模式支持,表现等同于 true,支持配置 eslint-webpack-plugin 的更多参数
mock
- 类型:
{ exclude: string[] }
- 默认值:
{}
配置忽略 mock 的文件。
import { defineConfig } from '@ice/app';
export default defineConfig(() => ({
mock: {
// 忽略 mock 目录中 custom 目录下的文件以及 api.ts 文件
exclude: ["custom/**", "api.ts"]
},
}));
htmlGenerating
- 类型:
boolean
- 默认值:
true
如果产物不想生成 html,可以设置为 false
,在 SSG 开启的情况下,强制关闭 html 生成,将导致 SSG 失效。
plugins
- 类型:
PluginList<Config, OverwritePluginAPI>
- 默认值:
[]
添加插件
import { defineConfig } from '@ice/app';
import customPlugin from './custom-plugin';
import myPlugin from '@ice/my-plugin';
export default defineConfig(() => ({
plugins: [
customPlugin(),
myPlugin(),
],
}));
webpack
ice.js 对 webpack 构建配置进行了定制,并借助 esbuild 等工具提升用户开发体验,直接修改 webpack 配置的方式并不推荐。
- 类型:
(config: WebpackConfig, taskConfig: TaskConfig) => WebpackConfig
- 默认值:
true
ice.js 默认基于 webpack 5 进行构建,在上述提供的构建配置无法满足的情况下,用户可以定制 webpack 配置:
import { defineConfig } from '@ice/app';
import SpeedMeasurePlugin from 'speed-measure-webpack-plugin';
export default defineConfig(() => ({
webpack: (webpackConfig) => {
if (process.env.NODE_ENV !== 'test') {
// 添加 webpack 插件
webpackConfig.plugins?.push(new SpeedMeasurePlugin());
}
return webpackConfig;
},
}));
cssModules
- 类型:
{ localIdentName: string }
- 默认值:
{}
构建 cssModules 时,定制 class 名称的生成规则,配置参考 https://webpack.js.org/loaders/css-loader/#localidentname
例如,配置 '[hash:8]' 可以只保留 hash 值,以精简 HTML 大小及 CSS 文件大小。默认情况 className="custom-head-tab-wrap"
会被构建为 class="custom-head-tab-wrap--rAEgGaqM"
,自定义构建规则后后样式名会被精简为 class="rAEgGaqM"
。
import { defineConfig } from '@ice/app';
export default defineConfig(() => ({
cssModules: {
localIdentName: '[hash:8]'
},
}));
optimization
- 类型:
{ disableRouter: boolean; optimizePackageImport: boolean | string[] }
- 默认值:
{}
框架提供内置的优化能力,针对不同场景提供优化策略:
disableRouter
:默认为false
,如果希望关闭路由能力,可以设置为true
。主要应用不存在依赖路由能力的场景,比如不存在 SPA 页面跳转。optimizePackageImport
:默认为false
,开启后框架默认会对已知三方依赖进行按需加载,项目的构建体验将进一步提升,内置三方依赖列表参考代码。
参考配置:
import { defineConfig } from '@ice/app';
export default defineConfig(() => ({
optimization: {
disableRouter: true,
// optimizePackageImport 配置为 true 则使用内置的三方依赖列表,如果配置为数组则会在内置列表基础上追加
optimizePackageImport: ['@ice/components'],
},
}));
如有定制需求欢迎👏 PR 或反馈:https://github.com/alibaba/ice/issues