前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
社区首页 >专栏 >【HarmonyOS Next】鸿蒙应用弹框和提示气泡详解(一)

【HarmonyOS Next】鸿蒙应用弹框和提示气泡详解(一)

原创
作者头像
GeorgeGcs
发布于 2025-03-20 14:47:08
发布于 2025-03-20 14:47:08
11100
代码可运行
举报
运行总次数:0
代码可运行

【HarmonyOS Next】鸿蒙应用弹框和提示气泡详解(一)

一、前言

在应用开发中,弹框Dialog和提示气泡Toast使用频繁,移动开发同学较为熟悉。但在鸿蒙响应式布局里,早期的弹框Dialog和提示气泡Toast因与UI紧密绑定,在纯逻辑类文件中无法使用。

后来API迭代对其升级,如今可在纯逻辑类中使用,实现与UI解耦。

迭代优化过程

从page界面UI上弹出 => 挂靠子窗口实现弹出 => UI框架层预留挂靠节点

由此可见,与UI强绑定的实现方式API已不推荐。

二、鸿蒙中的弹框使用

目前鸿蒙HarmonyOS对于弹框、提示气泡及相关延申组件(浮层,Popup,Menu, bindSheet, bindContentCover),都是挂靠到UI框架预留节点进行添加渲染。

1. 弹框的实现

弹框有两种方式:

(1) 系统定制弹框,可直接使用

系统定制弹框依业务复杂度有两种封装方式:

基础弹框(警告弹框,列表弹窗)

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
@Entry
@Component
struct AlertDialogTextPage {
  build() {
    Column() {
      Button('showAlertDialog')
       .margin(30)
       .onClick(() => {
          this.getUIContext().showAlertDialog(
            {
              title: 'title',
              message: 'text',
              autoCancel: true,
              alignment: DialogAlignment.Center,
              buttons: [{
                value: 'cancel',
                action: () => {
                  console.info('cancel')
                }
              },
                {
                  enabled: true,
                  defaultFocus: true,
                  style: DialogButtonStyle.HIGHLIGHT,
                  value: 'ok',
                  action: () => {
                    console.info('ok')
                  }
                }],
            }
          )
        })
    }
   .width('100%')
   .margin({ top: 5 })
  }
}

带业务性质的PickerDialog弹框

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
// 日历选择器弹窗示例 (CalendarPickerDialog)
@Entry
@Component
struct PickerDialogTextPage {
  private selectedDate: Date = new Date('2024-04-23')

  build() {
    Column() {
      Button("Show CalendarPicker Dialog")
       .margin(20)
       .onClick(() => {
          console.info("CalendarDialog.show")
          CalendarPickerDialog.show({
            selected: this.selectedDate,
            acceptButtonStyle: {
              fontColor: '#2787d9',
              fontSize: '16fp',
              backgroundColor: '#f7f7f7',
              borderRadius: 10
            },
            cancelButtonStyle: {
              fontColor: Color.Red,
              fontSize: '16fp',
              backgroundColor: '#f7f7f7',
              borderRadius: 10
            },
            onAccept: (date: Date)=>{
              // 当弹出框再次弹出时显示选中的是上一次确定的日期
              this.selectedDate = date
            }
          })
        })
    }.width('100%')
  }
}
(2) 自定义弹框

创建节点对象并添加自定义弹框布局:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
@State message: string = "测试文本"
  /**
 * 自定义弹框布局
 * @param params
 */
@Builder
function buildText(params: Params) {
  Column() {
    Text(params.text)
     .fontSize(50)
     .fontWeight(FontWeight.Bold)
     .margin({ bottom: 36 })
    Button('Close')
     .onClick(() => {
        PromptActionClass.closeDialog()
      })
  }.backgroundColor('#FFF0F0F0')
}
  private contentNode: ComponentContent<Object> =
    new ComponentContent(this.ctx, wrapBuilder(buildText), new Params(this.message));

获取PromptAction对象,调用openCustomDialog弹出自定义弹框:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
this.getContext().getPromptAction().openCustomDialog(PromptActionClass.contentNode)
        .then(() => {
          console.info('OpenCustomDialog complete.')
        })
        .catch((error: BusinessError) => {
          let message = (error as BusinessError).message;
          let code = (error as BusinessError).code;
          console.error(`OpenCustomDialog args error code is ${code}, message is ${message}`);
        })

修改自定义弹框对齐方式、偏移量等,添加promptAction.BaseDialogOptions:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
this.getContext().getPromptAction().openCustomDialog(PromptActionClass.contentNode,{ alignment: DialogAlignment.Top, offset: { dx: 0, dy: 50 } })
        .then(() => {
          console.info('OpenCustomDialog complete.')
        })

2. 气泡的实现

气泡有对齐方式和悬浮态设置,但不能设置字体大小和颜色。自定义字体大小和颜色需用自定义弹框或Popup。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
import { promptAction } from '@kit.ArkUI'
import { BusinessError } from '@kit.BasicServicesKit'

/**
 * 气泡示例
 */
@Entry
@Component
struct ToastTextPage {

  build() {
    Column() {
      Button('Show toast').fontSize(20)
       .onClick(() => {
          try {
            this.getUIContext().getPromptAction().showToast({
              message: '测试气泡',
              duration: 2000,
              showMode: promptAction.ToastShowMode.TOP_MOST
            });
          } catch (error) {
            let message = (error as BusinessError).message
            let code = (error as BusinessError).code
            console.error(`showToast args error code is ${code}, message is ${message}`);
          };
        })
    }
   .height('100%')
   .width('100%')
   .justifyContent(FlexAlign.Center)
  }
}

三、源码示例

PromptActionClass.ets

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
import { BusinessError } from '@kit.BasicServicesKit';
import { ComponentContent, promptAction } from '@kit.ArkUI';
import { PromptAction, UIContext } from '@ohos.arkui.UIContext';

/**
 * 自定义弹框封装
 */
export class PromptActionClass {
  static ctx: UIContext;
  static contentNode: ComponentContent<Object>;
  static options: promptAction.BaseDialogOptions;

  static setContext(context: UIContext) {
    PromptActionClass.ctx = context;
  }

  static setContentNode(node: ComponentContent<Object>) {
    PromptActionClass.contentNode = node;
  }

  static setOptions(options: promptAction.BaseDialogOptions) {
    PromptActionClass.options = options;
  }

  static openDialog() {
    if (PromptActionClass.contentNode!== null) {
      PromptActionClass.ctx.getPromptAction().openCustomDialog(PromptActionClass.contentNode, PromptActionClass.options)
       .then(() => {
          console.info('OpenCustomDialog complete.')
        })
       .catch((error: BusinessError) => {
          let message = (error as BusinessError).message;
          let code = (error as BusinessError).code;
          console.error(`OpenCustomDialog args error code is ${code}, message is ${message}`);
        })
    }
  }

  static closeDialog() {
    if (PromptActionClass.contentNode!== null) {
      PromptActionClass.ctx.getPromptAction().closeCustomDialog(PromptActionClass.contentNode)
       .then(() => {
          console.info('CloseCustomDialog complete.')
        })
       .catch((error: BusinessError) => {
          let message = (error as BusinessError).message;
          let code = (error as BusinessError).code;
          console.error(`CloseCustomDialog args error code is ${code}, message is ${message}`);
        })
    }
  }

  static updateDialog(options: promptAction.BaseDialogOptions) {
    if (PromptActionClass.contentNode!== null) {
      PromptActionClass.ctx.getPromptAction().updateCustomDialog(PromptActionClass.contentNode, options)
       .then(() => {
          console.info('UpdateCustomDialog complete.')
        })
       .catch((error: BusinessError) => {
          let message = (error as BusinessError).message;
          let code = (error as BusinessError).code;
          console.error(`UpdateCustomDialog args error code is ${code}, message is ${message}`);
        })
    }
  }
}


class Params {
  text: string = ""

  constructor(text: string) {
    this.text = text;
  }
}

/**
 * 自定义弹框布局
 * @param params
 */
@Builder
function buildText(params: Params) {
  Column() {
    Text(params.text)
     .fontSize(50)
     .fontWeight(FontWeight.Bold)
     .margin({ bottom: 36 })
    Button('Close')
     .onClick(() => {
        PromptActionClass.closeDialog()
      })
  }.backgroundColor('#FFF0F0F0')
}

/**
 * 首页
 */
@Entry
@Component
struct Index {
  @State message: string = "hello"
  private ctx: UIContext = this.getUIContext();

  private contentNode: ComponentContent<Object> =
    new ComponentContent(this.ctx, wrapBuilder(buildText), new Params(this.message));

  @State handlePopup: boolean = false

  aboutToAppear(): void {
    PromptActionClass.setContext(this.ctx);
    PromptActionClass.setContentNode(this.contentNode);
    PromptActionClass.setOptions({ alignment: DialogAlignment.Top, offset: { dx: 0, dy: 50 } });
  }



  build() {
    Row() {
      Column() {
        /**
         * 显示自定义气泡 更新样式
         */
        Button("open dialog and update options")
         .margin({ top: 50 })
         .onClick(() => {
            PromptActionClass.openDialog()

            setTimeout(() => {
              PromptActionClass.updateDialog({
                alignment: DialogAlignment.Bottom,
                offset: { dx: 0, dy: -50 }
              })
            }, 1500)
          })

        /**
         * 显示自定义气泡 更新内容数据
         */
        Button("open dialog and update content")
         .margin({ top: 50 })
         .onClick(() => {
            PromptActionClass.openDialog()

            setTimeout(() => {
              this.contentNode.update(new Params('update'))
            }, 1500)
          })

        /**
         * 气泡菜单
         */
        Button('PopupOptions')
         .onClick(() => {
            this.handlePopup =!this.handlePopup
          })
         .bindPopup(this.handlePopup, {
            message: 'This is a popup with PopupOptions',
          })

        /**
         * 提示气泡
         */
        Button('show Toast')
         .onClick(() => {
            let promptAction: PromptAction = this.ctx.getPromptAction();
            try {
              promptAction.showToast({
                message: 'Message Info',
                duration: 2000
              });
            } catch (error) {
              let message = (error as BusinessError).message;
              let code = (error as BusinessError).code;
              console.error(`showToast args error code is ${code}, message is ${message}`);
            };
          })

        /**
         * 系统弹框
         */
        Button('show Toast')
         .onClick(() => {
            let promptAction: PromptAction = this.ctx.getPromptAction();
            try {
              promptAction.showToast({
                message: 'Message Info',
                duration: 2000
              });
            } catch (error) {
              let message = (error as BusinessError).message;
              let code = (error as BusinessError).code;
              console.error(`showToast args error code is ${code}, message is ${message}`);
            };
          })
      }
     .width('100%')
     .height('100%')
    }
   .width('100%')
   .height('100%')
  }
}

原创声明:本文系作者授权腾讯云开发者社区发表,未经许可,不得转载。

如有侵权,请联系 cloudcommunity@tencent.com 删除。

原创声明:本文系作者授权腾讯云开发者社区发表,未经许可,不得转载。

如有侵权,请联系 cloudcommunity@tencent.com 删除。

评论
登录后参与评论
暂无评论
推荐阅读
编辑精选文章
换一批
【HarmonyOS Next】鸿蒙中自定义弹框OpenCustomDialog、CustomDialog与DialogHub的区别详解
CustomDialog = 》 OpenCustomDialog = 》 DialogHub
GeorgeGcs
2025/03/28
1420
HarmonyOS 开发实践 —— 基于promptAction全局弹窗
可以使用openCustomDialog接口, 创建并弹出dialogContent对应的自定义弹窗,使用Promise异步回调。
小帅聊鸿蒙
2024/11/25
1300
HarmonyOS 开发实践 —— 基于promptAction全局弹窗
HarmonyOS Next中的弹出框使用
弹出框是一种模态窗口,通常用于在保持当前上下文环境的同时,临时展示用户需关注的信息或待处理的操作。用户需在模态弹出框内完成相关交互任务之后,才能退出模态模式。弹出框可以不与任何组件绑定,其内容通常由多种组件组成,如文本、列表、输入框、图片等,以实现布局。ArkUI当前提供了自定义和固定样式两类弹出框组件。
儿歌八万首
2025/03/19
450
HarmonyOS Next中的弹出框使用
【HarmonyOS Next】鸿蒙应用实现弹框DialogHub详解
鸿蒙中实现弹框目前官方提供openCustomDialog和CustomDialog两种模式。推荐前者,详情见下图和官网文档链接:
GeorgeGcs
2025/03/28
1290
【HarmonyOS Next】鸿蒙应用弹框和提示气泡详解(二)之浮层(OverlayManager),半模态页面(bindSheet),全模态页面
上期围绕 HarmonyOS Next 最新API趋势,介绍了鸿蒙应用中最新的自定义弹框和提示气泡的使用。
GeorgeGcs
2025/03/26
890
【HarmonyOS Next】鸿蒙应用弹框和提示气泡详解(二)之浮层(OverlayManager),半模态页面(bindSheet),全模态页面
HarmonyOS 开发实践 —— 原生应用占用空间管理
1.可以通过storageStatistics.getCurrentBundleStats接口获取缓存大小。
小帅聊鸿蒙
2024/12/11
710
HarmonyOS 开发实践 —— 原生应用占用空间管理
HarmonyOS 开发实践——在ArkTS中,实现不在Enter模块中就可以创建的自定义弹窗
2、在@CustomDialog装饰的自定义组件中必须声明CustomDialogControlle类型的变量
小帅聊鸿蒙
2024/10/29
2981
深入解析鸿蒙系统的页面路由(Router)机制
鸿蒙系统以其独特的分布式架构和跨设备的统一体验而备受瞩目。在这个系统中,页面路由(Router)机制是连接应用各页面的关键组成部分。本文将深入探讨鸿蒙系统的页面路由,揭示其工作原理、特点以及在应用开发中的实际应用。
小帅聊鸿蒙
2024/07/20
7060
【HarmonyOS NEXT】鸿蒙应用使用后台任务之长时任务,解决屏幕录制音乐播放等操作不被挂起
了解后台任务和长时任务前,我们需要先明白鸿蒙的后台特性:所谓的后台,指的是设备返回主界面、锁屏、应用切换等操作会使应用退至后台这个状态。
GeorgeGcs
2025/03/24
1390
【HarmonyOS NEXT】鸿蒙应用使用后台任务之长时任务,解决屏幕录制音乐播放等操作不被挂起
HarmonyOS 开发实践——基于@ohos.data.unifiedDataChannel实现数据共享
本模块为统一数据管理框架(Unified Data Management Framework,UDMF)的组成部分,针对多对多跨应用数据共享的不同业务场景提供了标准化的数据通路,提供了标准化的数据接入与读取接口。同时对文本、图片等数据类型提供了标准化定义,方便不同应用间进行数据交互,减少数据类型适配的工作量。
小帅聊鸿蒙
2024/11/15
1100
HarmonyOS 开发实践——基于@ohos.data.unifiedDataChannel实现数据共享
【HarmonyOS NEXT】鸿蒙应用实现屏幕录制详解和源码
官方文档关于屏幕录制的API和示例介绍获取简单和突兀。使用起来会让上手程度变高。所以特意开篇文章,讲解屏幕录制的使用。官方文档参见:使用AVScreenCaptureRecorder录屏写文件(ArkTS)
GeorgeGcs
2025/03/24
700
【HarmonyOS NEXT】鸿蒙应用实现屏幕录制详解和源码
HarmonyOS 开发实践 —— 事件通信能力解决方案
元能力和事件通知当前提供的通信方式主要有Emitter、EventHub、CommonEvent,线程间通信也可以使用Worker和Taskpool提供的postMessage和sendData向数组线程发送消息。应用间通信可以使用自定义公共事件和IPC&RPC两种方式。本文主要介绍事件通知和元能力提供的通信能力。
小帅聊鸿蒙
2024/12/01
2540
HarmonyOS 开发实践 —— 事件通信能力解决方案
【HarmonyOS学习】用户文件访问
分为文档类Uri和媒体文件Uri两类,需要是normal等级的应用(默认的应用是这个)调用这两个类,要不然会报没有权限的错误,官方配图如下:
钟子翔
2024/12/24
1170
【HarmonyOS NEXT】鸿蒙应用压缩和解压处理
鸿蒙应用中经常会遇到使用压缩包的场景。例如H5的离线包逻辑,需要提前将资源包下载到本地,进行压缩操作获取资源。
GeorgeGcs
2025/03/28
1670
HarmonyOS 开发实践 —— 基于Picker的受限权限适配方案
Picker由系统独立进程实现,应用可以通过拉起Picker组件,用户在Picker上选择对应的资源(如图片、文档等),应用可以获取Picker返回的结果。
小帅聊鸿蒙
2024/12/06
1580
【鸿蒙基于API 13实战开发】—— ArkUI 组件:Router切换Navigation
从ArkUI组件树层级上来看,原先由Router管理的page在页面栈管理节点stage的下面。Navigation作为导航容器组件,可以挂载在单个page节点下,也可以叠加、嵌套。Navigation管理了标题栏、内容区和工具栏,内容区用于显示用户自定义页面的内容,并支持页面的路由能力。Navigation的这种设计上有如下优势:
小帅聊鸿蒙
2025/01/17
1090
【鸿蒙基于API 13实战开发】—— ArkUI 组件:Router切换Navigation
HarmonyOS 开发实践 —— 基于短距离通信的BLE蓝牙扫描
蓝牙是一种短距的无线通讯技术,可实现固定设备、移动设备之间的数据交换。一般将蓝牙3.0之前的BR/EDR蓝牙称为传统蓝牙,而将蓝牙4.0规范下的BLE蓝牙称为低功耗蓝牙。BLE模块提供了对蓝牙操作和管理的方法。
小帅聊鸿蒙
2024/11/27
1990
HarmonyOS 开发实践 —— 基于短距离通信的BLE蓝牙扫描
鸿蒙开发:wrapBuilder传递参数
在前边的文章中,关于wrapBuilder传递参数我们有过概述,无论是定义在局部还是全局,传递参数都是非常的简单,直接调用builder函数即可,简单案例如下所示:
程序员一鸣
2025/03/10
840
鸿蒙开发:wrapBuilder传递参数
鸿蒙(HarmonyOS)性能优化实战-电量优化
设备返回主界面、锁屏、应用切换等操作会使应用退至后台。为了降低设备耗电速度、保障用户使用流畅度,系统会对退至后台的应用进行管控,包括进程挂起和进程终止。为了保障后台音乐播放、日历提醒等功能的正常使用,系统提供了受规范约束的后台任务,扩展应用在后台的运行时间。
小帅聊鸿蒙
2024/10/24
1820
鸿蒙(HarmonyOS)性能优化实战-电量优化
鸿蒙安全控件之位置控件简介
位置控件使用直观且易懂的通用标识,让用户明确地知道这是一个获取位置信息的按钮。这满足了授权场景需要匹配用户真实意图的需求。只有当用户主观愿意,并且明确了解使用场景后点击位置控件,应用才会获得临时的授权,获取位置信息并完成相应的服务功能。
龙儿筝
2024/11/28
1210
推荐阅读
相关推荐
【HarmonyOS Next】鸿蒙中自定义弹框OpenCustomDialog、CustomDialog与DialogHub的区别详解
更多 >
领券
社区富文本编辑器全新改版!诚邀体验~
全新交互,全新视觉,新增快捷键、悬浮工具栏、高亮块等功能并同时优化现有功能,全面提升创作效率和体验
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档
查看详情【社区公告】 技术创作特训营有奖征文