import{_ as e,r as i,o,c,b as n,d as a,a as p,f as t}from"./app.bd9c95b7.js";const l={},u=t(`
一些关键的路径:
build
产物的绝对路径src
目录的绝对路径pages
目录的绝对路径.fes
临时文件目录的绝对路径执行命令的绝对路径
package.json
的内容
config
实例
用户配置
插件配置可被修改,此为最终的配置
p
环境变量
注册阶段执行,用于描述插件或插件集的 id、key、配置信息、启用方式等。
用法:describe({ id?: string, key?: string, config?: { default, schema, onChange } }, enableBy?)
例如:
api.describe({
key: 'esbuild',
config: {
schema(joi) {
return joi.object();
},
default: {},
},
enableBy: api.EnableBy.config,
});
注:
`,26),r=n("li",null,[n("code",null,"config.default"),a(" 为配置的默认值,用户没有配置时取这个")],-1),d=n("code",null,"config.schema",-1),k={href:"https://hapi.dev/module/joi",target:"_blank",rel:"noopener noreferrer"},v=n("li",null,[n("code",null,"config.onChange"),a(" 是 "),n("code",null,"dev"),a(" 阶段配置被修改后的处理机制,默认会重启 dev 进程,也可以修改为 api.ConfigChangeType.regenerateTmpFiles 只重新生成临时文件,还可以通过函数的格式自定义")],-1),m=n("li",null,[n("code",null,"enableBy"),a(" 为启用方式,默认是注册启用,可更改为 "),n("code",null,"api.EnableBy.config"),a(",还可以用自定义函数的方式决定其启用时机(动态生效)")],-1),h=t(`为 api.applyPlugins 注册可供其使用的 hook。
用法:register({ key: string, fn: Function, pluginId?: string, before?: string, stage?: number })
参数:
api.applyPlugins
时,此函数被执行。// 可同步
api.register({
key: 'foo',
fn() {
return 'a';
},
});
// 可异步
api.register({
key: 'foo',
async fn() {
await delay(100);
return 'b';
},
});
注意:
api.appyPlugins
的 type
参数来看,如果是 api.ApplyPluginsType.add
,需有返回值,这些返回值最终会被合成一个数组。如果是 api.ApplyPluginsType.modify
,需对第一个参数做修改,并返回它,它会作为下个 hook 的参数。 如果是 api.ApplyPluginsType.event
,无需返回值取得 register 注册的 hooks 执行后的数据。
用法:applyPlugins({ key: string, type: api.ApplyPluginsType, initialValue?: any, args?: any })
参数:
例如:
const foo = await api.applyPlugins({
key: 'foo',
type: api.ApplyPluginsType.add,
initialValue: [],
});
console.log(foo); // ['a', 'b']
编译时插件 hook 执行类型,enum 类型,包含三个属性:
往 api
上注册方法。如果有提供 fn
,则执行 fn
定义的函数。
用法:registerMethod({ name: string, fn?: Function, exitsError?: boolean })
例如:
api.registerMethod({
name: 'writeTmpFile',
fn({ path, content }) {
assert(api.stage >= api.ServiceStage.pluginReady, 'api.writeTmpFile() should not execute in register stage.');
const absPath = join(api.paths.absTmpPath, path);
api.utils.mkdirp.sync(dirname(absPath));
if (!existsSync(absPath) || readFileSync(absPath, 'utf-8') !== content) {
writeFileSync(absPath, content, 'utf-8');
}
},
});
然后在插件中可以使用:
api.writeTmpFile();
例如:
api.registerCommand({
command: 'webpack',
description: 'inspect webpack configurations',
options: [
{
name: '--rule <ruleName>',
description: 'inspect a specific module rule',
},
{
name: '--plugin <pluginName>',
description: 'inspect a specific plugin',
},
{
name: '--rules',
description: 'list all module rule names',
},
{
name: '--plugins',
description: 'list all plugin names',
},
{
name: '--verbose',
description: 'show full function definitions in output',
},
],
async fn({ rawArgv, args, options, program }) {},
});
当项目引入此插件后,使用:
fes webpack
注册插件集,参数为路径数组。
用法:registerPresets(presets: string[])
例如:
api.registerPresets([{ id: 'preset_2', key: 'preset2', apply: () => () => {} }, require.resolve('./preset_3')]);
注册插件,参数为路径数组。
用法:registerPlugins(plugins: string[])
例如:
api.registerPlugins([{ id: 'preset_2', key: 'preset2', apply: () => () => {} }, require.resolve('./preset_3')]);
判断是否有注册某个插件,插件的 id 规则:
@@
为前缀,比如 @@/registerMethod
用法:hasPlugins(pluginIds: string[])
例如
// 判断是否有注册 @fesjs/plugin-locale
api.hasPlugins(['@fesjs/plugin-locale']);
提示
如果在注册阶段使用,只能判断在他之前是否有注册某个插件。
判断是否有注册某个插件集。
用法:hasPresets(presetIds: string[])
例如
// 判断是否有注册
api.hasPlugins(['@fesjs/preset-xxx']);
提示
如果在注册阶段使用,只能判断在他之前是否有注册某个插件集。
声明哪些插件需要被禁用,参数为插件 id 的数组。
用法:hasPresets(presetIds: string[])
例如:
// 禁用 plugin-model 插件
api.skipPlugins(['@fesjs/plugin-model']);
通过 api.registerMethod() 扩展的方法。
把插件需要导出的运行时 API 写入@fesjs/fes
。
api.addPluginExports(() => [
{
specifiers: ['access', 'useAccess'],
source: absoluteFilePath,
},
]);
这样用户使用时:
import { access, useAccess } from '@fesjs/fes';
提供给其他插件运行时需要的 API。
api.addCoreExports(() => [
{
specifiers: ['getRoutes'],
source: absCoreFilePath,
},
]);
使用:
import { getHistory, destroyRouter } from '@@/core/coreExports';
添加运行时插件,返回值格式为表示文件路径的字符串。
例如:
api.addRuntimePlugin(() => join(__dirname, './runtime'));
添加插件提供的运行时配置的 key,返回值格式为字符串。
例如:
api.addRuntimePluginKey(() => 'some');
则用户可以:
// app.js
const some = function () {
return {};
};
在入口文件现有 import 的前面添加 import。
例如:
api.addEntryImportsAhead(() => [{ source: 'anypackage' }]);
在入口文件现有 import 的后面添加 import。
例如:
api.addEntryImport(() => {
return [
{
source: '/modulePath/xxx.js',
specifier: 'moduleName',
},
];
});
在入口文件最前面(import 之后)添加代码。
例如:
api.addEntryCodeAhead(
() => \`\${globalCSSFile
.map(file => \`require('\${winPath(relative(absTmpPath, file))}');\`)
.join('')}\`
在入口文件最后添加代码。
例如:
api.addEntryCode(() => {
return \`console.log('works!')\`;
});
在 HTML 头部添加脚本。
例如:
api.addHTMLHeadScripts(() => {
return [
{
content: '',
src: '',
// ...attrs
},
];
});
添加在 webpack compiler
中间件之前的中间件,返回值格式为 express
中间件。
例如:
api.addBeforeMiddlewares(() => {
return (req, res, next) => {
if (false) {
res.end('end');
} else {
next();
}
};
});
添加在 webpack compiler
中间件之后的中间件,返回值格式为 express
中间件。
添加重新生成临时文件的监听路径。
例如:
api.addTmpGenerateWatcherPaths(() => ['./app.js']);
通过 [webpack-chain] 的方式修改 webpack 配置。
例如:
api.chainWebpack((memo) => {
memo.resolve.alias.set('vue-i18n', 'vue-i18n/dist/vue-i18n.esm-bundler.js');
});
批量写临时文件。
例如:
api.copyTmpFiles({
namespace,
path: join(__dirname, 'runtime'),
ignore: ['.tpl'],
});
参数:
提示
不能在注册阶段使用,通常放在 api.onGenerateFiles() 里,这样能在需要时重新生成临时文件 临时文件的写入做了缓存处理,如果内容一致,不会做写的操作,以减少触发 webpack
的重新编译
获取端口号,dev 时有效。
获取 hostname,dev 时有效。
获取 devServer,dev 时有效。
获取 api.modifyRoutes
修改过后的路由信息。
获取格式化后的路由信息
修改路由。
例如:
// 把BaseLayout插入到路由配置中,作为根路由
api.modifyRoutes((routes) => [
{
path: '/',
component: winPath(join(api.paths.absTmpPath || '', absFilePath)),
children: routes,
},
]);
修改获取 bundleConfig 的函数参数。
例如:
api.modifyBundleConfigOpts((memo) => {
memo.miniCSSExtractPluginPath = require.resolve('mini-css-extract-plugin');
memo.miniCSSExtractPluginLoaderPath = require.resolve('mini-css-extract-plugin/dist/loader');
return memo;
});
修改 bundle 配置。
api.modifyBundleConfig((bundleConfig) => {
// do something
return bundleConfig;
});
修改 babel 配置项。
例如:
api.modifyBabelOpts((babelOpts) => {
if (api.config.babelPluginImport) {
api.config.babelPluginImport.forEach((config) => {
babelOpts.plugins.push(['import', config]);
});
}
return babelOpts;
});
修改 babel 插件的配置。
例如:
api.modifyBabelPresetOpts((opts) => {
return {
...opts,
};
});
修改 paths 对象。
修改默认配置。 例如:
api.modifyDefaultConfig((memo) => {
return {
...memo,
...defaultOptions,
};
});
修改最终配置。
例如:
api.modifyConfig((memo) => {
return {
...memo,
...defaultOptions,
};
});
修改 publicPath 字符串。
例如:
api.modifyPublicPathStr(() => {
return api.config.publicPath || '/';
});
在插件初始化完成触发。在 onStart 之前,此时还没有 config 和 paths,他们尚未解析好。
在命令注册函数执行前触发。可以使用 config 和 paths。
dev 退出时触发。
生成临时文件,触发时机在 webpack 编译之前。
重启 devServer,dev 时有效。
写临时文件。
例如:
api.writeTmpFile({
path: absoluteFilePath,
content: Mustache.render(readFileSync(join(__dirname, 'runtime/core.tpl'), 'utf-8'), {
REPLACE_ROLES: JSON.stringify(roles),
}),
});
参数:
提示
不能在注册阶段使用,通常放在 api.onGenerateFiles() 里,这样能在需要时重新生成临时文件 临时文件的写入做了缓存处理,如果内容一致,不会做写的操作,以减少触发 webpack 的重新编译