HarmonyOS NEXT开发进阶(十):UIAbility 组件交互
文章目录
- 一、前言
- 二、启动应用内的 UIAbility
- 三、启动应用内的UIAbility并获取返回结果
- 四、启动其他应用的UIAbility
- 五、启动其他应用的 UIAbility 并获取返回结果
- 六、启动 UIAbility 的指定页面
- 6.1 调用方 UIAbility 指定启动页面
- 6.2 目标 UIAbility 首次启动
- 6.3 目标UIAbility非首次启动
- 七、拓展阅读
一、前言
UIAbility
是系统调度的最小单元。在设备内的功能模块之间跳转时,会涉及到启动特定的UIAbility
,该UIAbility
可以是应用内的其他UIAbility
,也可以是其他应用的UIAbility
(例如启动三方支付UIAbility
)。
二、启动应用内的 UIAbility
当一个应用内包含多个UIAbility
时,存在应用内启动UIAbility
的场景。例如在支付应用中从入口UIAbility启动收付款UIAbility。
假设应用中有两个UIAbility:EntryAbility和FuncAbility(可以在同一个Module
中,也可以在不同的Module
中),需要从EntryAbility的页面中启动FuncAbility。
在EntryAbility中,通过调用startAbility()
方法启动UIAbility
,want
为UIAbility
实例启动的入口参数,其中bundleName
为待启动应用的Bundle名称,abilityName
为待启动的UIAbility
名称,moduleName
在待启动的UIAbility
属于不同的Module
时添加,parameters
为自定义信息参数。示例中的context的获取方式参见获取UIAbility的Context属性。
let wantInfo = {
deviceId: '', // deviceId为空表示本设备
bundleName: 'com.example.myapplication',
abilityName: 'FuncAbility',
moduleName: 'module1', // moduleName非必选
parameters: { // 自定义信息
info: '来自EntryAbility Index页面',
},
}
// context为调用方UIAbility的AbilityContext
this.context.startAbility(wantInfo).then(() => {
// ...
}).catch((err) => {
// ...
})
在FuncAbility的生命周期回调函数中接收EntryAbility传递过来的参数。
import UIAbility from '@ohos.app.ability.UIAbility';
import Window from '@ohos.window';
export default class FuncAbility extends UIAbility {
onCreate(want, launchParam) {
// 接收调用方UIAbility传过来的参数
let funcAbilityWant = want;
let info = funcAbilityWant?.parameters?.info;
// ...
}
}
在FuncAbility业务完成之后,如需要停止当前UIAbility实例,在FuncAbility中通过调用terminateSelf()
方法实现。
// context为需要停止的UIAbility实例的AbilityContext
this.context.terminateSelf((err) => {
// ...
});
三、启动应用内的UIAbility并获取返回结果
在一个EntryAbility启动另外一个FuncAbility时,希望在被启动的FuncAbility完成相关业务后,能将结果返回给调用方。例如在应用中将入口功能和帐号登录功能分别设计为两个独立的UIAbility,在帐号登录UIAbility中完成登录操作后,需要将登录的结果返回给入口UIAbility。
在EntryAbility中,调用startAbilityForResult()
接口启动FuncAbility
,异步回调中的data用于接收FuncAbility停止自身后返回给EntryAbility的信息。示例中的context的获取方式参见获取UIAbility的Context属性。
let wantInfo = {
deviceId: '', // deviceId为空表示本设备
bundleName: 'com.example.myapplication',
abilityName: 'FuncAbility',
moduleName: 'module1', // moduleName非必选
parameters: { // 自定义信息
info: '来自EntryAbility Index页面',
},
}
// context为调用方UIAbility的AbilityContext
this.context.startAbilityForResult(wantInfo).then((data) => {
// ...
}).catch((err) => {
// ...
})
在FuncAbility停止自身时,需要调用terminateSelfWithResult()
方法,入参abilityResult为FuncAbility需要返回给EntryAbility的信息。
const RESULT_CODE: number = 1001;
let abilityResult = {
resultCode: RESULT_CODE,
want: {
bundleName: 'com.example.myapplication',
abilityName: 'FuncAbility',
moduleName: 'module1',
parameters: {
info: '来自FuncAbility Index页面',
},
},
}
// context为被调用方UIAbility的AbilityContext
this.context.terminateSelfWithResult(abilityResult, (err) => {
// ...
});
FuncAbility停止自身后,EntryAbility通过startAbilityForResult()
方法回调接收被FuncAbility返回的信息,RESULT_CODE需要与前面的数值保持一致。
const RESULT_CODE: number = 1001;
// ...
// context为调用方UIAbility的AbilityContext
this.context.startAbilityForResult(want).then((data) => {
if (data?.resultCode === RESULT_CODE) {
// 解析被调用方UIAbility返回的信息
let info = data.want?.parameters?.info;
// ...
}
}).catch((err) => {
// ...
})
四、启动其他应用的UIAbility
启动其他应用的UIAbility
,通常用户只需要完成一个通用的操作(例如需要选择一个文档应用来查看某个文档的内容信息),推荐使用隐式Want
启动。系统会根据调用方的want
参数来识别和启动匹配到的应用UIAbility。
启动UIAbility
有显式Want启动和隐式Want启动两种方式。
-
显式Want启动:启动一个确定应用的
UIAbility
,在want
参数中需要设置该应用bundleName
和abilityName
,当需要拉起某个明确的UIAbility
时,通常使用显式Want启动方式。 -
隐式Want启动:根据匹配条件由用户选择启动哪一个
UIAbility
,即不明确指出要启动哪一个UIAbility
(abilityName
参数未设置),在调用startAbility()
方法时,其入参want
中指定了一系列的entities
字段(表示目标UIAbility
额外的类别信息,如浏览器、视频播放器)和actions
字段(表示要执行的通用操作,如查看、分享、应用详情等)等参数信息,然后由系统去分析want
,并帮助找到合适的UIAbility
来启动。当需要拉起其他应用的UIAbility
时,开发者通常不知道用户设备中应用的安装情况,也无法确定目标应用的bundleName
和abilityName
,通常使用隐式Want启动方式。
将多个待匹配的文档应用安装到设备,在其对应UIAbility
的module.json5
配置文件中,配置skills
的entities
字段和actions
字段。
{
"module": {
"abilities": [
{
// ...
"skills": [
{
"entities": [
// ...
"entity.system.default"
],
"actions": [
// ...
"ohos.want.action.viewData"
]
}
]
}
]
}
}
在调用方want
参数中的entities
和action
需要被包含在待匹配UIAbility
的skills
配置的entities
和actions
中。系统匹配到符合entities
和actions
参数条件的UIAbility
后,会弹出选择框展示匹配到的UIAbility
实例列表供用户选择使用。示例中的context的获取方式参见获取UIAbility的Context属性。
let wantInfo = {
deviceId: '', // deviceId为空表示本设备
// 如果希望隐式仅在特定的捆绑包中进行查询,请取消下面的注释。
// bundleName: 'com.example.myapplication',
action: 'ohos.want.action.viewData',
// entities可以被省略。
entities: ['entity.system.default'],
}
// context为调用方UIAbility的AbilityContext
this.context.startAbility(wantInfo).then(() => {
// ...
}).catch((err) => {
// ...
})
效果示意如下图所示,点击“打开PDF文档”时,会弹出选择框供用户选择。
在文档应用使用完成之后,如需要停止当前UIAbility实例,通过调用terminateSelf()
方法实现。
// context为需要停止的UIAbility实例的AbilityContext
this.context.terminateSelf((err) => {
// ...
});
五、启动其他应用的 UIAbility 并获取返回结果
当使用隐式Want启动其他应用的UIAbility并希望获取返回结果时,调用方需要使用startAbilityForResult()
方法启动目标UIAbility。例如主应用中需要启动三方支付并获取支付结果。
在支付应用对应UIAbility的module.json5
配置文件中,配置skills
的entities
字段和actions
字段。
{
"module": {
"abilities": [
{
// ...
"skills": [
{
"entities": [
// ...
"entity.system.default"
],
"actions": [
// ...
"ohos.want.action.editData"
]
}
]
}
]
}
}
调用方使用startAbilityForResult()方
法启动支付应用的UIAbility,在调用方want
参数中的entities
和actions
需要被包含在待匹配UIAbility
的skills配置的entities
和actions
中。异步回调中的data用于后续接收支付UIAbility停止自身后返回给调用方的信息。系统匹配到符合entities
和actions
参数条件的UIAbility
后,会弹出选择框展示匹配到的UIAbility实例列表供用户选择使用。
let wantInfo = {
deviceId: '', // deviceId为空表示本设备
// uncomment line below if wish to implicitly query only in the specific bundle.
// bundleName: 'com.example.myapplication',
action: 'ohos.want.action.editData',
// entities can be omitted.
entities: ['entity.system.default'],
}
// context为调用方UIAbility的AbilityContext
this.context.startAbilityForResult(wantInfo).then((data) => {
// ...
}).catch((err) => {
// ...
})
在支付UIAbility完成支付之后,需要调用terminateSelfWithResult()
方法实现停止自身,并将abilityResult参数信息返回给调用方。
const RESULT_CODE: number = 1001;
let abilityResult = {
resultCode: RESULT_CODE,
want: {
bundleName: 'com.example.myapplication',
abilityName: 'EntryAbility',
moduleName: 'entry',
parameters: {
payResult: 'OKay',
},
},
}
// context为被调用方UIAbility的AbilityContext
this.context.terminateSelfWithResult(abilityResult, (err) => {
// ...
});
在调用方startAbilityForResult()
方法回调中接收支付应用返回的信息,RESULT_CODE需要与前面terminateSelfWithResult()
返回的数值保持一致。
const RESULT_CODE: number = 1001;
let want = {
// Want参数信息
};
// context为调用方UIAbility的AbilityContext
this.context.startAbilityForResult(want).then((data) => {
if (data?.resultCode === RESULT_CODE) {
// 解析被调用方UIAbility返回的信息
let payResult = data.want?.parameters?.payResult;
// ...
}
}).catch((err) => {
// ...
})
六、启动 UIAbility 的指定页面
一个UIAbility
可以对应多个页面,在不同的场景下启动该UIAbility时需要展示不同的页面,例如从一个UIAbility
的页面中跳转到另外一个UIAbility
时,希望启动目标UIAbility
的指定页面。下面主要讲解目标UIAbility
首次启动和目标UIAbility
非首次启动两种启动指定页面的场景,以及在讲解启动指定页面之前会讲解到在调用方如何指定启动页面。
6.1 调用方 UIAbility 指定启动页面
调用方UIAbility
启动另外一个UIAbility
时,通常需要跳转到指定的页面。例如FuncAbility包含两个页面(Index对应首页,Second对应功能A页面),此时需要在传入的want参数中配置指定的页面路径信息,可以通过want
中的parameters参数增加一个自定义参数传递页面跳转信息。示例中的context的获取方式参见获取UIAbility的Context属性。
let wantInfo = {
deviceId: '', // deviceId为空表示本设备
bundleName: 'com.example.myapplication',
abilityName: 'FuncAbility',
moduleName: 'module1', // moduleName非必选
parameters: { // 自定义参数传递页面信息
router: 'funcA',
},
}
// context为调用方UIAbility的AbilityContext
this.context.startAbility(wantInfo).then(() => {
// ...
}).catch((err) => {
// ...
})
6.2 目标 UIAbility 首次启动
目标UIAbility首次启动时,在目标UIAbility的onWindowStageCreate()
生命周期回调中,解析EntryAbility传递过来的want
参数,获取到需要加载的页面信息url,传入windowStage.loadContent()
方法。
import UIAbility from '@ohos.app.ability.UIAbility'
import Window from '@ohos.window'
export default class FuncAbility extends UIAbility {
funcAbilityWant;
onCreate(want, launchParam) {
// 接收调用方UIAbility传过来的参数
this.funcAbilityWant = want;
}
onWindowStageCreate(windowStage: Window.WindowStage) {
// Main window is created, set main page for this ability
let url = 'pages/Index';
if (this.funcAbilityWant?.parameters?.router) {
if (this.funcAbilityWant.parameters.router === 'funcA') {
url = 'pages/Second';
}
}
windowStage.loadContent(url, (err, data) => {
// ...
});
}
}
6.3 目标UIAbility非首次启动
经常还会遇到一类场景,当应用A已经启动且处于主页面时,回到桌面,打开应用B,并从应用B再次启动应用A,且需要跳转到应用A的指定页面。例如联系人应用和短信应用配合使用的场景。打开短信应用主页,回到桌面,此时短信应用处于已打开状态且当前处于短信应用的主页。再打开联系人应用主页,进入联系人用户A查看详情,点击短信图标,准备给用户A发送短信,此时会再次拉起短信应用且当前处于短信应用的发送页面。
针对以上场景,即当应用A的UIAbility实例已创建,并且处于该UIAbility实例对应的主页面中,此时,从应用B中需要再次启动应用A的该UIAbility,并且需要跳转到不同的页面,这种情况下要如何实现呢?
在目标UIAbility中,默认加载的是Index页面。由于当前UIAbility实例之前已经创建完成,此时会进入UIAbility的onNewWant()
回调中且不会进入onCreate()
和onWindowStageCreate()
生命周期回调,在onNewWant()
回调中解析调用方传递过来的want参数,并挂载到全局变量globalThis
中,以便于后续在页面中获取。
import UIAbility from '@ohos.app.ability.UIAbility'
export default class FuncAbility extends UIAbility {
onNewWant(want, launchParam) {
// 接收调用方UIAbility传过来的参数
globalThis.funcAbilityWant = want;
// ...
}
}
在FuncAbility中,此时需要在Index页面中通过页面路由Router模块实现指定页面的跳转,由于此时FuncAbility对应的Index页面是处于激活状态,不会重新变量声明以及进入aboutToAppear()
生命周期回调中。因此可以在Index页面的onPageShow()
生命周期回调中实现页面路由跳转的功能。
import router from '@ohos.router';
@Entry
@Component
struct Index {
onPageShow() {
let funcAbilityWant = globalThis.funcAbilityWant;
let url2 = funcAbilityWant?.parameters?.router;
if (url2 && url2 === 'funcA') {
router.replaceUrl({
url: 'pages/Second',
})
}
}
// 页面展示
build() {
// ...
}
}
注⚠️: 当被调用方Ability的启动模式设置为multiton
启动模式时,每次启动都会创建一个新的实例,那么onNewWant()
回调就不会被用到。
七、拓展阅读
- 《UIAbility组件间交互官方文档》