HarmonyOS:弹出框控制器

一、简介

ArkUI的弹出框控制器在绑定弹出框后,可提供对弹出框的操作能力,当前支持关闭功能。可以将控制器传入弹出框内容区域后进行操作。

API version 18开始,可设置controller参数以绑定DialogController控制器,通过控制器能够操作弹出框。

二、使用约束

目前openCustomDialogWithController和presentCustomDialog支持通过controller参数来绑定弹出框进行操作,目前getDialogController支持获取自定义组件所在的弹出框的控制器。
说明 一个弹出框控制器只能绑定一个弹出框,且操作只对该弹出框生效。 使用getDialogController获取弹出框控制器时,如果当前自定义组件不在弹出框中显示则获取为undefined。

三、创建自定义内容为ComponentContent的弹出框控制器

  1. 初始化一个自定义弹出框内容区的入参类,内部包含弹出框控制器。
bash 复制代码
class Params {
  public text: string = ''
  public dialogController: promptAction.CommonController = new promptAction.DialogController()
  constructor(text: string, dialogController: promptAction.CommonController) {
    this.text = text
    this.dialogController = dialogController
  }
}
  1. 初始化一个自定义的弹出框内容区,内部包含一个按钮,该按钮通过该自定义组件自带的弹出框控制器实现关闭功能。
bash 复制代码
@Component
struct MyComponent {
  build() {
    Column({ space: 5 }) {
      Button('点我关闭弹窗:通过自定义组件自带的DialogController')
        .onClick(() => {
          let dialogController: promptAction.DialogController = this.getDialogController()
          if (dialogController !== undefined) {
            dialogController.close()
          }
        })
    }
  }
}
  1. 初始化另一自定义弹出框内容区,其中包含一个Text组件和一个按钮,该按钮通过外部传递的弹出框控制器用于关闭弹出框,并且该内容区还包含前一个自定义弹出框内容区。
bash 复制代码
@Builder
function buildText(params: Params) {
  Column({ space: 5 }) {
    Text(params.text)
      .fontSize(30)
    if (params.dialogController !== undefined) {
      Button('点我关闭弹窗:通过外部传递的DialogController')
        .onClick(() => {
          params.dialogController.close()
        })
    }
    MyComponent()
  }
  .width(300)
  .height(200)
  .backgroundColor('#FFF0F0F0')
}
  1. 初始化一个弹出框控制器,并通过设置控制器参数来初始化一个弹出框内容实体对象。最后,通过调用UIContext中的getPromptAction方法获取PromptAction对象,再通过该对象调用openCustomDialogWithController接口,并且设置初始化的内容实体对象和控制器参数以创建弹出框。
bash 复制代码
let dialogController: promptAction.CommonController = new promptAction.DialogController()
let contentNode: ComponentContent<Object> =
  new ComponentContent(this.getUIContext(), wrapBuilder(buildText), new Params(this.message, dialogController))
this.getUIContext().getPromptAction().openCustomDialogWithController(
  contentNode, dialogController, this.baseDialogOptions).catch((err: BusinessError) => {
  console.error('openCustomDialogWithController error: ' + err.code + ' ' + err.message)
})

四、创建自定义内容为CustomBuilder的弹出框控制器

  1. 初始化一个自定义弹出框内容区,内部包含一个Text组件和一个按钮,该按钮通过外部传递的弹出框控制器实现关闭功能。
bash 复制代码
@Builder customDialogComponent(dialogController: promptAction.DialogController) {
  Column({ space: 5 }) {
    Text(this.message)
      .fontSize(30)
    if (dialogController !== undefined) {
      Button('点击关闭弹窗:通过外部传递的DialogController')
        .onClick(() => {
          dialogController.close()
        })
    }
  }
  .height(200)
  .padding(5)
  .justifyContent(FlexAlign.SpaceBetween)
  .backgroundColor('#FFF0F0F0')
}
  1. 初始化一个弹出框控制器,并通过调用UIContext中的getPromptAction方法获取PromptAction对象,再通过该对象调用presentCustomDialog接口,设置初始化的内容实体对象和控制器参数以创建弹出框。
bash 复制代码
let dialogController: promptAction.CommonController = new promptAction.DialogController()
this.getUIContext().getPromptAction().presentCustomDialog(() => {
  this.customDialogComponent(dialogController)
}, dialogController, this.dialogOptions).catch((err: BusinessError) => {
  console.error('presentCustomDialog error: ' + err.code + ' ' + err.message)
})

五、在CustomDialogController内容区直接获取弹出框控制器

  1. 初始化一个自定义弹出框内容区,内部包含一个Text组件和一个按钮,该按钮通过弹出框控制器关闭弹出框。
bash 复制代码
@CustomDialog
@Component
struct CustomDialogExample {
  controller?: CustomDialogController

  build() {
    Column({ space: 5 }) {
      Text('我是内容')
        .fontSize(30)
      Button('点我关闭弹窗:通过自定义组件自带的DialogController')
        .onClick(() => {
          let dialogController: PromptActionDialogController = this.getDialogController()
          if (dialogController !== undefined) {
            dialogController.close()
          }
        })
    }
    .height(200)
    .backgroundColor('#FFF0F0F0')
  }
}
  1. 初始化一个自定义弹出框构造器,关联自定义弹出框内容区。
bash 复制代码
let customDialogController: CustomDialogController = new CustomDialogController({
  builder: CustomDialogExample(),
})
customDialogController.open()

六、使用控制器获取弹出框的状态

在自定义弹出框场景中,可以通过控制器调用getState接口获取弹出框状态。

初始化一个自定义弹出框内容区,内部包含一个Text组件和一个按钮,该按钮通过调用getState获取当前弹出框状态。

bash 复制代码
@Builder customDialogComponent(dialogController: promptAction.DialogController) {
  Column({ space: 5 }) {
    Text(this.message)
      .fontSize(30)
    if (dialogController !== undefined) {
      Button('点我查询弹窗状态')
        .onClick(() => {
          console.info('state:' + this.dialogController.getState())
        })
    }
  }
  .height(200)
  .padding(5)
  .justifyContent(FlexAlign.SpaceBetween)
  .backgroundColor('#FFF0F0F0')
}

七、完整示例

通过外部传递的弹出框控制器和自定义组件自带的弹出框控制器,在自定义弹出框内容区域内实现关闭功能。
效果图

bash 复制代码
import { ComponentContent, promptAction } from '@kit.ArkUI'
import { BusinessError } from '@kit.BasicServicesKit'

class Params {
  public text: string = ''
  public dialogController: promptAction.CommonController = new promptAction.DialogController()

  constructor(text: string, dialogController: promptAction.CommonController) {
    this.text = text
    this.dialogController = dialogController
  }
}

@Component
struct MyComponent {
  build() {
    Column({ space: 5 }) {
      Button('点我关闭弹窗:通过自定义组件自带的DialogController')
        .onClick(() => {
          let dialogController: promptAction.DialogController = this.getDialogController() // API version 18
          if (dialogController !== undefined) {
            dialogController.close()
          }
        })
    }
  }
}

@Builder
function buildText(params: Params) {
  Column({ space: 5 }) {
    Text(params.text)
      .fontSize(30)
    if (params.dialogController !== undefined) {
      Button('点我关闭弹窗:通过外部传递的DialogController')
        .onClick(() => {
          params.dialogController.close()
        })
    }
    MyComponent()
  }
  .width(300)
  .height(200)
  .backgroundColor('#FFF0F0F0')
}

@CustomDialog
@Component
struct CustomDialogExample {
  controller?: CustomDialogController

  build() {
    Column({ space: 5 }) {
      Text('我是内容')
        .fontSize(30)
      Button('点我关闭弹窗:通过自定义组件自带的DialogController')
        .onClick(() => {
          let dialogController: PromptActionDialogController = this.getDialogController()
          if (dialogController !== undefined) {
            dialogController.close()
          }
        })
    }
    .height(200)
    .backgroundColor('#FFF0F0F0')
  }
}

@Entry
@Component
export struct TestOpenCustomDialog5 {
  private message = '弹窗'

  @Builder
  customDialogComponent(dialogController: promptAction.DialogController) {
    Column({ space: 5 }) {
      Text(this.message)
        .fontSize(30)
      if (dialogController !== undefined) {
        Button('点击关闭弹窗:通过外部传递的DialogController')
          .onClick(() => {
            dialogController.close()
          })
      }
    }
    .height(200)
    .padding(5)
    .justifyContent(FlexAlign.SpaceBetween)
    .backgroundColor('#FFF0F0F0')
  }

  @Builder
  customDialogComponentWithId(dialogId: number, dialogController: promptAction.DialogController) {
    Column({ space: 5 }) {
      Text(this.message)
        .fontSize(30)
      if (dialogId !== undefined) {
        Button('点击关闭弹窗:通过DialogID')
          .onClick(() => {
            this.getUIContext().getPromptAction().closeCustomDialog(dialogId)
          })
      }
      if (dialogController !== undefined) {
        Button('点击关闭弹窗:通过外部传递的DialogController')
          .onClick(() => {
            dialogController.close()
          })
      }
    }
  }

  private baseDialogOptions: promptAction.BaseDialogOptions = {
    isModal: false,
    autoCancel: false
  }
  private dialogOptions: promptAction.DialogOptions = {
    isModal: false,
    autoCancel: false
  }

  openCustomDialogWithController() {
    let dialogController: promptAction.CommonController = new promptAction.DialogController()
    let contentNode: ComponentContent<Object> =
      new ComponentContent(this.getUIContext(), wrapBuilder(buildText),
        new Params(this.message, dialogController))
    this.getUIContext().getPromptAction().openCustomDialogWithController(
      contentNode, dialogController, this.baseDialogOptions).catch((err: BusinessError) => {
      console.error('openCustomDialogWithController error: ' + err.code + ' ' + err.message)
    })
  }

  presentCustomDialogWithCustomBuilder() {
    let dialogController: promptAction.CommonController = new promptAction.DialogController()
    this.getUIContext().getPromptAction().presentCustomDialog(() => {
      this.customDialogComponent(dialogController)
    }, dialogController, this.dialogOptions).catch((err: BusinessError) => {
      console.error('presentCustomDialog error: ' + err.code + ' ' + err.message)
    })
  }

  presentCustomDialogWithCustomBuilderWithId() {
    let dialogController: promptAction.CommonController = new promptAction.DialogController()
    this.getUIContext().getPromptAction().presentCustomDialog((dialogId: number) => {
      this.customDialogComponentWithId(dialogId, dialogController)
    }, dialogController, this.dialogOptions).catch((err: BusinessError) => {
      console.error('presentCustomDialog error: ' + err.code + ' ' + err.message)
    })
  }

  customDialogController() {
    let customDialogController: CustomDialogController = new CustomDialogController({
      builder: CustomDialogExample(),
    })
    customDialogController.open()
  }

  build() {
    Column({ space: 20 }) {
      Button('openCustomDialogWithController弹窗').margin({top: 40})
        .onClick(() => {
          this.openCustomDialogWithController()
        })
      Button('presentCustomDialog+CustomBuilder弹窗')
        .onClick(() => {
          this.presentCustomDialogWithCustomBuilder()
        })
      Button('presentCustomDialog+CustomBuilderWithId弹窗')
        .onClick(() => {
          this.presentCustomDialogWithCustomBuilderWithId()
        })
      Button('CustomDialogController弹窗')
        .onClick(() => {
          this.customDialogController()
        })
    }.width('100%')
  }
}
相关推荐
金鸿客3 小时前
鸿蒙Text文本设置选中菜单和AI菜单
harmonyos
特立独行的猫a9 小时前
鸿蒙PC平台三方库移植实战:以libid3tag库为例(附完整移植流程)
华为·harmonyos·移植·鸿蒙pc·libid3tag
爱笑的眼睛1114 小时前
深入探讨HarmonyOS分布式剪贴板:技术原理与开发实践
华为·harmonyos
特立独行的猫a15 小时前
鸿蒙PC平台三方库移植实战:以libogg库移植为例(附完整移植流程与工具链配置)
华为·harmonyos·三方库移植·鸿蒙pc·libogg
爱笑的眼睛1115 小时前
深入理解HarmonyOS通知渠道与优先级设置:从基础到高级实践
华为·harmonyos
夏文强19 小时前
HarmonyOS开发者认证练习题-判断题
华为·harmonyos
Kisang.19 小时前
【HarmonyOS】ArkTS的多线程并发(下)——线程间通信对象的传递
华为·typescript·harmonyos·鸿蒙
柒儿吖1 天前
Qt for HarmonyOS 3D图片轮播组件开源鸿蒙开发实战
qt·3d·harmonyos
爱笑的眼睛111 天前
HarmonyOS分布式输入法开发:实现多设备无缝输入体验
华为·harmonyos