mirror of
https://github.com/WeBankFinTech/fes.js.git
synced 2025-04-06 03:59:53 +08:00
feat: 优化构建
This commit is contained in:
parent
7cc8ea51eb
commit
242787c358
@ -40,7 +40,7 @@
|
|||||||
"@vuepress/plugin-docsearch": "2.0.0-beta.28",
|
"@vuepress/plugin-docsearch": "2.0.0-beta.28",
|
||||||
"@vuepress/plugin-pwa": "2.0.0-beta.28",
|
"@vuepress/plugin-pwa": "2.0.0-beta.28",
|
||||||
"@vuepress/plugin-pwa-popup": "2.0.0-beta.28",
|
"@vuepress/plugin-pwa-popup": "2.0.0-beta.28",
|
||||||
"@webank/eslint-config-webank": "0.3.1",
|
"@webank/eslint-config-webank": "^1.2.0",
|
||||||
"chalk": "^4.1.2",
|
"chalk": "^4.1.2",
|
||||||
"chokidar": "^3.5.2",
|
"chokidar": "^3.5.2",
|
||||||
"commitizen": "^4.2.1",
|
"commitizen": "^4.2.1",
|
||||||
@ -56,7 +56,7 @@
|
|||||||
},
|
},
|
||||||
"lint-staged": {
|
"lint-staged": {
|
||||||
"*.{js,jsx,vue,ts}": [
|
"*.{js,jsx,vue,ts}": [
|
||||||
"eslint --format=codeframe"
|
"eslint"
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
"husky": {
|
"husky": {
|
||||||
|
@ -10,8 +10,8 @@ export default (api) => {
|
|||||||
config: {
|
config: {
|
||||||
schema(joi) {
|
schema(joi) {
|
||||||
return joi.object();
|
return joi.object();
|
||||||
}
|
},
|
||||||
}
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
const namespace = 'plugin-icon';
|
const namespace = 'plugin-icon';
|
||||||
@ -24,34 +24,29 @@ export default (api) => {
|
|||||||
api.onGenerateFiles(async () => {
|
api.onGenerateFiles(async () => {
|
||||||
const base = join(api.paths.absSrcPath, 'icons');
|
const base = join(api.paths.absSrcPath, 'icons');
|
||||||
const iconFiles = api.utils.glob.sync('**/*', {
|
const iconFiles = api.utils.glob.sync('**/*', {
|
||||||
cwd: join(api.paths.absSrcPath, 'icons')
|
cwd: join(api.paths.absSrcPath, 'icons'),
|
||||||
});
|
});
|
||||||
const svgDatas = await optimizeSvg(iconFiles.map(item => join(base, item)));
|
const svgDatas = await optimizeSvg(iconFiles.map((item) => join(base, item)));
|
||||||
const iconNames = [];
|
const iconNames = [];
|
||||||
const SVG_COMPONENT_TMPLATE = 'export default () => (SVG)';
|
const SVG_COMPONENT_TMPLATE = 'export default () => (SVG)';
|
||||||
for (const { fileName, data } of svgDatas) {
|
for (const { fileName, data } of svgDatas) {
|
||||||
iconNames.push(basename(fileName, '.svg'));
|
iconNames.push(basename(fileName, '.svg'));
|
||||||
api.writeTmpFile({
|
api.writeTmpFile({
|
||||||
path: `${namespace}/icons/${basename(fileName, '.svg')}.js`,
|
path: `${namespace}/icons/${basename(fileName, '.svg')}.jsx`,
|
||||||
content: SVG_COMPONENT_TMPLATE
|
content: SVG_COMPONENT_TMPLATE.replace('SVG', data),
|
||||||
.replace('SVG', data)
|
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
api.writeTmpFile({
|
api.writeTmpFile({
|
||||||
path: `${namespace}/icons.js`,
|
path: `${namespace}/icons.js`,
|
||||||
content: api.utils.Mustache.render(
|
content: api.utils.Mustache.render(readFileSync(join(__dirname, 'runtime/icons.tpl'), 'utf-8'), {
|
||||||
readFileSync(join(__dirname, 'runtime/icons.tpl'), 'utf-8'),
|
ICON_NAMES: iconNames,
|
||||||
{
|
}),
|
||||||
ICON_NAMES: iconNames
|
|
||||||
}
|
|
||||||
)
|
|
||||||
});
|
});
|
||||||
|
|
||||||
api.writeTmpFile({
|
api.writeTmpFile({
|
||||||
path: absRuntimeFilePath,
|
path: absRuntimeFilePath,
|
||||||
content: api.utils.Mustache.render(readFileSync(join(__dirname, 'runtime/runtime.tpl'), 'utf-8'), {
|
content: api.utils.Mustache.render(readFileSync(join(__dirname, 'runtime/runtime.tpl'), 'utf-8'), {}),
|
||||||
})
|
|
||||||
});
|
});
|
||||||
|
|
||||||
if (!generatedOnce) {
|
if (!generatedOnce) {
|
||||||
@ -59,7 +54,7 @@ export default (api) => {
|
|||||||
api.copyTmpFiles({
|
api.copyTmpFiles({
|
||||||
namespace,
|
namespace,
|
||||||
path: join(__dirname, 'runtime'),
|
path: join(__dirname, 'runtime'),
|
||||||
ignore: ['.tpl']
|
ignore: ['.tpl'],
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
@ -1,9 +1,8 @@
|
|||||||
<script>
|
|
||||||
import { computed } from 'vue';
|
import { computed } from 'vue';
|
||||||
// eslint-disable-next-line
|
// eslint-disable-next-line
|
||||||
import icons from '../icons';
|
import icons from '../icons';
|
||||||
|
|
||||||
const noop = () => { };
|
const noop = () => {};
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
name: 'FesIcon',
|
name: 'FesIcon',
|
||||||
@ -17,25 +16,21 @@ export default {
|
|||||||
}
|
}
|
||||||
return tabIndex;
|
return tabIndex;
|
||||||
});
|
});
|
||||||
const svgStyle = computed(() => (props.rotate
|
const svgStyle = computed(() =>
|
||||||
? {
|
props.rotate
|
||||||
msTransform: `rotate(${props.rotate}deg)`,
|
? {
|
||||||
transform: `rotate(${props.rotate}deg)`
|
msTransform: `rotate(${props.rotate}deg)`,
|
||||||
}
|
transform: `rotate(${props.rotate}deg)`,
|
||||||
: null));
|
}
|
||||||
|
: null,
|
||||||
|
);
|
||||||
const svgCls = computed(() => ({
|
const svgCls = computed(() => ({
|
||||||
'inner-icon--spin': !!props.spin || props.type === 'loading'
|
'inner-icon--spin': !!props.spin || props.type === 'loading',
|
||||||
}));
|
}));
|
||||||
return () => (
|
return () => (
|
||||||
<span
|
<span tabIndex={iconTabIndex.value} role="img" class="inner-icon" onClick={attrs.onClick || noop}>
|
||||||
tabIndex={iconTabIndex.value}
|
|
||||||
role="img"
|
|
||||||
class="inner-icon"
|
|
||||||
onClick={attrs.onClick || noop}
|
|
||||||
>
|
|
||||||
<CurrentIcon.value class={svgCls.value} style={svgStyle.value} />
|
<CurrentIcon.value class={svgCls.value} style={svgStyle.value} />
|
||||||
</span>
|
</span>
|
||||||
);
|
);
|
||||||
}
|
},
|
||||||
};
|
};
|
||||||
</script>
|
|
@ -42,13 +42,10 @@
|
|||||||
"publishConfig": {
|
"publishConfig": {
|
||||||
"access": "public"
|
"access": "public"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
|
||||||
"@webank/eslint-config-webank": "0.3.1"
|
|
||||||
},
|
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@fesjs/fes": "^2.0.0",
|
"@fesjs/fes": "^2.0.0",
|
||||||
"vue": "^3.0.5",
|
"vue": "^3.0.5",
|
||||||
"@fesjs/fes-design": "^0.1.10"
|
"@fesjs/fes-design": "^0.1.10"
|
||||||
},
|
},
|
||||||
"private": true
|
"private": true
|
||||||
}
|
}
|
@ -42,13 +42,10 @@
|
|||||||
"publishConfig": {
|
"publishConfig": {
|
||||||
"access": "public"
|
"access": "public"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
|
||||||
"@webank/eslint-config-webank": "0.3.1"
|
|
||||||
},
|
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@fesjs/fes": "^2.0.0",
|
"@fesjs/fes": "^2.0.0",
|
||||||
"vue": "^3.0.5",
|
"vue": "^3.0.5",
|
||||||
"@fesjs/fes-design": "^0.1.10"
|
"@fesjs/fes-design": "^0.1.10"
|
||||||
},
|
},
|
||||||
"private": true
|
"private": true
|
||||||
}
|
}
|
@ -1,4 +1,3 @@
|
|||||||
|
|
||||||
import WindiCSSWebpackPlugin from 'windicss-webpack-plugin';
|
import WindiCSSWebpackPlugin from 'windicss-webpack-plugin';
|
||||||
|
|
||||||
export default (api) => {
|
export default (api) => {
|
||||||
@ -8,8 +7,8 @@ export default (api) => {
|
|||||||
schema(joi) {
|
schema(joi) {
|
||||||
return joi.object();
|
return joi.object();
|
||||||
},
|
},
|
||||||
default: {}
|
default: {},
|
||||||
}
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
api.addEntryImportsAhead(() => [{ source: 'windi-base.css' }, { source: 'windi-components.css' }, { source: 'windi-utilities.css' }]);
|
api.addEntryImportsAhead(() => [{ source: 'windi-base.css' }, { source: 'windi-components.css' }, { source: 'windi-utilities.css' }]);
|
||||||
@ -23,12 +22,12 @@ export default (api) => {
|
|||||||
// A common use case is scanning files from the root directory
|
// A common use case is scanning files from the root directory
|
||||||
include: ['**/*.{vue,jsx,js,ts,tsx}'],
|
include: ['**/*.{vue,jsx,js,ts,tsx}'],
|
||||||
// if you are excluding files, make sure you always include node_modules and .git
|
// if you are excluding files, make sure you always include node_modules and .git
|
||||||
exclude: ['node_modules', '.git', 'dist']
|
exclude: ['node_modules', '.git', 'dist', '.fes'],
|
||||||
},
|
},
|
||||||
...config
|
...config,
|
||||||
},
|
},
|
||||||
...otherOption
|
...otherOption,
|
||||||
}
|
},
|
||||||
]);
|
]);
|
||||||
if (api.env === 'development') {
|
if (api.env === 'development') {
|
||||||
memo.module.rule('css').test((path) => {
|
memo.module.rule('css').test((path) => {
|
||||||
@ -41,8 +40,8 @@ export default (api) => {
|
|||||||
lang: 'windicss',
|
lang: 'windicss',
|
||||||
test: /windi-utilities.css$/,
|
test: /windi-utilities.css$/,
|
||||||
styleLoaderOption: {
|
styleLoaderOption: {
|
||||||
insert: 'body'
|
insert: 'body',
|
||||||
}
|
},
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -35,7 +35,10 @@
|
|||||||
"@babel/preset-typescript": "^7.15.0",
|
"@babel/preset-typescript": "^7.15.0",
|
||||||
"@fesjs/compiler": "^2.0.5",
|
"@fesjs/compiler": "^2.0.5",
|
||||||
"@fesjs/utils": "^2.0.4",
|
"@fesjs/utils": "^2.0.4",
|
||||||
|
"@originjs/vite-plugin-commonjs": "^1.0.3",
|
||||||
"@soda/friendly-errors-webpack-plugin": "^1.8.0",
|
"@soda/friendly-errors-webpack-plugin": "^1.8.0",
|
||||||
|
"@vitejs/plugin-vue": "^2.2.4",
|
||||||
|
"@vitejs/plugin-vue-jsx": "^1.3.8",
|
||||||
"@vue/babel-plugin-jsx": "^1.0.2",
|
"@vue/babel-plugin-jsx": "^1.0.2",
|
||||||
"autoprefixer": "^10.2.4",
|
"autoprefixer": "^10.2.4",
|
||||||
"babel-loader": "^8.2.2",
|
"babel-loader": "^8.2.2",
|
||||||
@ -65,6 +68,7 @@
|
|||||||
"raw-loader": "^4.0.2",
|
"raw-loader": "^4.0.2",
|
||||||
"style-loader": "^2.0.0",
|
"style-loader": "^2.0.0",
|
||||||
"url-loader": "^4.1.1",
|
"url-loader": "^4.1.1",
|
||||||
|
"vite": "^2.8.6",
|
||||||
"vue-loader": "^16.1.2",
|
"vue-loader": "^16.1.2",
|
||||||
"webpack": "^5.24.2",
|
"webpack": "^5.24.2",
|
||||||
"webpack-bundle-analyzer": "^4.4.0",
|
"webpack-bundle-analyzer": "^4.4.0",
|
||||||
@ -75,5 +79,6 @@
|
|||||||
"peerDependencies": {
|
"peerDependencies": {
|
||||||
"@vue/compiler-sfc": "^3.0.5",
|
"@vue/compiler-sfc": "^3.0.5",
|
||||||
"core-js": "^3.8.3"
|
"core-js": "^3.8.3"
|
||||||
}
|
},
|
||||||
|
"devDependencies": {}
|
||||||
}
|
}
|
||||||
|
@ -0,0 +1,8 @@
|
|||||||
|
export default {
|
||||||
|
name: 'sfc-config',
|
||||||
|
transform(code, id) {
|
||||||
|
if (/vue&type=config/.test(id)) {
|
||||||
|
return `export default ''`;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
};
|
@ -0,0 +1,209 @@
|
|||||||
|
/**
|
||||||
|
* @copy 该文件代码大部分出自 umi,有需要请参考:
|
||||||
|
* https://github.com/umijs/umi/blob/master/packages/preset-built-in/src/plugins/commands/dev/dev.ts
|
||||||
|
*/
|
||||||
|
|
||||||
|
const assert = require('assert');
|
||||||
|
|
||||||
|
export default (api) => {
|
||||||
|
const {
|
||||||
|
env,
|
||||||
|
paths,
|
||||||
|
utils: { chalk, portfinder },
|
||||||
|
} = api;
|
||||||
|
|
||||||
|
const unwatchs = [];
|
||||||
|
let port;
|
||||||
|
let hostname;
|
||||||
|
let server;
|
||||||
|
|
||||||
|
function destroy() {
|
||||||
|
for (const unwatch of unwatchs) {
|
||||||
|
unwatch();
|
||||||
|
}
|
||||||
|
server?.close();
|
||||||
|
}
|
||||||
|
|
||||||
|
api.registerCommand({
|
||||||
|
command: 'dev',
|
||||||
|
description: 'start a local http service for development',
|
||||||
|
options: [
|
||||||
|
{
|
||||||
|
name: '--port',
|
||||||
|
description: 'http service port, like 8080',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: '--https',
|
||||||
|
description: 'whether to turn on the https service',
|
||||||
|
},
|
||||||
|
],
|
||||||
|
async fn({ args = {} }) {
|
||||||
|
const { cleanTmpPathExceptCache, getBundleAndConfigs } = require('../buildDevUtils');
|
||||||
|
const { delay } = require('@fesjs/utils');
|
||||||
|
const createRouteMiddleware = require('./createRouteMiddleware').default;
|
||||||
|
const generateFiles = require('../../../utils/generateFiles').default;
|
||||||
|
const { watchPkg } = require('./watchPkg');
|
||||||
|
|
||||||
|
const defaultPort = process.env.PORT || args.port || api.config.devServer?.port;
|
||||||
|
port = await portfinder.getPortPromise({
|
||||||
|
port: defaultPort ? parseInt(String(defaultPort), 10) : 8000,
|
||||||
|
});
|
||||||
|
hostname = process.env.HOST || api.config.devServer?.host || 'localhost';
|
||||||
|
|
||||||
|
process.send({
|
||||||
|
type: 'UPDATE_PORT',
|
||||||
|
port,
|
||||||
|
});
|
||||||
|
|
||||||
|
// enable https
|
||||||
|
const isHTTPS = process.env.HTTPS || args.https;
|
||||||
|
|
||||||
|
console.log(chalk.cyan(`Starting the development server ${isHTTPS ? 'https' : 'http'}://${hostname}:${port} ...`));
|
||||||
|
|
||||||
|
cleanTmpPathExceptCache({
|
||||||
|
absTmpPath: paths.absTmpPath,
|
||||||
|
});
|
||||||
|
const watch = process.env.WATCH !== 'none';
|
||||||
|
|
||||||
|
// generate files
|
||||||
|
const unwatchGenerateFiles = await generateFiles({
|
||||||
|
api,
|
||||||
|
watch,
|
||||||
|
});
|
||||||
|
if (unwatchGenerateFiles) unwatchs.push(unwatchGenerateFiles);
|
||||||
|
|
||||||
|
if (watch) {
|
||||||
|
// watch pkg changes
|
||||||
|
const unwatchPkg = watchPkg({
|
||||||
|
cwd: api.cwd,
|
||||||
|
onChange() {
|
||||||
|
console.log();
|
||||||
|
api.logger.info('Plugins in package.json changed.');
|
||||||
|
api.restartServer();
|
||||||
|
},
|
||||||
|
});
|
||||||
|
unwatchs.push(unwatchPkg);
|
||||||
|
|
||||||
|
// watch config change
|
||||||
|
const unwatchConfig = api.service.configInstance.watch({
|
||||||
|
userConfig: api.service.userConfig,
|
||||||
|
onChange: async ({ pluginChanged, valueChanged }) => {
|
||||||
|
if (pluginChanged.length) {
|
||||||
|
console.log();
|
||||||
|
api.logger.info(`Plugins of ${pluginChanged.map((p) => p.key).join(', ')} changed.`);
|
||||||
|
api.restartServer();
|
||||||
|
}
|
||||||
|
if (valueChanged.length) {
|
||||||
|
let reload = false;
|
||||||
|
let regenerateTmpFiles = false;
|
||||||
|
const fns = [];
|
||||||
|
const reloadConfigs = [];
|
||||||
|
valueChanged.forEach(({ key, pluginId }) => {
|
||||||
|
const { onChange } = api.service.plugins[pluginId].config || {};
|
||||||
|
if (onChange === api.ConfigChangeType.regenerateTmpFiles) {
|
||||||
|
regenerateTmpFiles = true;
|
||||||
|
}
|
||||||
|
if (!onChange || onChange === api.ConfigChangeType.reload) {
|
||||||
|
reload = true;
|
||||||
|
reloadConfigs.push(key);
|
||||||
|
}
|
||||||
|
if (typeof onChange === 'function') {
|
||||||
|
fns.push(onChange);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
if (reload) {
|
||||||
|
console.log();
|
||||||
|
api.logger.info(`Config ${reloadConfigs.join(', ')} changed.`);
|
||||||
|
api.restartServer();
|
||||||
|
} else {
|
||||||
|
api.service.userConfig = api.service.configInstance.getUserConfig();
|
||||||
|
|
||||||
|
await api.setConfig();
|
||||||
|
|
||||||
|
if (regenerateTmpFiles) {
|
||||||
|
await generateFiles({
|
||||||
|
api,
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
fns.forEach((fn) => fn());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
});
|
||||||
|
unwatchs.push(unwatchConfig);
|
||||||
|
}
|
||||||
|
|
||||||
|
// delay dev server 启动,避免重复 compile
|
||||||
|
// https://github.com/webpack/watchpack/issues/25
|
||||||
|
// https://github.com/yessky/webpack-mild-compile
|
||||||
|
await delay(500);
|
||||||
|
|
||||||
|
// dev
|
||||||
|
const { bundleConfig } = await getBundleAndConfigs({ api });
|
||||||
|
|
||||||
|
const beforeMiddlewares = await api.applyPlugins({
|
||||||
|
key: 'addBeforeMiddlewares',
|
||||||
|
type: api.ApplyPluginsType.add,
|
||||||
|
initialValue: [],
|
||||||
|
args: {},
|
||||||
|
});
|
||||||
|
const middlewares = await api.applyPlugins({
|
||||||
|
key: 'addMiddlewares',
|
||||||
|
type: api.ApplyPluginsType.add,
|
||||||
|
initialValue: [],
|
||||||
|
args: {},
|
||||||
|
});
|
||||||
|
const { startDevServer } = require('./devServer');
|
||||||
|
server = startDevServer({
|
||||||
|
webpackConfig: bundleConfig,
|
||||||
|
host: hostname,
|
||||||
|
port,
|
||||||
|
proxy: api.config.proxy,
|
||||||
|
https: isHTTPS,
|
||||||
|
beforeMiddlewares: [...beforeMiddlewares, createRouteMiddleware(api)],
|
||||||
|
afterMiddlewares: [...middlewares],
|
||||||
|
customerDevServerConfig: api.config.devServer,
|
||||||
|
});
|
||||||
|
return {
|
||||||
|
destroy,
|
||||||
|
};
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
api.registerMethod({
|
||||||
|
name: 'getPort',
|
||||||
|
fn() {
|
||||||
|
assert(env === 'development', 'api.getPort() is only valid in development.');
|
||||||
|
return port;
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
api.registerMethod({
|
||||||
|
name: 'getHostname',
|
||||||
|
fn() {
|
||||||
|
assert(env === 'development', 'api.getHostname() is only valid in development.');
|
||||||
|
return hostname;
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
api.registerMethod({
|
||||||
|
name: 'getServer',
|
||||||
|
fn() {
|
||||||
|
assert(env === 'development', 'api.getServer() is only valid in development.');
|
||||||
|
return server;
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
api.registerMethod({
|
||||||
|
name: 'restartServer',
|
||||||
|
fn() {
|
||||||
|
console.log(chalk.gray('Try to restart dev server...'));
|
||||||
|
destroy();
|
||||||
|
process.send({
|
||||||
|
type: 'RESTART',
|
||||||
|
});
|
||||||
|
},
|
||||||
|
});
|
||||||
|
};
|
@ -2,6 +2,11 @@
|
|||||||
* @copy 该文件代码大部分出自 umi,有需要请参考:
|
* @copy 该文件代码大部分出自 umi,有需要请参考:
|
||||||
* https://github.com/umijs/umi/blob/master/packages/preset-built-in/src/plugins/commands/dev/dev.ts
|
* https://github.com/umijs/umi/blob/master/packages/preset-built-in/src/plugins/commands/dev/dev.ts
|
||||||
*/
|
*/
|
||||||
|
import { createServer } from 'vite';
|
||||||
|
import vue from '@vitejs/plugin-vue';
|
||||||
|
import vueJsx from '@vitejs/plugin-vue-jsx';
|
||||||
|
import { viteCommonjs } from '@originjs/vite-plugin-commonjs';
|
||||||
|
import SFCConfigBlockPlugin from './SFCConfigBlockPlugin';
|
||||||
|
|
||||||
const assert = require('assert');
|
const assert = require('assert');
|
||||||
|
|
||||||
@ -9,7 +14,7 @@ export default (api) => {
|
|||||||
const {
|
const {
|
||||||
env,
|
env,
|
||||||
paths,
|
paths,
|
||||||
utils: { chalk, portfinder }
|
utils: { chalk, portfinder },
|
||||||
} = api;
|
} = api;
|
||||||
|
|
||||||
const unwatchs = [];
|
const unwatchs = [];
|
||||||
@ -27,48 +32,44 @@ export default (api) => {
|
|||||||
api.registerCommand({
|
api.registerCommand({
|
||||||
command: 'dev',
|
command: 'dev',
|
||||||
description: 'start a local http service for development',
|
description: 'start a local http service for development',
|
||||||
options: [{
|
options: [
|
||||||
name: '--port',
|
{
|
||||||
description: 'http service port, like 8080'
|
name: '--port',
|
||||||
}, {
|
description: 'http service port, like 8080',
|
||||||
name: '--https',
|
},
|
||||||
description: 'whether to turn on the https service'
|
{
|
||||||
}],
|
name: '--https',
|
||||||
|
description: 'whether to turn on the https service',
|
||||||
|
},
|
||||||
|
],
|
||||||
async fn({ args = {} }) {
|
async fn({ args = {} }) {
|
||||||
const {
|
const { cleanTmpPathExceptCache } = require('../buildDevUtils');
|
||||||
cleanTmpPathExceptCache,
|
|
||||||
getBundleAndConfigs
|
|
||||||
} = require('../buildDevUtils');
|
|
||||||
const { delay } = require('@fesjs/utils');
|
|
||||||
const createRouteMiddleware = require('./createRouteMiddleware').default;
|
|
||||||
const generateFiles = require('../../../utils/generateFiles').default;
|
const generateFiles = require('../../../utils/generateFiles').default;
|
||||||
const { watchPkg } = require('./watchPkg');
|
const { watchPkg } = require('./watchPkg');
|
||||||
|
|
||||||
const defaultPort = process.env.PORT || args.port || api.config.devServer?.port;
|
const defaultPort = process.env.PORT || args.port || api.config.devServer?.port;
|
||||||
port = await portfinder.getPortPromise({
|
port = await portfinder.getPortPromise({
|
||||||
port: defaultPort ? parseInt(String(defaultPort), 10) : 8000
|
port: defaultPort ? parseInt(String(defaultPort), 10) : 8000,
|
||||||
});
|
});
|
||||||
hostname = process.env.HOST || api.config.devServer?.host || 'localhost';
|
hostname = process.env.HOST || api.config.devServer?.host || 'localhost';
|
||||||
|
|
||||||
process.send({
|
process.send({
|
||||||
type: 'UPDATE_PORT',
|
type: 'UPDATE_PORT',
|
||||||
port
|
port,
|
||||||
});
|
});
|
||||||
|
|
||||||
// enable https
|
// enable https
|
||||||
const isHTTPS = process.env.HTTPS || args.https;
|
// const isHTTPS = process.env.HTTPS || args.https;
|
||||||
|
|
||||||
console.log(chalk.cyan(`Starting the development server ${isHTTPS ? 'https' : 'http'}://${hostname}:${port} ...`));
|
|
||||||
|
|
||||||
cleanTmpPathExceptCache({
|
cleanTmpPathExceptCache({
|
||||||
absTmpPath: paths.absTmpPath
|
absTmpPath: paths.absTmpPath,
|
||||||
});
|
});
|
||||||
const watch = process.env.WATCH !== 'none';
|
const watch = process.env.WATCH !== 'none';
|
||||||
|
|
||||||
// generate files
|
// generate files
|
||||||
const unwatchGenerateFiles = await generateFiles({
|
const unwatchGenerateFiles = await generateFiles({
|
||||||
api,
|
api,
|
||||||
watch
|
watch,
|
||||||
});
|
});
|
||||||
if (unwatchGenerateFiles) unwatchs.push(unwatchGenerateFiles);
|
if (unwatchGenerateFiles) unwatchs.push(unwatchGenerateFiles);
|
||||||
|
|
||||||
@ -80,7 +81,7 @@ export default (api) => {
|
|||||||
console.log();
|
console.log();
|
||||||
api.logger.info('Plugins in package.json changed.');
|
api.logger.info('Plugins in package.json changed.');
|
||||||
api.restartServer();
|
api.restartServer();
|
||||||
}
|
},
|
||||||
});
|
});
|
||||||
unwatchs.push(unwatchPkg);
|
unwatchs.push(unwatchPkg);
|
||||||
|
|
||||||
@ -90,11 +91,7 @@ export default (api) => {
|
|||||||
onChange: async ({ pluginChanged, valueChanged }) => {
|
onChange: async ({ pluginChanged, valueChanged }) => {
|
||||||
if (pluginChanged.length) {
|
if (pluginChanged.length) {
|
||||||
console.log();
|
console.log();
|
||||||
api.logger.info(
|
api.logger.info(`Plugins of ${pluginChanged.map((p) => p.key).join(', ')} changed.`);
|
||||||
`Plugins of ${pluginChanged
|
|
||||||
.map(p => p.key)
|
|
||||||
.join(', ')} changed.`
|
|
||||||
);
|
|
||||||
api.restartServer();
|
api.restartServer();
|
||||||
}
|
}
|
||||||
if (valueChanged.length) {
|
if (valueChanged.length) {
|
||||||
@ -104,16 +101,10 @@ export default (api) => {
|
|||||||
const reloadConfigs = [];
|
const reloadConfigs = [];
|
||||||
valueChanged.forEach(({ key, pluginId }) => {
|
valueChanged.forEach(({ key, pluginId }) => {
|
||||||
const { onChange } = api.service.plugins[pluginId].config || {};
|
const { onChange } = api.service.plugins[pluginId].config || {};
|
||||||
if (
|
if (onChange === api.ConfigChangeType.regenerateTmpFiles) {
|
||||||
onChange
|
|
||||||
=== api.ConfigChangeType.regenerateTmpFiles
|
|
||||||
) {
|
|
||||||
regenerateTmpFiles = true;
|
regenerateTmpFiles = true;
|
||||||
}
|
}
|
||||||
if (
|
if (!onChange || onChange === api.ConfigChangeType.reload) {
|
||||||
!onChange
|
|
||||||
|| onChange === api.ConfigChangeType.reload
|
|
||||||
) {
|
|
||||||
reload = true;
|
reload = true;
|
||||||
reloadConfigs.push(key);
|
reloadConfigs.push(key);
|
||||||
}
|
}
|
||||||
@ -124,11 +115,7 @@ export default (api) => {
|
|||||||
|
|
||||||
if (reload) {
|
if (reload) {
|
||||||
console.log();
|
console.log();
|
||||||
api.logger.info(
|
api.logger.info(`Config ${reloadConfigs.join(', ')} changed.`);
|
||||||
`Config ${reloadConfigs.join(
|
|
||||||
', '
|
|
||||||
)} changed.`
|
|
||||||
);
|
|
||||||
api.restartServer();
|
api.restartServer();
|
||||||
} else {
|
} else {
|
||||||
api.service.userConfig = api.service.configInstance.getUserConfig();
|
api.service.userConfig = api.service.configInstance.getUserConfig();
|
||||||
@ -137,86 +124,91 @@ export default (api) => {
|
|||||||
|
|
||||||
if (regenerateTmpFiles) {
|
if (regenerateTmpFiles) {
|
||||||
await generateFiles({
|
await generateFiles({
|
||||||
api
|
api,
|
||||||
});
|
});
|
||||||
} else {
|
} else {
|
||||||
fns.forEach(fn => fn());
|
fns.forEach((fn) => fn());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
},
|
||||||
});
|
});
|
||||||
unwatchs.push(unwatchConfig);
|
unwatchs.push(unwatchConfig);
|
||||||
}
|
}
|
||||||
|
|
||||||
// delay dev server 启动,避免重复 compile
|
server = await createServer({
|
||||||
// https://github.com/webpack/watchpack/issues/25
|
mode: 'development',
|
||||||
// https://github.com/yessky/webpack-mild-compile
|
plugins: [vue(), SFCConfigBlockPlugin, vueJsx(), viteCommonjs()],
|
||||||
await delay(500);
|
configFile: false,
|
||||||
|
resolve: {
|
||||||
|
alias: {
|
||||||
|
'@': paths.absSrcPath,
|
||||||
|
'@@': paths.absTmpPath,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
server: {
|
||||||
|
port: 8000,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
await server.listen();
|
||||||
|
|
||||||
|
server.printUrls();
|
||||||
|
|
||||||
// dev
|
// dev
|
||||||
const { bundleConfig } = await getBundleAndConfigs({ api });
|
// const { bundleConfig } = await getBundleAndConfigs({ api });
|
||||||
|
|
||||||
|
// const beforeMiddlewares = await api.applyPlugins({
|
||||||
|
// key: 'addBeforeMiddlewares',
|
||||||
|
// type: api.ApplyPluginsType.add,
|
||||||
|
// initialValue: [],
|
||||||
|
// args: {}
|
||||||
|
// });
|
||||||
|
// const middlewares = await api.applyPlugins({
|
||||||
|
// key: 'addMiddlewares',
|
||||||
|
// type: api.ApplyPluginsType.add,
|
||||||
|
// initialValue: [],
|
||||||
|
// args: {}
|
||||||
|
// });
|
||||||
|
// const { startDevServer } = require('./devServer');
|
||||||
|
// server = startDevServer({
|
||||||
|
// webpackConfig: bundleConfig,
|
||||||
|
// host: hostname,
|
||||||
|
// port,
|
||||||
|
// proxy: api.config.proxy,
|
||||||
|
// https: isHTTPS,
|
||||||
|
// beforeMiddlewares: [...beforeMiddlewares, createRouteMiddleware(api)],
|
||||||
|
// afterMiddlewares: [...middlewares],
|
||||||
|
// customerDevServerConfig: api.config.devServer
|
||||||
|
// });
|
||||||
|
|
||||||
const beforeMiddlewares = await api.applyPlugins({
|
|
||||||
key: 'addBeforeMiddlewares',
|
|
||||||
type: api.ApplyPluginsType.add,
|
|
||||||
initialValue: [],
|
|
||||||
args: {}
|
|
||||||
});
|
|
||||||
const middlewares = await api.applyPlugins({
|
|
||||||
key: 'addMiddlewares',
|
|
||||||
type: api.ApplyPluginsType.add,
|
|
||||||
initialValue: [],
|
|
||||||
args: {}
|
|
||||||
});
|
|
||||||
const { startDevServer } = require('./devServer');
|
|
||||||
server = startDevServer({
|
|
||||||
webpackConfig: bundleConfig,
|
|
||||||
host: hostname,
|
|
||||||
port,
|
|
||||||
proxy: api.config.proxy,
|
|
||||||
https: isHTTPS,
|
|
||||||
beforeMiddlewares: [...beforeMiddlewares, createRouteMiddleware(api)],
|
|
||||||
afterMiddlewares: [...middlewares],
|
|
||||||
customerDevServerConfig: api.config.devServer
|
|
||||||
});
|
|
||||||
return {
|
return {
|
||||||
destroy
|
destroy,
|
||||||
};
|
};
|
||||||
}
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
api.registerMethod({
|
api.registerMethod({
|
||||||
name: 'getPort',
|
name: 'getPort',
|
||||||
fn() {
|
fn() {
|
||||||
assert(
|
assert(env === 'development', 'api.getPort() is only valid in development.');
|
||||||
env === 'development',
|
|
||||||
'api.getPort() is only valid in development.'
|
|
||||||
);
|
|
||||||
return port;
|
return port;
|
||||||
}
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
api.registerMethod({
|
api.registerMethod({
|
||||||
name: 'getHostname',
|
name: 'getHostname',
|
||||||
fn() {
|
fn() {
|
||||||
assert(
|
assert(env === 'development', 'api.getHostname() is only valid in development.');
|
||||||
env === 'development',
|
|
||||||
'api.getHostname() is only valid in development.'
|
|
||||||
);
|
|
||||||
return hostname;
|
return hostname;
|
||||||
}
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
api.registerMethod({
|
api.registerMethod({
|
||||||
name: 'getServer',
|
name: 'getServer',
|
||||||
fn() {
|
fn() {
|
||||||
assert(
|
assert(env === 'development', 'api.getServer() is only valid in development.');
|
||||||
env === 'development',
|
|
||||||
'api.getServer() is only valid in development.'
|
|
||||||
);
|
|
||||||
return server;
|
return server;
|
||||||
}
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
api.registerMethod({
|
api.registerMethod({
|
||||||
@ -225,8 +217,8 @@ export default (api) => {
|
|||||||
console.log(chalk.gray('Try to restart dev server...'));
|
console.log(chalk.gray('Try to restart dev server...'));
|
||||||
destroy();
|
destroy();
|
||||||
process.send({
|
process.send({
|
||||||
type: 'RESTART'
|
type: 'RESTART',
|
||||||
});
|
});
|
||||||
}
|
},
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
@ -0,0 +1,3 @@
|
|||||||
|
import { defineComponent } from 'vue';
|
||||||
|
|
||||||
|
export default defineComponent(() => () => (<RouterView></RouterView>));
|
@ -2,13 +2,14 @@
|
|||||||
import {
|
import {
|
||||||
createApp,
|
createApp,
|
||||||
reactive,
|
reactive,
|
||||||
defineComponent
|
|
||||||
} from 'vue';
|
} from 'vue';
|
||||||
import { plugin } from './core/plugin';
|
import { plugin } from './core/plugin';
|
||||||
import './core/pluginRegister';
|
import './core/pluginRegister';
|
||||||
import { ApplyPluginsType } from '{{{ runtimePath }}}';
|
import { ApplyPluginsType } from '{{{ runtimePath }}}';
|
||||||
import { getRoutes } from './core/routes/routes';
|
import { getRoutes } from './core/routes/routes';
|
||||||
|
|
||||||
|
import DefaultContainer from './defaultContainer';
|
||||||
|
|
||||||
{{{ imports }}}
|
{{{ imports }}}
|
||||||
|
|
||||||
{{{ entryCodeAhead }}}
|
{{{ entryCodeAhead }}}
|
||||||
@ -18,7 +19,7 @@ const renderClient = (opts = {}) => {
|
|||||||
const rootContainer = plugin.applyPlugins({
|
const rootContainer = plugin.applyPlugins({
|
||||||
type: ApplyPluginsType.modify,
|
type: ApplyPluginsType.modify,
|
||||||
key: 'rootContainer',
|
key: 'rootContainer',
|
||||||
initialValue: defineComponent(() => () => (<RouterView></RouterView>)),
|
initialValue: DefaultContainer,
|
||||||
args: {
|
args: {
|
||||||
routes: routes,
|
routes: routes,
|
||||||
plugin: plugin
|
plugin: plugin
|
||||||
|
@ -15,7 +15,7 @@ export function importsToStr(imports) {
|
|||||||
|
|
||||||
export default function (api) {
|
export default function (api) {
|
||||||
const {
|
const {
|
||||||
utils: { Mustache }
|
utils: { Mustache },
|
||||||
} = api;
|
} = api;
|
||||||
|
|
||||||
api.onGenerateFiles(async () => {
|
api.onGenerateFiles(async () => {
|
||||||
@ -31,31 +31,37 @@ export default function (api) {
|
|||||||
await api.applyPlugins({
|
await api.applyPlugins({
|
||||||
key: 'addEntryCode',
|
key: 'addEntryCode',
|
||||||
type: api.ApplyPluginsType.add,
|
type: api.ApplyPluginsType.add,
|
||||||
initialValue: []
|
initialValue: [],
|
||||||
})
|
})
|
||||||
).join('\r\n'),
|
).join('\r\n'),
|
||||||
entryCodeAhead: (
|
entryCodeAhead: (
|
||||||
await api.applyPlugins({
|
await api.applyPlugins({
|
||||||
key: 'addEntryCodeAhead',
|
key: 'addEntryCodeAhead',
|
||||||
type: api.ApplyPluginsType.add,
|
type: api.ApplyPluginsType.add,
|
||||||
initialValue: []
|
initialValue: [],
|
||||||
})
|
})
|
||||||
).join('\r\n'),
|
).join('\r\n'),
|
||||||
importsAhead: importsToStr(
|
importsAhead: importsToStr(
|
||||||
await api.applyPlugins({
|
await api.applyPlugins({
|
||||||
key: 'addEntryImportsAhead',
|
key: 'addEntryImportsAhead',
|
||||||
type: api.ApplyPluginsType.add,
|
type: api.ApplyPluginsType.add,
|
||||||
initialValue: []
|
initialValue: [],
|
||||||
})
|
}),
|
||||||
).join('\r\n'),
|
).join('\r\n'),
|
||||||
imports: importsToStr(
|
imports: importsToStr(
|
||||||
await api.applyPlugins({
|
await api.applyPlugins({
|
||||||
key: 'addEntryImports',
|
key: 'addEntryImports',
|
||||||
type: api.ApplyPluginsType.add,
|
type: api.ApplyPluginsType.add,
|
||||||
initialValue: []
|
initialValue: [],
|
||||||
})
|
}),
|
||||||
).join('\r\n')
|
).join('\r\n'),
|
||||||
})
|
}),
|
||||||
|
});
|
||||||
|
|
||||||
|
const defaultContainerName = 'defaultContainer';
|
||||||
|
api.writeTmpFile({
|
||||||
|
path: `${defaultContainerName}.jsx`,
|
||||||
|
content: readFileSync(join(__dirname, `./${defaultContainerName}.tpl`), 'utf-8'),
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
@ -1,7 +1,5 @@
|
|||||||
import { readdirSync, statSync, readFileSync } from 'fs';
|
import { readdirSync, statSync, readFileSync } from 'fs';
|
||||||
import {
|
import { join, extname, posix, basename } from 'path';
|
||||||
join, extname, posix, basename
|
|
||||||
} from 'path';
|
|
||||||
import { lodash, parser, generator } from '@fesjs/utils';
|
import { lodash, parser, generator } from '@fesjs/utils';
|
||||||
import { parse } from '@vue/compiler-sfc';
|
import { parse } from '@vue/compiler-sfc';
|
||||||
import { Logger } from '@fesjs/compiler';
|
import { Logger } from '@fesjs/compiler';
|
||||||
@ -43,16 +41,13 @@ const checkHasLayout = function (path) {
|
|||||||
|
|
||||||
const getRouteName = function (parentRoutePath, fileName) {
|
const getRouteName = function (parentRoutePath, fileName) {
|
||||||
const routeName = posix.join(parentRoutePath, fileName);
|
const routeName = posix.join(parentRoutePath, fileName);
|
||||||
return routeName
|
return routeName.slice(1).replace(/\//g, '_').replace(/@/g, '_').replace(/\*/g, 'FUZZYMATCH');
|
||||||
.slice(1)
|
|
||||||
.replace(/\//g, '_')
|
|
||||||
.replace(/@/g, '_')
|
|
||||||
.replace(/\*/g, 'FUZZYMATCH');
|
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const getPagePathPrefix = (config) => `@/${config.singular ? 'page' : 'pages'}`;
|
||||||
|
|
||||||
const getComponentPath = function (parentRoutePath, fileName, config) {
|
const getComponentPath = function (parentRoutePath, fileName, config) {
|
||||||
const pagesName = config.singular ? 'page' : 'pages';
|
return posix.join(`${getPagePathPrefix(config)}/`, parentRoutePath, fileName);
|
||||||
return posix.join(`@/${pagesName}/`, parentRoutePath, fileName);
|
|
||||||
};
|
};
|
||||||
|
|
||||||
const getRoutePath = function (parentRoutePath, fileName) {
|
const getRoutePath = function (parentRoutePath, fileName) {
|
||||||
@ -74,9 +69,14 @@ const getRoutePath = function (parentRoutePath, fileName) {
|
|||||||
function getRouteMeta(content) {
|
function getRouteMeta(content) {
|
||||||
const ast = parser.parse(content, {
|
const ast = parser.parse(content, {
|
||||||
sourceType: 'module',
|
sourceType: 'module',
|
||||||
plugins: ['jsx', 'typescript']
|
plugins: ['jsx', 'typescript'],
|
||||||
});
|
});
|
||||||
const defineRouteExpression = ast.program.body.filter(expression => expression.type === 'ExpressionStatement' && expression.expression.type === 'CallExpression' && expression.expression.callee.name === 'defineRouteMeta')[0];
|
const defineRouteExpression = ast.program.body.filter(
|
||||||
|
(expression) =>
|
||||||
|
expression.type === 'ExpressionStatement' &&
|
||||||
|
expression.expression.type === 'CallExpression' &&
|
||||||
|
expression.expression.callee.name === 'defineRouteMeta',
|
||||||
|
)[0];
|
||||||
if (defineRouteExpression) {
|
if (defineRouteExpression) {
|
||||||
const argument = generator(defineRouteExpression.expression.arguments[0]);
|
const argument = generator(defineRouteExpression.expression.arguments[0]);
|
||||||
return JSON.parse(argument.code.replace(/'/g, '"').replace(/(\S+):/g, (global, m1) => `"${m1}":`));
|
return JSON.parse(argument.code.replace(/'/g, '"').replace(/(\S+):/g, (global, m1) => `"${m1}":`));
|
||||||
@ -91,7 +91,7 @@ const genRoutes = function (parentRoutes, path, parentRoutePath, config) {
|
|||||||
const dirList = readdirSync(path);
|
const dirList = readdirSync(path);
|
||||||
const hasLayout = checkHasLayout(path);
|
const hasLayout = checkHasLayout(path);
|
||||||
const layoutRoute = {
|
const layoutRoute = {
|
||||||
children: []
|
children: [],
|
||||||
};
|
};
|
||||||
if (hasLayout) {
|
if (hasLayout) {
|
||||||
layoutRoute.path = parentRoutePath;
|
layoutRoute.path = parentRoutePath;
|
||||||
@ -106,22 +106,22 @@ const genRoutes = function (parentRoutes, path, parentRoutePath, config) {
|
|||||||
// 路由的path
|
// 路由的path
|
||||||
const routePath = getRoutePath(parentRoutePath, fileName);
|
const routePath = getRoutePath(parentRoutePath, fileName);
|
||||||
if (cacheGenRoutes[routePath]) {
|
if (cacheGenRoutes[routePath]) {
|
||||||
logger.warn(`[WARNING]: The file path: ${routePath}(.jsx/.tsx/.vue) conflict in router,will only use ${routePath}.tsx or ${routePath}.jsx,please remove one of.`);
|
logger.warn(
|
||||||
|
`[WARNING]: The file path: ${routePath}(.jsx/.tsx/.vue) conflict in router,will only use ${routePath}.tsx or ${routePath}.jsx,please remove one of.`,
|
||||||
|
);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
cacheGenRoutes[routePath] = true;
|
cacheGenRoutes[routePath] = true;
|
||||||
|
|
||||||
// 路由名称
|
// 路由名称
|
||||||
const routeName = getRouteName(parentRoutePath, fileName);
|
const routeName = getRouteName(parentRoutePath, fileName);
|
||||||
const componentPath = getComponentPath(parentRoutePath, fileName, config);
|
const componentPath = getComponentPath(parentRoutePath, ext === '.vue' ? `${fileName}${ext}` : fileName, config);
|
||||||
|
|
||||||
let content = readFileSync(component, 'utf-8');
|
let content = readFileSync(component, 'utf-8');
|
||||||
let routeMeta = {};
|
let routeMeta = {};
|
||||||
if (ext === '.vue') {
|
if (ext === '.vue') {
|
||||||
const { descriptor } = parse(content);
|
const { descriptor } = parse(content);
|
||||||
const routeMetaBlock = descriptor.customBlocks.find(
|
const routeMetaBlock = descriptor.customBlocks.find((b) => b.type === 'config');
|
||||||
b => b.type === 'config'
|
|
||||||
);
|
|
||||||
routeMeta = routeMetaBlock?.content ? JSON.parse(routeMetaBlock.content) : {};
|
routeMeta = routeMetaBlock?.content ? JSON.parse(routeMetaBlock.content) : {};
|
||||||
if (descriptor.script) {
|
if (descriptor.script) {
|
||||||
content = descriptor.script.content;
|
content = descriptor.script.content;
|
||||||
@ -136,7 +136,7 @@ const genRoutes = function (parentRoutes, path, parentRoutePath, config) {
|
|||||||
path: routePath,
|
path: routePath,
|
||||||
component: componentPath,
|
component: componentPath,
|
||||||
name: routeMeta.name || routeName,
|
name: routeMeta.name || routeName,
|
||||||
meta: routeMeta
|
meta: routeMeta,
|
||||||
};
|
};
|
||||||
if (hasLayout) {
|
if (hasLayout) {
|
||||||
if (fileName === 'layout') {
|
if (fileName === 'layout') {
|
||||||
@ -215,18 +215,22 @@ const getRoutes = function ({ config, absPagesPath }) {
|
|||||||
return routes;
|
return routes;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
function genComponentName(component, config) {
|
||||||
|
return lodash.camelCase(component.replace(getPagePathPrefix(config), '').replace('.vue', ''));
|
||||||
|
}
|
||||||
|
|
||||||
|
function isFunctionComponent(component) {
|
||||||
|
// eslint-disable-next-line
|
||||||
|
return (
|
||||||
|
/^\((.+)?\)(\s+)?=>/.test(component)
|
||||||
|
|| /^function([^(]+)?\(([^)]+)?\)([^{]+)?{/.test(component)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
const getRoutesJSON = function ({ routes, config }) {
|
const getRoutesJSON = function ({ routes, config }) {
|
||||||
// 因为要往 routes 里加无用的信息,所以必须 deep clone 一下,避免污染
|
// 因为要往 routes 里加无用的信息,所以必须 deep clone 一下,避免污染
|
||||||
const clonedRoutes = lodash.cloneDeep(routes);
|
const clonedRoutes = lodash.cloneDeep(routes);
|
||||||
|
|
||||||
function isFunctionComponent(component) {
|
|
||||||
// eslint-disable-next-line
|
|
||||||
return (
|
|
||||||
/^\((.+)?\)(\s+)?=>/.test(component)
|
|
||||||
|| /^function([^(]+)?\(([^)]+)?\)([^{]+)?{/.test(component)
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
function replacer(key, value) {
|
function replacer(key, value) {
|
||||||
switch (key) {
|
switch (key) {
|
||||||
case 'component':
|
case 'component':
|
||||||
@ -235,36 +239,51 @@ const getRoutesJSON = function ({ routes, config }) {
|
|||||||
// TODO 针对目录进行 chunk 划分,import(/* webpackChunkName: "group-user" */ './UserDetails.vue')
|
// TODO 针对目录进行 chunk 划分,import(/* webpackChunkName: "group-user" */ './UserDetails.vue')
|
||||||
return `() => import('${value}')`;
|
return `() => import('${value}')`;
|
||||||
}
|
}
|
||||||
return `require('${value}').default`;
|
return genComponentName(value, config);
|
||||||
default:
|
default:
|
||||||
return value;
|
return value;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return JSON.stringify(clonedRoutes, replacer, 2)
|
return JSON.stringify(clonedRoutes, replacer, 2)
|
||||||
.replace(
|
.replace(/"component": ("(.+?)")/g, (global, m1, m2) => `"component": ${m2.replace(/\^/g, '"')}`)
|
||||||
/"component": ("(.+?)")/g,
|
|
||||||
(global, m1, m2) => `"component": ${m2.replace(/\^/g, '"')}`
|
|
||||||
)
|
|
||||||
.replace(/\\r\\n/g, '\r\n')
|
.replace(/\\r\\n/g, '\r\n')
|
||||||
.replace(/\\n/g, '\r\n');
|
.replace(/\\n/g, '\r\n');
|
||||||
};
|
};
|
||||||
|
|
||||||
|
function genComponentImportExpression(routes, config) {
|
||||||
|
if (config.dynamicImport) {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
|
||||||
|
const result = [];
|
||||||
|
for (const routeConfig of routes) {
|
||||||
|
if (routeConfig.children) {
|
||||||
|
result.push(...genComponentImportExpression(routeConfig.children, config));
|
||||||
|
}
|
||||||
|
|
||||||
|
if (routeConfig.component && !isFunctionComponent(routeConfig.component)) {
|
||||||
|
result.push(`import ${genComponentName(routeConfig.component, config)} from '${routeConfig.component}';`);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
export default function (api) {
|
export default function (api) {
|
||||||
api.describe({
|
api.describe({
|
||||||
key: 'router',
|
key: 'router',
|
||||||
config: {
|
config: {
|
||||||
schema(joi) {
|
schema(joi) {
|
||||||
return joi
|
return joi.object({
|
||||||
.object({
|
routes: joi.array(),
|
||||||
routes: joi.array(),
|
mode: joi.string(),
|
||||||
mode: joi.string()
|
});
|
||||||
});
|
|
||||||
},
|
},
|
||||||
default: {
|
default: {
|
||||||
mode: 'hash'
|
mode: 'hash',
|
||||||
}
|
},
|
||||||
}
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
api.registerMethod({
|
api.registerMethod({
|
||||||
@ -275,22 +294,21 @@ export default function (api) {
|
|||||||
type: api.ApplyPluginsType.modify,
|
type: api.ApplyPluginsType.modify,
|
||||||
initialValue: getRoutes({
|
initialValue: getRoutes({
|
||||||
config: api.config,
|
config: api.config,
|
||||||
absPagesPath: api.paths.absPagesPath
|
absPagesPath: api.paths.absPagesPath,
|
||||||
})
|
}),
|
||||||
});
|
});
|
||||||
}
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
api.registerMethod({
|
api.registerMethod({
|
||||||
name: 'getRoutesJSON',
|
name: 'getRoutesJSON',
|
||||||
async fn() {
|
async fn(routes) {
|
||||||
const routes = await api.getRoutes();
|
return getRoutesJSON({ routes: await (routes || api.getRoutes()), config: api.config });
|
||||||
return getRoutesJSON({ routes, config: api.config });
|
},
|
||||||
}
|
|
||||||
});
|
});
|
||||||
|
|
||||||
const {
|
const {
|
||||||
utils: { Mustache }
|
utils: { Mustache },
|
||||||
} = api;
|
} = api;
|
||||||
|
|
||||||
const namespace = 'core/routes';
|
const namespace = 'core/routes';
|
||||||
@ -302,35 +320,36 @@ export default function (api) {
|
|||||||
const historyType = {
|
const historyType = {
|
||||||
history: 'createWebHistory',
|
history: 'createWebHistory',
|
||||||
hash: 'createWebHashHistory',
|
hash: 'createWebHashHistory',
|
||||||
memory: 'createMemoryHistory'
|
memory: 'createMemoryHistory',
|
||||||
};
|
};
|
||||||
|
|
||||||
api.onGenerateFiles(async () => {
|
api.onGenerateFiles(async () => {
|
||||||
const routesTpl = readFileSync(join(__dirname, 'template/routes.tpl'), 'utf-8');
|
const routesTpl = readFileSync(join(__dirname, 'template/routes.tpl'), 'utf-8');
|
||||||
const routes = await api.getRoutesJSON();
|
const routes = await api.getRoutes();
|
||||||
|
|
||||||
api.writeTmpFile({
|
api.writeTmpFile({
|
||||||
path: absCoreFilePath,
|
path: absCoreFilePath,
|
||||||
content: Mustache.render(routesTpl, {
|
content: Mustache.render(routesTpl, {
|
||||||
runtimePath,
|
runtimePath,
|
||||||
routes,
|
COMPONENTS_IMPORT: genComponentImportExpression(routes, api.config).join('\n'),
|
||||||
|
routes: await api.getRoutesJSON(),
|
||||||
config: api.config,
|
config: api.config,
|
||||||
routerBase: api.config.base,
|
routerBase: api.config.base,
|
||||||
CREATE_HISTORY: historyType[api.config.router.mode] || 'createWebHashHistory'
|
CREATE_HISTORY: historyType[api.config.router.mode] || 'createWebHashHistory',
|
||||||
})
|
}),
|
||||||
});
|
});
|
||||||
|
|
||||||
api.writeTmpFile({
|
api.writeTmpFile({
|
||||||
path: absRuntimeFilePath,
|
path: absRuntimeFilePath,
|
||||||
content: readFileSync(join(__dirname, 'template/runtime.tpl'), 'utf-8')
|
content: readFileSync(join(__dirname, 'template/runtime.tpl'), 'utf-8'),
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
api.addCoreExports(() => [
|
api.addCoreExports(() => [
|
||||||
{
|
{
|
||||||
specifiers: ['getRoutes', 'getRouter', 'getHistory', 'destroyRouter', 'defineRouteMeta'],
|
specifiers: ['getRoutes', 'getRouter', 'getHistory', 'destroyRouter', 'defineRouteMeta'],
|
||||||
source: absCoreFilePath
|
source: absCoreFilePath,
|
||||||
}
|
},
|
||||||
]);
|
]);
|
||||||
|
|
||||||
api.addRuntimePlugin(() => `@@/${absRuntimeFilePath}`);
|
api.addRuntimePlugin(() => `@@/${absRuntimeFilePath}`);
|
||||||
|
@ -1,6 +1,8 @@
|
|||||||
import { createRouter as createVueRouter, {{{ CREATE_HISTORY }}}, ApplyPluginsType } from '{{{ runtimePath }}}';
|
import { createRouter as createVueRouter, {{{ CREATE_HISTORY }}}, ApplyPluginsType } from '{{{ runtimePath }}}';
|
||||||
import { plugin } from '@@/core/coreExports';
|
import { plugin } from '@@/core/coreExports';
|
||||||
|
|
||||||
|
{{{ COMPONENTS_IMPORT }}}
|
||||||
|
|
||||||
export function getRoutes() {
|
export function getRoutes() {
|
||||||
const routes = {{{ routes }}};
|
const routes = {{{ routes }}};
|
||||||
return routes;
|
return routes;
|
||||||
|
@ -9,7 +9,6 @@ export default {
|
|||||||
},
|
},
|
||||||
publicPath: '/',
|
publicPath: '/',
|
||||||
request: {
|
request: {
|
||||||
base: '/ras-mas',
|
|
||||||
dataField: 'result'
|
dataField: 'result'
|
||||||
},
|
},
|
||||||
html: {
|
html: {
|
||||||
@ -35,15 +34,15 @@ export default {
|
|||||||
devServer: {
|
devServer: {
|
||||||
port: 8000
|
port: 8000
|
||||||
},
|
},
|
||||||
windicss: {
|
// windicss: {
|
||||||
config: {
|
// config: {
|
||||||
theme: {
|
// theme: {
|
||||||
extend: {
|
// extend: {
|
||||||
colors: {
|
// colors: {
|
||||||
green: '#7cb305'
|
// green: '#7cb305'
|
||||||
}
|
// }
|
||||||
}
|
// }
|
||||||
}
|
// }
|
||||||
}
|
// }
|
||||||
}
|
// }
|
||||||
};
|
};
|
||||||
|
25
packages/fes-template-h5/index.html
Normal file
25
packages/fes-template-h5/index.html
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
<!DOCTYPE html>
|
||||||
|
<html lang="en">
|
||||||
|
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8">
|
||||||
|
<meta http-equiv="X-UA-Compatible" content="IE=edge">
|
||||||
|
<meta name="apple-mobile-web-app-capable" content="yes" />
|
||||||
|
<meta name="apple-mobile-web-app-status-bar-style" content="black" />
|
||||||
|
<meta name="format-detection" content="telephone=no" />
|
||||||
|
<meta name="format-detection" content="email=no" />
|
||||||
|
<meta name="viewport"
|
||||||
|
content="viewport-fit=cover,width=device-width,initial-scale=1,maximum-scale=1,minimum-scale=1,user-scalable=no">
|
||||||
|
<title>
|
||||||
|
<%= htmlWebpackPlugin.options.title %>
|
||||||
|
</title>
|
||||||
|
<link rel="shortcut icon" type="image/x-icon" href="./logo.png">
|
||||||
|
</head>
|
||||||
|
|
||||||
|
<body ontouchstart="">
|
||||||
|
<div id="app"></div>
|
||||||
|
<script type="module" src="/src/.fes/fes.js"></script>
|
||||||
|
<!-- built files will be auto injected -->
|
||||||
|
</body>
|
||||||
|
|
||||||
|
</html>
|
@ -40,7 +40,6 @@
|
|||||||
"access": "public"
|
"access": "public"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@webank/eslint-config-webank": "0.3.1",
|
|
||||||
"@ttou/postcss-px-to-viewport": "1.1.4",
|
"@ttou/postcss-px-to-viewport": "1.1.4",
|
||||||
"@vue/compiler-sfc": "^3.2.2"
|
"@vue/compiler-sfc": "^3.2.2"
|
||||||
},
|
},
|
||||||
@ -48,7 +47,6 @@
|
|||||||
"@fesjs/fes": "^2.0.0",
|
"@fesjs/fes": "^2.0.0",
|
||||||
"@fesjs/plugin-icon": "^2.0.0",
|
"@fesjs/plugin-icon": "^2.0.0",
|
||||||
"@fesjs/plugin-request": "^2.0.0",
|
"@fesjs/plugin-request": "^2.0.0",
|
||||||
"@fesjs/plugin-windicss": "^2.0.0",
|
|
||||||
"vue": "^3.2.2"
|
"vue": "^3.2.2"
|
||||||
},
|
},
|
||||||
"private": true
|
"private": true
|
||||||
|
@ -1,3 +0,0 @@
|
|||||||
<template>
|
|
||||||
<div>hello vue</div>
|
|
||||||
</template>
|
|
@ -1,12 +1,7 @@
|
|||||||
<template>
|
<template>
|
||||||
<div class="onepiece m-10px text-green">
|
<div class="onepiece m-10px text-green">
|
||||||
fes h5 & 拉夫德鲁<br />
|
fes h5 & 拉夫德鲁<br />
|
||||||
<fes-icon
|
<fes-icon :spin="true" class="one-icon" type="smile" @click="clickIcon" />
|
||||||
:spin="true"
|
|
||||||
class="one-icon"
|
|
||||||
type="smile"
|
|
||||||
@click="clickIcon"
|
|
||||||
/>
|
|
||||||
<HelloWorld />
|
<HelloWorld />
|
||||||
<HelloTSX />
|
<HelloTSX />
|
||||||
<helloTS />
|
<helloTS />
|
||||||
@ -15,21 +10,21 @@
|
|||||||
<script>
|
<script>
|
||||||
import { ref } from 'vue';
|
import { ref } from 'vue';
|
||||||
import { request, defineRouteMeta, useRoute } from '@fesjs/fes';
|
import { request, defineRouteMeta, useRoute } from '@fesjs/fes';
|
||||||
import HelloWorld from '@/components/helloWorld';
|
import HelloWorld from '@/components/helloWorld.vue';
|
||||||
import HelloTSX from '@/components/helloTSX';
|
import HelloTSX from '@/components/helloTSX.vue';
|
||||||
import helloTS from '@/components/helloTS';
|
import helloTS from '@/components/helloTS.vue';
|
||||||
|
|
||||||
defineRouteMeta({
|
defineRouteMeta({
|
||||||
title: '首页',
|
title: '首页',
|
||||||
name: 'testIndex',
|
name: 'testIndex',
|
||||||
layout: false
|
layout: false,
|
||||||
});
|
});
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
components: {
|
components: {
|
||||||
HelloWorld,
|
HelloWorld,
|
||||||
HelloTSX,
|
HelloTSX,
|
||||||
helloTS
|
helloTS,
|
||||||
},
|
},
|
||||||
setup() {
|
setup() {
|
||||||
const fes = ref('fes upgrade to vue3');
|
const fes = ref('fes upgrade to vue3');
|
||||||
@ -68,8 +63,8 @@ export default {
|
|||||||
'/get/api',
|
'/get/api',
|
||||||
{ id },
|
{ id },
|
||||||
{
|
{
|
||||||
method: 'get'
|
method: 'get',
|
||||||
}
|
},
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -78,8 +73,8 @@ export default {
|
|||||||
'/api',
|
'/api',
|
||||||
{ id },
|
{ id },
|
||||||
{
|
{
|
||||||
responseType: 'blob'
|
responseType: 'blob',
|
||||||
}
|
},
|
||||||
).then((data) => {
|
).then((data) => {
|
||||||
console.log(data);
|
console.log(data);
|
||||||
});
|
});
|
||||||
@ -143,15 +138,15 @@ export default {
|
|||||||
return {
|
return {
|
||||||
fes,
|
fes,
|
||||||
rotate,
|
rotate,
|
||||||
clickIcon
|
clickIcon,
|
||||||
};
|
};
|
||||||
}
|
},
|
||||||
};
|
};
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style lang="less" scoped>
|
<style lang="less" scoped>
|
||||||
@import '~@/styles/mixins/hairline';
|
@import '@/styles/mixins/hairline';
|
||||||
@import '~@/styles/mixins/hover';
|
@import '@/styles/mixins/hover';
|
||||||
|
|
||||||
div {
|
div {
|
||||||
padding: 20px;
|
padding: 20px;
|
||||||
|
@ -42,9 +42,6 @@
|
|||||||
"publishConfig": {
|
"publishConfig": {
|
||||||
"access": "public"
|
"access": "public"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
|
||||||
"@webank/eslint-config-webank": "0.3.1"
|
|
||||||
},
|
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@fesjs/fes": "^2.0.0",
|
"@fesjs/fes": "^2.0.0",
|
||||||
"@fesjs/plugin-access": "^2.0.0",
|
"@fesjs/plugin-access": "^2.0.0",
|
||||||
@ -66,4 +63,4 @@
|
|||||||
"pinia": "^2.0.11"
|
"pinia": "^2.0.11"
|
||||||
},
|
},
|
||||||
"private": true
|
"private": true
|
||||||
}
|
}
|
Loading…
x
Reference in New Issue
Block a user