mirror of
https://github.com/WeBankFinTech/fes.js.git
synced 2025-04-05 19:41:57 +08:00
feat: 页面支持tsx,提供defineRoute配置tsx和jsx的route (#106)
* feat: 页面支持tsx,提供defineRoute配置tsx和jsx的route * .vue文件也支持defineRouteMeta * feat: 改回声明用法
This commit is contained in:
parent
c87cecf0a9
commit
930541e726
@ -2,7 +2,7 @@ import { readdirSync, statSync, readFileSync } from 'fs';
|
||||
import {
|
||||
join, extname, posix, basename
|
||||
} from 'path';
|
||||
import { lodash } from '@fesjs/utils';
|
||||
import { lodash, parser, generator } from '@fesjs/utils';
|
||||
import { parse } from '@vue/compiler-sfc';
|
||||
import { Logger } from '@fesjs/compiler';
|
||||
import { runtimePath } from '../../../utils/constants';
|
||||
@ -21,7 +21,7 @@ const logger = new Logger('fes:router');
|
||||
|
||||
const isProcessFile = function (path) {
|
||||
const ext = extname(path);
|
||||
return statSync(path).isFile() && ['.vue', '.jsx'].includes(ext);
|
||||
return statSync(path).isFile() && ['.vue', '.jsx', '.tsx'].includes(ext);
|
||||
};
|
||||
|
||||
const isProcessDirectory = function (path, item) {
|
||||
@ -71,6 +71,19 @@ const getRoutePath = function (parentRoutePath, fileName) {
|
||||
return posix.join(parentRoutePath, fileName);
|
||||
};
|
||||
|
||||
function getRouteMeta(content) {
|
||||
const ast = parser.parse(content, {
|
||||
sourceType: 'module',
|
||||
plugins: ['jsx', 'typescript']
|
||||
});
|
||||
const defineRouteExpression = ast.program.body.filter(expression => expression.type === 'ExpressionStatement' && expression.expression.type === 'CallExpression' && expression.expression.callee.name === 'defineRouteMeta')[0];
|
||||
if (defineRouteExpression) {
|
||||
const argument = generator(defineRouteExpression.expression.arguments[0]);
|
||||
return JSON.parse(argument.code.replace(/'/g, '"').replace(/(\S+):/g, (global, m1) => `"${m1}":`));
|
||||
}
|
||||
return null;
|
||||
}
|
||||
|
||||
let cacheGenRoutes = {};
|
||||
|
||||
// TODO 约定 layout 目录作为布局文件夹,
|
||||
@ -88,16 +101,12 @@ const genRoutes = function (parentRoutes, path, parentRoutePath, config) {
|
||||
// 文件或者目录的绝对路径
|
||||
const component = join(path, item);
|
||||
if (isProcessFile(component)) {
|
||||
const { descriptor } = parse(readFileSync(component, 'utf-8'));
|
||||
const routeMetaBlock = descriptor.customBlocks.find(
|
||||
b => b.type === 'config'
|
||||
);
|
||||
const ext = extname(item);
|
||||
const fileName = basename(item, ext);
|
||||
// 路由的path
|
||||
const routePath = getRoutePath(parentRoutePath, fileName);
|
||||
if (cacheGenRoutes[routePath]) {
|
||||
logger.warn(`[WARNING]: The file path: ${routePath}(.jsx/.vue) conflict in router,will only use ${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;
|
||||
}
|
||||
cacheGenRoutes[routePath] = true;
|
||||
@ -105,7 +114,24 @@ const genRoutes = function (parentRoutes, path, parentRoutePath, config) {
|
||||
// 路由名称
|
||||
const routeName = getRouteName(parentRoutePath, fileName);
|
||||
const componentPath = getComponentPath(parentRoutePath, fileName, config);
|
||||
const routeMeta = routeMetaBlock?.content ? JSON.parse(routeMetaBlock.content) : {};
|
||||
|
||||
let content = readFileSync(component, 'utf-8');
|
||||
let routeMeta = {};
|
||||
if (ext === '.vue') {
|
||||
const { descriptor } = parse(content);
|
||||
const routeMetaBlock = descriptor.customBlocks.find(
|
||||
b => b.type === 'config'
|
||||
);
|
||||
routeMeta = routeMetaBlock?.content ? JSON.parse(routeMetaBlock.content) : {};
|
||||
if (descriptor.script) {
|
||||
content = descriptor.script.content;
|
||||
routeMeta = getRouteMeta(content) || routeMeta;
|
||||
}
|
||||
}
|
||||
if (ext === '.jsx' || ext === '.tsx') {
|
||||
routeMeta = getRouteMeta(content) || {};
|
||||
}
|
||||
|
||||
const routeConfig = {
|
||||
path: routePath,
|
||||
component: componentPath,
|
||||
@ -302,7 +328,7 @@ export default function (api) {
|
||||
|
||||
api.addCoreExports(() => [
|
||||
{
|
||||
specifiers: ['getRoutes', 'getRouter', 'getHistory', 'destroyRouter'],
|
||||
specifiers: ['getRoutes', 'getRouter', 'getHistory', 'destroyRouter', 'defineRouteMeta'],
|
||||
source: absCoreFilePath
|
||||
}
|
||||
]);
|
||||
|
@ -60,3 +60,7 @@ export const destroyRouter = ()=>{
|
||||
router = null;
|
||||
history = null;
|
||||
}
|
||||
|
||||
export const defineRouteMeta = (param)=>{
|
||||
return param
|
||||
}
|
||||
|
@ -1,26 +1,30 @@
|
||||
<template>
|
||||
<div class="onepiece m-10px text-green">
|
||||
fes h5 & 拉夫德鲁<br />
|
||||
<fes-icon :spin="true" class="one-icon" type="smile" @click="clickIcon" />
|
||||
<fes-icon
|
||||
:spin="true"
|
||||
class="one-icon"
|
||||
type="smile"
|
||||
@click="clickIcon"
|
||||
/>
|
||||
<HelloWorld />
|
||||
<HelloTSX />
|
||||
<helloTS />
|
||||
</div>
|
||||
</template>
|
||||
<config>
|
||||
{
|
||||
"title": "首页",
|
||||
"name": "testIndex",
|
||||
"layout": "false"
|
||||
}
|
||||
</config>
|
||||
<script>
|
||||
import { ref } from 'vue';
|
||||
import { request } from '@fesjs/fes';
|
||||
import { request, defineRouteMeta, useRoute } from '@fesjs/fes';
|
||||
import HelloWorld from '@/components/helloWorld';
|
||||
import HelloTSX from '@/components/helloTSX';
|
||||
import helloTS from '@/components/helloTS';
|
||||
|
||||
defineRouteMeta({
|
||||
title: '首页',
|
||||
name: 'testIndex',
|
||||
layout: false
|
||||
});
|
||||
|
||||
export default {
|
||||
components: {
|
||||
HelloWorld,
|
||||
@ -33,6 +37,7 @@ export default {
|
||||
const clickIcon = () => {
|
||||
console.log('click Icon');
|
||||
};
|
||||
console.log(useRoute());
|
||||
// request('/api', null, {
|
||||
// }).then((res) => {
|
||||
// console.log(res);
|
||||
@ -59,15 +64,23 @@ export default {
|
||||
// });
|
||||
|
||||
const get = (id) => {
|
||||
request('/get/api', { id }, {
|
||||
method: 'get'
|
||||
});
|
||||
request(
|
||||
'/get/api',
|
||||
{ id },
|
||||
{
|
||||
method: 'get'
|
||||
}
|
||||
);
|
||||
};
|
||||
|
||||
const post = (id) => {
|
||||
request('/api', { id }, {
|
||||
responseType: 'blob'
|
||||
}).then((data) => {
|
||||
request(
|
||||
'/api',
|
||||
{ id },
|
||||
{
|
||||
responseType: 'blob'
|
||||
}
|
||||
).then((data) => {
|
||||
console.log(data);
|
||||
});
|
||||
};
|
||||
@ -80,7 +93,6 @@ export default {
|
||||
// post(2);
|
||||
post(3);
|
||||
|
||||
|
||||
// setTimeout(() => {
|
||||
// request('/api', null, {
|
||||
// throttle: 3000,
|
||||
@ -138,8 +150,8 @@ export default {
|
||||
</script>
|
||||
|
||||
<style lang="less" scoped>
|
||||
@import "~@/styles/mixins/hairline";
|
||||
@import "~@/styles/mixins/hover";
|
||||
@import '~@/styles/mixins/hairline';
|
||||
@import '~@/styles/mixins/hover';
|
||||
|
||||
div {
|
||||
padding: 20px;
|
||||
@ -154,6 +166,6 @@ div {
|
||||
}
|
||||
.onepiece {
|
||||
text-align: center;
|
||||
.hairline("top");
|
||||
.hairline('top');
|
||||
}
|
||||
</style>
|
||||
|
@ -5,7 +5,7 @@
|
||||
<config>
|
||||
{
|
||||
"title": "onepiece",
|
||||
"layout": "true"
|
||||
"layout": true
|
||||
}
|
||||
</config>
|
||||
<script>
|
||||
|
@ -1,7 +0,0 @@
|
||||
import { defineComponent } from 'vue';
|
||||
|
||||
export default defineComponent({
|
||||
setup() {
|
||||
return () => <div>hello jsx</div>;
|
||||
}
|
||||
});
|
15
packages/fes-template-h5/src/pages/test.tsx
Normal file
15
packages/fes-template-h5/src/pages/test.tsx
Normal file
@ -0,0 +1,15 @@
|
||||
import { defineRouteMeta, useRoute } from '@fesjs/fes';
|
||||
import { defineComponent } from 'vue';
|
||||
|
||||
defineRouteMeta({
|
||||
title: 'test',
|
||||
name: 'test'
|
||||
})
|
||||
|
||||
export default defineComponent({
|
||||
setup() {
|
||||
const route = useRoute();
|
||||
console.log(route)
|
||||
return () => <div>hello tsx</div>;
|
||||
}
|
||||
});
|
@ -25,6 +25,7 @@
|
||||
},
|
||||
"dependencies": {
|
||||
"@babel/parser": "^7.15.0",
|
||||
"@babel/generator": "^7.15.0",
|
||||
"@babel/traverse": "^7.15.0",
|
||||
"chalk": "^4.1.2",
|
||||
"chokidar": "^3.5.2",
|
||||
|
@ -8,6 +8,7 @@ import glob from 'glob';
|
||||
import createDebug from 'debug';
|
||||
import * as parser from '@babel/parser';
|
||||
import traverse from '@babel/traverse';
|
||||
import generator from '@babel/generator';
|
||||
import rimraf from 'rimraf';
|
||||
import mkdirp from 'mkdirp';
|
||||
import pkgUp from 'pkg-up';
|
||||
@ -40,7 +41,8 @@ export {
|
||||
traverse,
|
||||
pkgUp,
|
||||
portfinder,
|
||||
resolve
|
||||
resolve,
|
||||
generator
|
||||
};
|
||||
|
||||
export {
|
||||
|
7
packages/fes/types.d.ts
vendored
7
packages/fes/types.d.ts
vendored
@ -1,5 +1,10 @@
|
||||
|
||||
// @ts-ignore
|
||||
export * from '@@/core/coreExports';
|
||||
// @ts-ignore
|
||||
export * from '@@/core/pluginExports';
|
||||
|
||||
export declare function defineRouteMeta(routeMeta: {
|
||||
name?: string;
|
||||
title?: string;
|
||||
layout?: boolean | { sidebar?: boolean; header?: boolean; logo?: boolean };
|
||||
});
|
||||
|
Loading…
x
Reference in New Issue
Block a user