HarmonyOS鸿蒙开发 弹窗及加载中指示器HUD功能实现

HarmonyOS鸿蒙开发 弹窗及加载中指示器HUD功能实现

最近在学习鸿蒙开发过程中,阅读了官方文档,在之前做flutter时候,经常使用overlay,使用OverlayEntry加入到overlayState来做添加悬浮按钮、提示弹窗、加载中指示器、加载失败的toast等功能。那在HarmonyOS鸿蒙开发中也可能有类似的功能需求。

HarmonyOS鸿蒙开发的使用弹窗文档中已经非常详细了

地址:https://developer.huawei.com/consumer/cn/doc/harmonyos-guides-V5/arkts-use-dialog-V5

一、子窗口window

在弹出的loading指示器中,我们可以使用创建子window的方式,调用window的loadContentByName方法来实现。

实现步骤

效果预览

  • 1、实现加载中的loading组件,这里定义名字为LoadingHud

在LoadingHud中有LoadingProgress、Text提示文本,Text显示的信息由LocalStorage进行传递

需要传递的数据message,在aboutToAppear进行赋值

@Local message: string = '';

  aboutToAppear(): void {
    this.message = LocalStorage.getShared().get("message") ?? "";
  }

当然在调用window的loadContentByName时候,需要确定加载的主角的routeName,这就需要在LoadingHud组件中使用装饰器来设置

/// 通用的hud,弹出框,或者loading框
@Entry({ routeName: "hudLoading", storage: LocalStorage.getShared() })
@ComponentV2
export struct LoadingHud {
  ... 其他代码
}

LoadingHud的完整代码如下:

/// 通用的hud,弹出框,或者loading框
@Entry({ routeName: "hudLoading", storage: LocalStorage.getShared() })
@ComponentV2
export struct LoadingHud {
  @Local message: string = '';

  aboutToAppear(): void {
    this.message = LocalStorage.getShared().get("message") ?? "";
  }

  build() {
    Column() {
      Column(){
        Row() {
          // 从左往右,1号环形进度条,默认前景色为蓝色渐变,默认strokeWidth进度条宽度为2.0vp
          LoadingProgress()
            .color($r('app.color.success'))
            .width(40)
            .height(40)

          // message
          Text(this.message)
            .fontSize(14)
            .fontColor($r('app.color.dataset_empty_message'))
            .margin({
              left: 10
            })
        }
        .padding({
          top: 15,
          bottom: 15,
          left: 15,
          right: 20
        })
        .justifyContent(FlexAlign.Center)

        Button("点击消失")
          .width(100)
          .height(40)
          .fontSize(12)
          .backgroundColor('#ef04792c')
          .margin({
            top: 10
          })
          .onClick(()=> {
             LoadingHudUtil.dismissLoading();
          })
      }
      .justifyContent(FlexAlign.Center)
      .constraintSize({
        minWidth: 200,
        minHeight: 150,
      })
      .backgroundColor($r('app.color.white'))
      .borderRadius(10)
    }
    .justifyContent(FlexAlign.Center)
    .width('100%')
    .height('100%')
    .backgroundColor('#00000000')
    .hitTestBehavior(HitTestMode.Transparent)
  }
}
  • 2、创建子Window并显示

在创建LoadingHud后,我们需要创建创建子Window并显示window,显示我们的loadingHUD

创建window的createWindow,这里使用的windowType是window.WindowType.TYPE_DIALOG,也可以换成其他的试试看。

let windowName = "loading";
      // 创建窗口
      let subWindow = await window.createWindow(
        {
          name: windowName,
          windowType: window.WindowType.TYPE_DIALOG,
          ctx: ctx,
        }
      );

设置LocalStorage数据,存储message

//创建存储
      let storage = new LocalStorage();
      //存储数据
      storage.setOrCreate('message',  tip);

调用window的loadContentByName,设置Window的大小及背景颜色,显示Window

await subWindow.loadContentByName('hudLoading', storage);
      let dp = display.getDefaultDisplaySync();
      await subWindow.resize(dp.width, dp.height);
      subWindow.setWindowBackgroundColor('#30000000');
      await subWindow.showWindow();

显示后Window,在需要消失的时候调用destroyWindow

static async dismissLoading(): Promise<void> {
    if (LoadingHudUtil.cacheWindow) {
      await LoadingHudUtil.cacheWindow.destroyWindow();
    }
  }

完整的LoadingHudUtil的代码如下

import { display, window } from '@kit.ArkUI';
import { common } from '@kit.AbilityKit';
import('../components/hud/LoadingHud'); // 引入命名路由页面

// 自定义弹出窗口
export class LoadingHudUtil {
  private static cacheWindow: window.Window;

  static async showLoading(tip: string): Promise<void> {
    let ctx = getContext() as common.UIAbilityContext;
    try {
      let windowName = "loading";
      // 创建窗口
      let subWindow = await window.createWindow(
        {
          name: windowName,
          windowType: window.WindowType.TYPE_DIALOG,
          ctx: ctx,
        }
      );

      LoadingHudUtil.cacheWindow = subWindow;

      //创建存储
      let storage = new LocalStorage();
      //存储数据
      storage.setOrCreate('message',  tip);

      console.log("LoadingHudUtil loadContentByName" + tip);

      // subWindow.setGestureBackEnabled(false);
      // subWindow.setDialogBackGestureEnabled(false);
      // subWindow.setWindowTouchable(true);
      await subWindow.loadContentByName('hudLoading', storage);
      let dp = display.getDefaultDisplaySync();
      await subWindow.resize(dp.width, dp.height);
      subWindow.setWindowBackgroundColor('#30000000');
      await subWindow.showWindow();
    } catch (e) {
      console.log("LoadingHudUtil showLoading e:" + JSON.stringify(e));
    }
  }

  static async dismissLoading(): Promise<void> {
    if (LoadingHudUtil.cacheWindow) {
      await LoadingHudUtil.cacheWindow.destroyWindow();
    }
  }
}

二、自定义Dialog

在HarmonyOS鸿蒙开发中,可以使用CustomDialogController来实现自定义的弹窗。

效果预览

  • 1.自定义弹窗组件CustomAlertDialog

在CustomAlertDialog中实现一个消息提示,并且点击按钮可以关闭dialog

代码如下:

@CustomDialog
export struct CustomAlertDialog {
  controller?: CustomDialogController
  title?: string

  build() {
    Column() {
      Column() {
        // message
        Text(this.title)
          .fontSize(14)
          .fontColor($r('app.color.dataset_empty_message'))
          .margin({
            left: 10
          })

        Button("点击消失")
          .width(100)
          .height(40)
          .fontSize(12)
          .backgroundColor('#ef04792c')
          .margin({
            top: 10
          })
          .onClick(() => {
            this.controller?.close();
          })
      }
      .justifyContent(FlexAlign.Center)
      .constraintSize({
        minWidth: 200,
        minHeight: 100,
      })
      .backgroundColor($r('app.color.white'))
      .borderRadius(10)
    }
    .justifyContent(FlexAlign.Center)
    .width('100%')
    .height('100%')
    .backgroundColor(Color.Transparent)
    .hitTestBehavior(HitTestMode.Transparent)
  }
}
  • 2.使用CustomDialogController来展示弹窗

定义CustomDialogController

// 自定义CustomDialog
  customDialogController: CustomDialogController | null = new CustomDialogController({
    builder: CustomAlertDialog({
      title: "温馨提示"
    }),
    alignment: DialogAlignment.Center,
    onWillDismiss: (dismissDialogAction: DismissDialogAction) => {
      console.info("reason=" + JSON.stringify(dismissDialogAction.reason))
      console.log("dialog onWillDismiss")
      if (dismissDialogAction.reason == DismissReason.PRESS_BACK) {
        dismissDialogAction.dismiss()
      }
      if (dismissDialogAction.reason == DismissReason.TOUCH_OUTSIDE) {
        dismissDialogAction.dismiss()
      }
    },
    autoCancel: true,
    customStyle: true,
  });

在需要展示弹窗的时候调用customDialogController的open方法。

 if (this.customDialogController != null) {
                      this.customDialogController.open()
                    }

当然如果页面消失,尽量在aboutToDisappear中将customDialogController置空

  // 在自定义组件即将销毁时将dialogController置空
  aboutToDisappear() {
    this.customDialogController = null // 将dialogController置空
  }

三、Overlay浮层

在官方文档中有一段描述

浮层(OverlayManager) 用于将自定义的UI内容展示在页面(Page)之上,在Dialog、Popup、Menu、BindSheet、BindContentCover和Toast等组件之下,展示的范围为当前窗口安全区内。可适用于常驻悬浮等场景。

使用OverlayManager来添加、删除、隐藏、显示节点Component

效果预览


  • 1.定义CustomOverlayView组件界面

在CustomOverlayView中,我们定义了加载中,加载失败,加载成功的几种类型,用于展示不同的样式

定义OverlayConfig类为展示的界面配置、OverlayScaleImage缩放的icon

export enum OverlayType {
  loading,
  success,
  fail
}

export class OverlayConfig {
  message: string = ""
  offset: Position = { x: 0, y: -50 }
  index: number = 0
  autoDismiss: boolean = true;
  duration: number = 3000 // 持续时间
  onCallback?: (index: number) => void
  type: OverlayType = OverlayType.loading

  constructor(message: string) {
    this.message = message
  }
}

@Builder
export function builderCustomOverlayView(overlayConfig: OverlayConfig) {
  CustomOverlayView({
    olConfig: overlayConfig
  })
}

@ComponentV2
struct OverlayScaleImage {
  @Param @Require src: PixelMap | ResourceStr | DrawableDescriptor;
  @Param imgWidth: number = 40;
  @Local imgScale: number = 0.0;

  build() {
    Image(this.src)
      .width(this.imgWidth)
      .aspectRatio(1)
      .scale({ x: this.imgScale, y: this.imgScale })
      .animation({
        duration: 300, // 时长
        iterations: 1, // 设置-1表示动画无限循环
      })
      .onAppear(() => {
        // 组件挂载完毕,修改数值触发动画效果
        this.imgScale = 1.0
      })
  }
}

@ComponentV2
export struct CustomOverlayView {
  @Param olConfig: OverlayConfig = new OverlayConfig("");

  aboutToAppear(): void {
    setTimeout(() => {
      console.log("CustomOverlayView aboutToAppear");
      if (this.olConfig.onCallback != null) {
        this.olConfig.onCallback(this.olConfig.index);
      }
    }, this.olConfig.duration);
  }

  build() {
    Column() {
      if (OverlayType.loading == this.olConfig.type) {
        // message
        LoadingProgress()
          .color($r('app.color.success'))
          .width(40)
          .height(40)
      } else if (OverlayType.success == this.olConfig.type) {
        // message
        OverlayScaleImage({
          src: $r('app.media.ic_hud_success'),
          imgWidth: 40
        })
      } else if (OverlayType.fail == this.olConfig.type) {
        // message
        OverlayScaleImage({
          src: $r('app.media.ic_hud_fail'),
          imgWidth: 30
        })
      }

      Text(this.olConfig.message)
        .fontSize(14)
        .fontColor($r('app.color.white'))
        .margin({
          top: 10,
        })
    }
    .padding({
      top: 20,
      bottom: 20,
      left: 15,
      right: 15
    })
    .justifyContent(FlexAlign.Center)
    .constraintSize({
      minWidth: 180,
      minHeight: 80,
    })
    .backgroundColor($r('app.color.overlay_bg_color'))
    .borderRadius(10)
    .offset(this.olConfig.offset)
  }
}
  • 2.CustomOverlayStorage

由于在OverlayManager来添加、删除、隐藏、显示节点过程中,需要使用index索引参数。这里使用一个类,类中有一个数组记录一下展示的节点Component

@ObservedV2
export class CustomOverlayStorage {
  @Trace contentArray: ComponentContent<OverlayConfig>[] = []
}
  • 3.自定义MyOverlayManager进行封装OverlayManager

首先确定属性uiContext,创建ComponentContent需要该参数,这个我在index.ets中进行初始化传入。

CustomOverlayStorage存储ComponentContent的数组,确定index

overlayManager用来来添加、删除、隐藏、显示节点

MyOverlayManager代码如下

/// 用于管理Overlay
/// 浮层(OverlayManager) 用于将自定义的UI内容展示在页面(Page)之上,
/// 在Dialog、Popup、Menu、BindSheet、BindContentCover和Toast等组件之下,
/// 展示的范围为当前窗口安全区内。可适用于常驻悬浮等场景。
/// 与OverlayManager相关的属性推荐采用AppStorage来进行应用全局存储,以免切换页面后属性值发生变化从而导致业务错误。
import { AppStorageV2, ComponentContent, OverlayManager, router } from '@kit.ArkUI';
import {
  builderCustomOverlayView,
  CustomOverlayStorage,
  OverlayConfig
} from '../common/components/hud/CustomOverlayView';

export class MyOverlayManager {
  private static currentIndex: number = 0;
  private uiContext?: UIContext
  private overlayManager?: OverlayManager
  private overlayStorage: CustomOverlayStorage =
    AppStorageV2.connect(CustomOverlayStorage, 'overlayStorage', () => new CustomOverlayStorage())!;
  private static instance: MyOverlayManager;

  public static getInstance(): MyOverlayManager {
    if (MyOverlayManager.instance == null) {
      MyOverlayManager.instance = new MyOverlayManager();
    }
    return MyOverlayManager.instance;
  }

  initOverlayNode(uiContext: UIContext): void {
    this.uiContext = uiContext;
    this.overlayManager = uiContext.getOverlayManager();
  }

  addOverlayView(overlayConfig: OverlayConfig): void {
    if (this.uiContext != null && this.uiContext != undefined) {
      // 设置索引下标
      let index = MyOverlayManager.currentIndex++;
      overlayConfig.index = index;

      // 创建componentContent
      let componentContent = new ComponentContent(
        this.uiContext!, wrapBuilder<[OverlayConfig]>(builderCustomOverlayView),
        overlayConfig
      )

      this.overlayStorage.contentArray.push(componentContent);

      if (this.overlayManager != null && this.overlayManager != undefined) {
        this.overlayManager.addComponentContent(componentContent, index)
      }
    }
  }

  hideOverlayView(index: number) {
    if (this.overlayManager != null && this.overlayManager != undefined) {
      if (index < this.overlayStorage.contentArray.length) {
        this.overlayManager.hideComponentContent(this.overlayStorage.contentArray[index])
      }
    }
  }

  showOverlayView(index: number) {
    if (this.overlayManager != null && this.overlayManager != undefined) {
      if (index < this.overlayStorage.contentArray.length) {
        this.overlayManager.showComponentContent(this.overlayStorage.contentArray[index])
      }
    }
  }

  removeOverlayView(index: number) {
    if (this.overlayManager != null && this.overlayManager != undefined) {
      if (index < this.overlayStorage.contentArray.length) {
        this.overlayManager.removeComponentContent(this.overlayStorage.contentArray[index])
      }
    }
  }

  removeAllOverlayView() {
    if (this.overlayManager != null && this.overlayManager != undefined) {
      this.overlayManager.hideAllComponentContents();
      for (let index: number = 0; index < this.overlayStorage.contentArray.length; index++) {
        this.overlayManager.removeComponentContent(this.overlayStorage.contentArray[index])
      }
    }
  }
}
  • 4.index.ets传入uiContext

初始化配置uiContext

aboutToAppear(): void {
    console.log("aboutToAppear");
    MyOverlayManager.getInstance().initOverlayNode(this.getUIContext());
  }
  • 5.调用OverlayManager进行显示加载中、加载成功、加载失败提示

定义type及message

let config = new OverlayConfig(message);
    config.type = OverlayType.loading;
    config.onCallback = (index: number)=>{
      MyOverlayManager.getInstance().removeOverlayView(index)
    }
    MyOverlayManager.getInstance().addOverlayView(config);

加载中、加载成功、加载失败的Util

import { MyOverlayManager } from "../../manager/MyOverlayManager";
import { OverlayConfig, OverlayType } from "../components/hud/CustomOverlayView";

export class EasyLoadingHud {
  static showLoading(message: string) {
    let config = new OverlayConfig(message);
    config.type = OverlayType.loading;
    config.onCallback = (index: number)=>{
      MyOverlayManager.getInstance().removeOverlayView(index)
    }
    MyOverlayManager.getInstance().addOverlayView(config);
  }

  static showSuccess(message: string) {
    let config = new OverlayConfig(message);
    config.type = OverlayType.success;
    config.onCallback = (index: number)=>{
      MyOverlayManager.getInstance().removeOverlayView(index)
    }
    MyOverlayManager.getInstance().addOverlayView(config);
  }

  static showFail(message: string) {
    let config = new OverlayConfig(message);
    config.type = OverlayType.fail;
    config.onCallback = (index: number)=>{
      MyOverlayManager.getInstance().removeOverlayView(index)
    }
    MyOverlayManager.getInstance().addOverlayView(config);
  }
}
  • 6.页面调用EasyLoadingHud进行显示

可以在页面需要的地方调用EasyLoadingHud进行显示

代码如下

// 加载中
EasyLoadingHud.showLoading("加载中...")
// 加载成功
EasyLoadingHud.showSuccess("加载成功")
// 加载失败
EasyLoadingHud.showFail("加载失败")

四、小结

在开发过程中会遇到提示弹窗、加载中指示器、加载失败的toast等功能,这里是学习HarmonyOS鸿蒙开发的学习记录,如果对你有用,你可以点个赞哦~~。详细的文档还是以官方文档为主。

相关推荐
程序猿阿伟2 小时前
《探秘鸿蒙NEXT中的人工智能核心架构》
人工智能·架构·harmonyos
轻口味2 小时前
HarmonyOS Next 日志工具介绍
华为·pdf·harmonyos
青椒10138 小时前
【习题】<HarmonyOS第一课>应用程序框架基础
harmonyos
李洋-蛟龙腾飞公司8 小时前
HarmonyOS NEXT 应用开发练习:AI智能语音播报
人工智能·harmonyos
谢道韫66614 小时前
鸿蒙面试 2025-01-09
华为·harmonyos
青椒101317 小时前
25年01月HarmonyOS应用基础认证最新题库
华为·harmonyos·鸿蒙系统
HarmonyOS_SDK20 小时前
主体分割技术,提升图像信息提取能力
harmonyos
咸鱼过江1 天前
OpenHarmony编译构建流程概览[源码级]
python·shell·harmonyos