介绍
本示例将介绍如何使用装饰器和插件,自动生成动态路由表,并通过动态路由跳转到模块中的页面,以及如何使用动态import的方式加载模块
使用说明
- 自定义装饰器
- 添加装饰器和插件配置文件,编译时自动生成动态路由表
- 配置动态路由,通过WrapBuilder接口,动态创建页面并跳转。
- 动态import变量表达式,需要DevEco Studio NEXT Developer Preview1(4.1.3.500)版本IDE,配合hvigor 4.0.2版本使用。
- 支持自定义路由栈管理,相关内容请参考路由来源页的相关说明
实现思路
动态路由的实现
1.初始化动态路由
public static routerInit(config: RouterConfig, context: Context) {
DynamicsRouter.config = config;
DynamicsRouter.appRouterStack.push(HOME_PAGE);
RouterLoader.load(config.mapPath, DynamicsRouter.routerMap, context);
}
2.获取路由
private static getNavPathStack(): NavPathStack {
return DynamicsRouter.navPathStack;
}
3.通过builderName,注册WrappedBuilder对象,用于动态创建页面
private static registerBuilder(builderName: string, builder: WrappedBuilder<[object]>): void {
DynamicsRouter.builderMap.set(builderName, builder);
}
4.通过builderName,获取注册的WrappedBuilder对象
public static getBuilder(builderName: string): WrappedBuilder<[object]> {
let builder = DynamicsRouter.builderMap.get(builderName);
if (!builder) {
let msg = "not found builder";
console.info(msg + builderName);
}
return builder as WrappedBuilder<[object]>;
}
5.通过页面栈跳转到指定页面
public static pushUri(name: string, param?: Object) {
if (!DynamicsRouter.routerMap.has(name)) {
return;
}
let routerInfo: AppRouterInfo = DynamicsRouter.routerMap.get(name)!;
if (!DynamicsRouter.builderMap.has(name)) {
import(`${DynamicsRouter.config.libPrefix}/${routerInfo.pageModule}`)
.then((module: ESObject) => {
module[routerInfo.registerFunction!](routerInfo);
DynamicsRouter.navPathStack.pushPath({ name: name, param: param });
})
.catch((error: BusinessError) => {
logger.error(`promise import module failed, error code:${error.code}, message:${error.message}`);
});
} else {
DynamicsRouter.navPathStack.pushPath({ name: name, param: param });
DynamicsRouter.pushRouterStack(routerInfo);
}
}
6.注册动态路由跳转的页面信息
public static registerAppRouterPage(routerInfo: AppRouterInfo, wrapBuilder: WrappedBuilder<[object]>): void {
const builderName: string = routerInfo.name;
if (!DynamicsRouter.builderMap.has(builderName)) {
DynamicsRouter.registerBuilder(builderName, wrapBuilder);
}
}
动态路由的使用
1.在工程的hvigor/hvigor-config.json5中配置插件
{
...
"dependencies": {
...
"@app/ets-generator": "file:../plugin/AutoBuildRouter"
}
}
2.在工程的根目录的build-profile.json5中添加动态路由模块和需要加载的子模块的依赖,详细代码参考build-profile.json5。
{
"app":{
...
}
"modules":{
...
{
"name": "eventpropagation",
"srcPath": "./feature/eventpropagation"
},
{
"name": "routermodule",
"srcPath": "./feature/routermodule"
}
...
}
}
3.在主模块中添加动态路由和需要加载的子模块的依赖,详细代码请参考oh-package.json。
"dependencies": {
"@ohos/dynamicsrouter": "file:../../feature/routermodule",
"@ohos/event-propagation": "file:../../feature/eventpropagation",
...
}
4.在主模块中添加动态import变量表达式需要的参数,此处在packages中配置的模块名必须和oh-package.json中配置的名称相同,详细代码请参考build-profile.json5。
...
"buildOption": {
"arkOptions": {
"runtimeOnly": {
"packages": [
"@ohos/event-propagation",
...
]
}
}
}
5.在主模块EntryAbility的onCreate接口初始化动态路由,详细代码请参考EntryAbility.ets。
...
onCreate(want: Want, launchParam: AbilityConstant.LaunchParam): void {
DynamicsRouter.routerInit({
libPrefix: "@ohos", mapPath: "routerMap"
}, this.context);
...
}
...
6.在主模块的WaterFlowData.ets中,将子模块要加载的页面,添加到列表中,详细代码请参考WaterFlowData.ets和SceneModuleInfo。
export const waterFlowData: SceneModuleInfo[] = [
...
new SceneModuleInfo($r('app.media.address_exchange'), '地址交换动画', new RouterInfo("", ""), '动效', 2, "addressexchange/AddressExchangeView"),
...
}
7.在需要加载时将页面放入路由栈,详细代码请参考FunctionalScenes.ets。
@Builder
methodPoints(listData: ListData) {
Column() {
...
.onClick(() => {
...
DynamicsRouter.pushUri(this.listData.appUri);
...
})
}
8.在子模块中添加动态路由的依赖,详细代码可参考oh-package.json。
...
"dependencies": {
...
"@ohos/dynamicsrouter": "file:../../feature/routermodule"
}
以上是需要在主模块中添加的配置,如果已经添加过相关代码,则可以直接略过,按照下面的步骤在子模块中添加相关即可自动生成动态路由相关文件。
1.在子模块的oh-package.json5中添加路由模块依赖,可参考oh-package.json5
{
...
"dependencies": {
...
// 动态路由模块,用于配置动态路由
"@ohos/dynamicsrouter": "file:../../feature/routermodule"
}
}
2.在子模块的hvigorfile.ts文件中添加插件配置,可参考hvigorfile.ts
...
import { PluginConfig, etsGeneratorPlugin } from '@app/ets-generator';
// 配置路由信息
const config: PluginConfig = {
// 需要扫描的文件的路径,即配置自定义装饰AppRouter的文件路径
scanFiles: ["src/main/ets/view/AddressExchangeView"]
}
export default {
...
plugins: [etsGeneratorPlugin(config)] /* Custom plugin to extend the functionality of Hvigor. */
}
3.在需要跳转的页面的自定义组件上添加装饰器,可参考AddressExchangeView.ets,如果需要通过路由传递参数,则需要设置hasParam为true,可参考NavigationParameterTransferView.ets。
// 自定义装饰器,用于自动生成动态路由代码及页面的跳转。命名规则:模块名/自定义组件名
@AppRouter({ name: "addressexchange/AddressExchangeView" })
@Component
export struct AddressExchangeView {
...
}
高性能知识点
本示例使用动态import的方式加载模块,只有需要进入页面时才加载对应的模块,减少主页面启动时间和初始化效率,减少内存的占用。
工程结构&模块类型
routermodule // har类型
|---annotation
|---|---AppRouter.ets // 自定义装饰器
|---constants
| |---RouterInfo.ets // 路由信息类,用于配置动态路由跳转页面的名称和模块名(后续会删除)
|---model
| |---AppRouterInfo.ets // 路由信息类
| |---RouterParam.ets // 路由参数
|---router
| |---DynamicsRouter.ets // 动态路由实现类
|---util
| |---RouterLoader.ets // 路由表加载类
模块依赖
不涉及
我这边特意整理了《鸿蒙语法ArkTS、TypeScript、ArkUI、实战开发视频教程》以及《鸿蒙生态应用开发白皮书V2.0PDF》《鸿蒙开发学习手册》(共计890页)鸿蒙开发资料等…希望对大家有所帮助:https://docs.qq.com/doc/DZVVBYlhuRkZQZlB3
鸿蒙语法ArkTS、TypeScript、ArkUI等…视频教程:https://docs.qq.com/doc/DZVVBYlhuRkZQZlB3

OpenHarmony APP开发教程步骤:https://docs.qq.com/doc/DZVVBYlhuRkZQZlB3

鸿蒙生态应用开发白皮书V2.0PDF:https://docs.qq.com/doc/DZVVkRGRUd3pHSnFG

应用开发中级就业技术:https://docs.qq.com/doc/DZVVBYlhuRkZQZlB3

应用开发中高级就业技术:https://docs.qq.com/doc/DZVVBYlhuRkZQZlB3

南北双向高工技能基础:https://docs.qq.com/doc/DZVVBYlhuRkZQZlB3

全网首发-工业级 南向设备开发就业技术:https://docs.qq.com/doc/DZVVBYlhuRkZQZlB3

《鸿蒙开发学习手册》:
如何快速入门:https://docs.qq.com/doc/DZVVBYlhuRkZQZlB3
1.基本概念
2.构建第一个ArkTS应用
3.……

开发基础知识:https://docs.qq.com/doc/DZVVBYlhuRkZQZlB3
1.应用基础知识
2.配置文件
3.应用数据管理
4.应用安全管理
5.应用隐私保护
6.三方应用调用管控机制
7.资源分类与访问
8.学习ArkTS语言
9.……

基于ArkTS 开发:https://docs.qq.com/doc/DZVVBYlhuRkZQZlB3
1.Ability开发
2.UI开发
3.公共事件与通知
4.窗口管理
5.媒体
6.安全
7.网络与链接
8.电话服务
9.数据管理
10.后台任务(Background Task)管理
11.设备管理
12.设备使用信息统计
13.DFX
14.国际化开发
15.折叠屏系列
16.……


1190

被折叠的 条评论
为什么被折叠?



