feat: 页面支持tsx,提供defineRoute配置tsx和jsx的route (#106)

* feat: 页面支持tsx,提供defineRoute配置tsx和jsx的route

* .vue文件也支持defineRouteMeta

* feat: 改回声明用法
This commit is contained in:
harrywan 2022-03-09 10:27:11 +08:00 committed by GitHub
parent c87cecf0a9
commit 930541e726
9 changed files with 96 additions and 38 deletions

View File

@ -2,7 +2,7 @@ import { readdirSync, statSync, readFileSync } from 'fs';
import { import {
join, extname, posix, basename join, extname, posix, basename
} from 'path'; } from 'path';
import { lodash } 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';
import { runtimePath } from '../../../utils/constants'; import { runtimePath } from '../../../utils/constants';
@ -21,7 +21,7 @@ const logger = new Logger('fes:router');
const isProcessFile = function (path) { const isProcessFile = function (path) {
const ext = extname(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) { const isProcessDirectory = function (path, item) {
@ -71,6 +71,19 @@ const getRoutePath = function (parentRoutePath, fileName) {
return posix.join(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 = {}; let cacheGenRoutes = {};
// TODO 约定 layout 目录作为布局文件夹, // TODO 约定 layout 目录作为布局文件夹,
@ -88,16 +101,12 @@ const genRoutes = function (parentRoutes, path, parentRoutePath, config) {
// 文件或者目录的绝对路径 // 文件或者目录的绝对路径
const component = join(path, item); const component = join(path, item);
if (isProcessFile(component)) { if (isProcessFile(component)) {
const { descriptor } = parse(readFileSync(component, 'utf-8'));
const routeMetaBlock = descriptor.customBlocks.find(
b => b.type === 'config'
);
const ext = extname(item); const ext = extname(item);
const fileName = basename(item, ext); const fileName = basename(item, ext);
// 路由的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/.vue) conflict in routerwill only use ${routePath}.jsxplease remove one of.`); logger.warn(`[WARNING]: The file path: ${routePath}(.jsx/.tsx/.vue) conflict in routerwill only use ${routePath}.tsx or ${routePath}.jsxplease remove one of.`);
return; return;
} }
cacheGenRoutes[routePath] = true; cacheGenRoutes[routePath] = true;
@ -105,7 +114,24 @@ const genRoutes = function (parentRoutes, path, parentRoutePath, config) {
// 路由名称 // 路由名称
const routeName = getRouteName(parentRoutePath, fileName); const routeName = getRouteName(parentRoutePath, fileName);
const componentPath = getComponentPath(parentRoutePath, fileName, config); 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 = { const routeConfig = {
path: routePath, path: routePath,
component: componentPath, component: componentPath,
@ -302,7 +328,7 @@ export default function (api) {
api.addCoreExports(() => [ api.addCoreExports(() => [
{ {
specifiers: ['getRoutes', 'getRouter', 'getHistory', 'destroyRouter'], specifiers: ['getRoutes', 'getRouter', 'getHistory', 'destroyRouter', 'defineRouteMeta'],
source: absCoreFilePath source: absCoreFilePath
} }
]); ]);

View File

@ -60,3 +60,7 @@ export const destroyRouter = ()=>{
router = null; router = null;
history = null; history = null;
} }
export const defineRouteMeta = (param)=>{
return param
}

View File

@ -1,26 +1,30 @@
<template> <template>
<div class="onepiece m-10px text-green"> <div class="onepiece m-10px text-green">
fes h5 & 拉夫德鲁<br /> 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 /> <HelloWorld />
<HelloTSX /> <HelloTSX />
<helloTS /> <helloTS />
</div> </div>
</template> </template>
<config>
{
"title": "首页",
"name": "testIndex",
"layout": "false"
}
</config>
<script> <script>
import { ref } from 'vue'; import { ref } from 'vue';
import { request } from '@fesjs/fes'; import { request, defineRouteMeta, useRoute } from '@fesjs/fes';
import HelloWorld from '@/components/helloWorld'; import HelloWorld from '@/components/helloWorld';
import HelloTSX from '@/components/helloTSX'; import HelloTSX from '@/components/helloTSX';
import helloTS from '@/components/helloTS'; import helloTS from '@/components/helloTS';
defineRouteMeta({
title: '首页',
name: 'testIndex',
layout: false
});
export default { export default {
components: { components: {
HelloWorld, HelloWorld,
@ -33,6 +37,7 @@ export default {
const clickIcon = () => { const clickIcon = () => {
console.log('click Icon'); console.log('click Icon');
}; };
console.log(useRoute());
// request('/api', null, { // request('/api', null, {
// }).then((res) => { // }).then((res) => {
// console.log(res); // console.log(res);
@ -59,15 +64,23 @@ export default {
// }); // });
const get = (id) => { const get = (id) => {
request('/get/api', { id }, { request(
method: 'get' '/get/api',
}); { id },
{
method: 'get'
}
);
}; };
const post = (id) => { const post = (id) => {
request('/api', { id }, { request(
responseType: 'blob' '/api',
}).then((data) => { { id },
{
responseType: 'blob'
}
).then((data) => {
console.log(data); console.log(data);
}); });
}; };
@ -80,7 +93,6 @@ export default {
// post(2); // post(2);
post(3); post(3);
// setTimeout(() => { // setTimeout(() => {
// request('/api', null, { // request('/api', null, {
// throttle: 3000, // throttle: 3000,
@ -138,8 +150,8 @@ export default {
</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;
@ -154,6 +166,6 @@ div {
} }
.onepiece { .onepiece {
text-align: center; text-align: center;
.hairline("top"); .hairline('top');
} }
</style> </style>

View File

@ -5,7 +5,7 @@
<config> <config>
{ {
"title": "onepiece", "title": "onepiece",
"layout": "true" "layout": true
} }
</config> </config>
<script> <script>

View File

@ -1,7 +0,0 @@
import { defineComponent } from 'vue';
export default defineComponent({
setup() {
return () => <div>hello jsx</div>;
}
});

View 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>;
}
});

View File

@ -25,6 +25,7 @@
}, },
"dependencies": { "dependencies": {
"@babel/parser": "^7.15.0", "@babel/parser": "^7.15.0",
"@babel/generator": "^7.15.0",
"@babel/traverse": "^7.15.0", "@babel/traverse": "^7.15.0",
"chalk": "^4.1.2", "chalk": "^4.1.2",
"chokidar": "^3.5.2", "chokidar": "^3.5.2",

View File

@ -8,6 +8,7 @@ import glob from 'glob';
import createDebug from 'debug'; import createDebug from 'debug';
import * as parser from '@babel/parser'; import * as parser from '@babel/parser';
import traverse from '@babel/traverse'; import traverse from '@babel/traverse';
import generator from '@babel/generator';
import rimraf from 'rimraf'; import rimraf from 'rimraf';
import mkdirp from 'mkdirp'; import mkdirp from 'mkdirp';
import pkgUp from 'pkg-up'; import pkgUp from 'pkg-up';
@ -40,7 +41,8 @@ export {
traverse, traverse,
pkgUp, pkgUp,
portfinder, portfinder,
resolve resolve,
generator
}; };
export { export {

View File

@ -1,5 +1,10 @@
// @ts-ignore // @ts-ignore
export * from '@@/core/coreExports'; export * from '@@/core/coreExports';
// @ts-ignore // @ts-ignore
export * from '@@/core/pluginExports'; export * from '@@/core/pluginExports';
export declare function defineRouteMeta(routeMeta: {
name?: string;
title?: string;
layout?: boolean | { sidebar?: boolean; header?: boolean; logo?: boolean };
});