跳到主要内容

弹出框蒙层控制

开发者对弹出框的定制不仅限于弹出框里的内容,对弹出框蒙层的定制需求也逐渐增加。本文介绍ArkUI弹出框的蒙层控制,包括点击蒙层时是否消失、蒙层区域、蒙层颜色和蒙层动画等特性。

使用约束

ArkUI提供多种弹出框,不同类型的弹出框具备不同的蒙层定制能力。详情请参阅下表:

接口&组件autoCancelmaskRectisModalimmersiveMode
openCustomDialog支持支持支持支持
openCustomDialogWithController支持支持支持支持
presentCustomDialog支持支持支持支持
updateCustomDialog支持不支持不支持不支持
CustomDialog支持支持支持支持
showDialog不支持支持支持支持
showAlertDialog支持支持支持支持
showActionSheet支持支持支持支持
showActionMenu不支持不支持支持支持
showDatePickerDialog不支持支持不支持不支持
CalendarPickerDialog不支持不支持不支持不支持
showTimePickerDialog不支持支持不支持不支持
showTextPickerDialog不支持支持不支持不支持

  • 设置autoCancel参数,可控制弹出框蒙层被点击时是否消失。
  • 设置maskRect参数,可定制弹出框的蒙层的大小和位置。此外,蒙层范围内的事件无法透传,而蒙层范围外的事件可以透传。
  • 设置isModal参数,可定制弹出框的模态状态:非模态弹出框无蒙层,支持与周围组件交互;模态弹出框有蒙层,禁止与周围组件交互。
  • 从API version 15开始,当levelMode属性设置为LevelMode.EMBEDDED时,设置immersiveMode参数,可定制弹出框蒙层是否延伸至状态栏及导航栏。
接口&组件maskColortransitionmaskTransition
openCustomDialog支持支持支持
openCustomDialogWithController支持支持支持
presentCustomDialog支持支持支持
updateCustomDialog支持不支持不支持
CustomDialog支持不支持(可由openAnimation和closeAnimation替代)不支持
showDialog不支持不支持不支持
showAlertDialog不支持支持不支持
showActionSheet不支持支持不支持
showActionMenu不支持不支持不支持
showDatePickerDialog不支持不支持不支持
CalendarPickerDialog不支持不支持不支持
showTimePickerDialog不支持不支持不支持
showTextPickerDialog不支持不支持不支持

  • 设置maskColor参数,可定制弹出框蒙层的颜色。
  • 设置openAnimation参数,可定制弹出框的进入动画,同时影响蒙层动画。该接口仅支持简单的动画设置,不支持复杂动画定制。
  • 设置closeAnimation参数,可定制弹出框的退出动画,同时影响蒙层动画。该接口仅支持简单的动画设置,不支持复杂动画定制。
  • 设置transition参数,可定制弹出框的进入和退出动画,同时影响蒙层动画。
  • 从API version 19开始,设置maskTransition参数,可定制弹出框的蒙层动画。

弹出框蒙层显隐控制

通过autoCancel和isModal属性控制弹出框的蒙层显隐。

设置autoCancel为false,取消默认点击蒙层时弹窗消失。

autoCancelOpt: promptAction.CustomDialogOptions = {
builder: () => {
this.myBuilder();
},
autoCancel: false,
} as promptAction.CustomDialogOptions;
// ···
build() {
NavDestination() {
Column() {
Button('openCustomDialog autoCancel:false')
.width('100%')
.margin({ top: 10 })
.onClick(() => {
this.getUIContext().getPromptAction().openCustomDialog(this.autoCancelOpt)
})

// ···
}
.width('100%')
.height('100%')
}
}

设置isModal为false,将默认的模态弹出框变为非模态弹出框。

modalOpt: promptAction.CustomDialogOptions = {
builder: () => {
this.myBuilder();
},
isModal: false,
} as promptAction.CustomDialogOptions;
// ···
build() {
NavDestination() {
Column() {
// ···
Button('openCustomDialog isModal:false')
.width('100%')
.margin({ top: 10 })
.onClick(() => {
this.getUIContext().getPromptAction().openCustomDialog(this.modalOpt)
})

// ···
}
.width('100%')
.height('100%')
}
}

弹出框蒙层样式控制

该示例通过maskRect、immersiveMode和maskColor展示弹出框在蒙层样式控制方面的能力。

设置maskRect和maskColor,实现蒙层区域和蒙层颜色的设置。

maskOpt: promptAction.CustomDialogOptions = {
builder: () => {
this.myBuilder();
},
maskRect: {
x: 0,
y: 10,
width: '100%',
height: '90%'
},
maskColor: '#33AA0000'
} as promptAction.CustomDialogOptions;
// ···
build() {
NavDestination() {
Column() {
// ···
Button('openCustomDialog maskOpt')
.width('100%')
.margin({ top: 10 })
.onClick(() => {
this.getUIContext().getPromptAction().openCustomDialog(this.maskOpt)
})

// ···
}
.width('100%')
.height('100%')
}
}

在levelMode为LevelMode.EMBEDDED下,展示不同immersiveMode对蒙层在导航栏和状态栏的延伸效果。

@State immersiveMode: ImmersiveMode = ImmersiveMode.DEFAULT;
// ···
build() {
NavDestination() {
Column() {
// ···
Button('openCustomDialog immersiveMode')
.width('100%')
.margin({ top: 10 })
.onClick(() => {
this.immersiveMode =
this.immersiveMode == ImmersiveMode.DEFAULT ? ImmersiveMode.EXTEND : ImmersiveMode.DEFAULT;
this.getUIContext().getPromptAction().openCustomDialog({
builder: () => {
this.myBuilder();
},
levelMode: LevelMode.EMBEDDED,
immersiveMode: this.immersiveMode,
})
})

// ···
}
.width('100%')
.height('100%')
}
}

弹出框蒙层动画控制

该示例通过transition和maskTransition分别展示弹出框在蒙层动画方面的能力。

设置transition,实现弹出框与蒙层整体的动画。

transitionOpt: promptAction.CustomDialogOptions = {
builder: () => {
this.myBuilder();
},
transition: TransitionEffect.OPACITY.animation({ duration: 3000 })
} as promptAction.CustomDialogOptions;
// ···
build() {
NavDestination() {
Column() {
// ···
Button('openCustomDialog transition')
.width('100%')
.margin({ top: 10 })
.onClick(() => {
this.getUIContext().getPromptAction().openCustomDialog(this.transitionOpt);
})

// ···
}
.width('100%')
.height('100%')
}
}

设置maskTransition,实现弹出框中蒙层单独的动画定制能力。

Button('openCustomDialog maskTransition')
.width('100%')
.margin({ top: 10 })
.onClick(() => {
this.getUIContext().getPromptAction().openCustomDialog({
builder: () => {
this.myBuilder();
},
maskTransition: TransitionEffect.OPACITY.animation({ duration: 2000 })
.combine(TransitionEffect.rotate({ z: 1, angle: 180 })),
});
})

CustomDialog虽然不支持transition接口,但与之对应的openAnimation和closeAnimation接口在动画的打开和关闭时可进行定制,示例代码如下:

// xxx.ets

@CustomDialog
@Component
struct CustomDialogAnimationBuilder {
controller?: CustomDialogController;

build() {
Column() {
Text('title')
.margin(10)
.fontSize(20)
Button('button1')
.margin(10)
.fontSize(20)
.onClick(() => {
this.controller?.close();
})
Button('button2')
.margin(10)
.fontSize(20)
.onClick(() => {
this.controller?.close();
})
}.width('100%')
.height('50%')
}
}

@Entry
@Component
export struct CustomDialogAnimation {
animationController: CustomDialogController | null =
new CustomDialogController({
builder: CustomDialogAnimationBuilder(),
closeAnimation: { duration: 2000 },
openAnimation: { duration: 2000 }
});

aboutToDisappear(): void {
this.animationController = null;
}

build() {
NavDestination() {
Column() {
Button('CustomDialogController animate')
.width('100%')
.margin({ top: 10 })
.onClick(() => {
this.animationController?.open();
})
}
}
}
}

完整示例

// xxx.ets
import { ImmersiveMode, LevelMode, promptAction } from '@kit.ArkUI';

@Entry
@Component
export struct CustomDialogControl {
@State immersiveMode: ImmersiveMode = ImmersiveMode.DEFAULT;

autoCancelOpt: promptAction.CustomDialogOptions = {
builder: () => {
this.myBuilder();
},
autoCancel: false,
} as promptAction.CustomDialogOptions;

modalOpt: promptAction.CustomDialogOptions = {
builder: () => {
this.myBuilder();
},
isModal: false,
} as promptAction.CustomDialogOptions;

maskOpt: promptAction.CustomDialogOptions = {
builder: () => {
this.myBuilder();
},
maskRect: {
x: 0,
y: 10,
width: '100%',
height: '90%'
},
maskColor: '#33AA0000'
} as promptAction.CustomDialogOptions;

transitionOpt: promptAction.CustomDialogOptions = {
builder: () => {
this.myBuilder();
},
transition: TransitionEffect.OPACITY.animation({ duration: 3000 })
} as promptAction.CustomDialogOptions;

@Builder
myBuilder() {
Column() {
Text('title').margin(10).fontSize(20)
Button('button1').margin(10).fontSize(20)
Button('button2').margin(10).fontSize(20)
}.width('100%').height('50%')
}

build() {
NavDestination() {
Column() {
Button('openCustomDialog autoCancel:false')
.width('100%')
.margin({ top: 10 })
.onClick(() => {
this.getUIContext().getPromptAction().openCustomDialog(this.autoCancelOpt)
})

Button('openCustomDialog isModal:false')
.width('100%')
.margin({ top: 10 })
.onClick(() => {
this.getUIContext().getPromptAction().openCustomDialog(this.modalOpt)
})

Button('openCustomDialog maskOpt')
.width('100%')
.margin({ top: 10 })
.onClick(() => {
this.getUIContext().getPromptAction().openCustomDialog(this.maskOpt)
})

Button('openCustomDialog transition')
.width('100%')
.margin({ top: 10 })
.onClick(() => {
this.getUIContext().getPromptAction().openCustomDialog(this.transitionOpt);
})

Button('openCustomDialog immersiveMode')
.width('100%')
.margin({ top: 10 })
.onClick(() => {
this.immersiveMode =
this.immersiveMode == ImmersiveMode.DEFAULT ? ImmersiveMode.EXTEND : ImmersiveMode.DEFAULT;
this.getUIContext().getPromptAction().openCustomDialog({
builder: () => {
this.myBuilder();
},
levelMode: LevelMode.EMBEDDED,
immersiveMode: this.immersiveMode,
})
})

Button('openCustomDialog maskTransition')
.width('100%')
.margin({ top: 10 })
.onClick(() => {
this.getUIContext().getPromptAction().openCustomDialog({
builder: () => {
this.myBuilder();
},
maskTransition: TransitionEffect.OPACITY.animation({ duration: 2000 })
.combine(TransitionEffect.rotate({ z: 1, angle: 180 })),
});
})
}
.width('100%')
.height('100%')
}
}
}