首页 > 其他分享 >HarmonyOS鸿蒙开发 - 解决上下两栏白边 - 沉浸式效果

HarmonyOS鸿蒙开发 - 解决上下两栏白边 - 沉浸式效果

时间:2024-12-18 09:42:36浏览次数:4  
标签:两栏 topRectHeight 状态栏 沉浸 白边 HarmonyOS setWindowLayoutFullScreen 设置 页面

鸿蒙应用开发从入门到入行

预览器上下两栏白边

  • 自从HarmonyOS升级到release版后,很多同学会问猫林老师:为什么他的预览器上下有白边,为什么明明根容器写了宽高百分百但没铺满。如下图

    image-20241210141135933

白边原因

  • 其实上面的白边,称之为状态栏。上面会放手机wifi信号、电池电量等信息。一般情况下我们不需要把应用中具有交互效果的界面延伸到上面去,免得影响操作。

  • 同样,下面的白边称之为导航栏,也即切换手机内应用的地方。会有一个小横条方便你切换不同应用以及回到桌面。

  • 如下图所示

    image-20241210142736603

  • 而HarmonyOS升级到release版本后,特意在预览器里把这上下两栏给你留白空出来,就是为了方便让开发者知道,自己的界面并没占用这两个区域,所以一般情况下,如果你的应用整体背景颜色就是白色的,其实是无需处理的。

沉浸式效果介绍

  • 根据上面说的白边情况,如果你的app背景色正好也是白色,那么可以和上下白边融为一体,显得不那么突兀。但如果你的app是别的颜色,那么可能会有明显的突兀感。

  • 举个例子:大家经常用的美团。我们看看它目前的情况,以及假设有白边的情况

    image-20241210143740406 这是美团正常情况,会看到顶部是黄色,状态栏也变为黄色,视觉效果上浑然一体

  • 以下假设状态栏白色

    image-20241210144614483

    可以看到视觉效果上会比较突兀

  • 通过对比我们发现,确实在实际app开发过程中,状态栏上可以不放任何界面元素,但是需要将状态栏的颜色定义的与app背景色保持一致,才会视觉上显得更好看,更融为一体。像这样的效果,我们称之为沉浸式效果

  • 通过上面的描述我们已经发现沉浸式效果能提供比较好的视觉效果,但如何实现呢?

  • 有三种方案都可以实现:

    • 通过设置Window背景色来实现
    • 通过调用窗口强制全屏布局接口setWindowLayoutFullScreen() + padding避让实现 (麻烦)
    • 直接使用扩展到避让区功能

通过设置Window背景色实现沉浸式

  • 设置窗体背景色实现

    • 先看不设置的情况下,我们写的一个宽高百分百,且背景颜色为红色的界面,如下图,可以看到状态栏和整体背景色不一致,有明显突兀感

      image-20241210150551860

    • 此时,我们可以设置窗体全局背景色也为红色实现视觉沉浸,来到EntryAbility.ets,找到onWindowStageCreate生命周期函数,在windowStage.loadContent回调里设置如下代码即可

      windowStage.getMainWindowSync().setWindowBackgroundColor('#ff0000')
      
      
    • 注意:这里只能给16进制颜色,且必须满6位

    • 效果如下

      image-20241210151024558

      • 此时浑然一体
    • 这种方法虽然简单,但有缺点:

      1. 预览器依然会有白边,只有模拟器或真机运行才能看到效果

      2. 它写死了颜色,每个App里不管是哪个页面都是此颜色,假如你App里多个页面的主题颜色不一样,会导致非常突兀,如下图

      image-20241210151943780

使用setWindowLayoutFullScreen实现沉浸式

  • 这是Window提供的一个方法,可以设置让App整屏(即覆盖状态栏与导航栏)实现整块屏幕都可以布局,但是大部分使用时必须配合避让偏移,否则会有问题。至于什么问题呢,我们往下看。

  • 首先来到EntryAbility.ets,继续找到onWindowStageCreate生命周期函数,在windowStage.loadContent回调里设置如下代码即可

    windowStage.getMainWindow().then(w => {
        // 设置占用全屏
        w.setWindowLayoutFullScreen(true)
    })
    
    
  • 这样虽然实现了沉浸式效果,但也存在了问题,例如,我们第一页中本来有Button,但是此时Button位置跑到原来的状态栏去了,如下图

    image-20241210152535258

  • 这样的话,会导致原本不该布局的区域也会存在我们的布局元素。第一巨丑,第二用户也点击不了。

  • 因此,我们使用这个方法实现沉浸式时,一般还要做让页面根容器padding避让。也即让我们布局的组件,通过padding的方式挪动他们位置,避让原本的状态栏和导航栏。

  • 例:

    Column() {
        Button('去下一页')
          .onClick(() => {
            router.pushUrl({
              url: 'pages/Second'
            })
          })
      }
      .width('100%')
      .height('100%')
      .backgroundColor(Color.Red)
      .padding({ top: 50, bottom: 50 })
    
    
  • 此时,我们发现写的按钮确实不会被刘海屏挡住了。但是细心的同学发现了,我们这里写死的50vp。不合理,有可能给少了,也有可能给多了。毕竟不同设备的状态栏可能不一样。所以如果我们使用这种方案还需要获取屏幕的状态栏与导航栏的高度。然后把高度存到本地存储里,方便所有页面都可以使用并设置padding

  • 具体步骤:继续来到onWindowStageCreate,填写如下代码

    onWindowStageCreate(windowStage: window.WindowStage): void {
        // Main window is created, set main page for this ability
        hilog.info(0x0000, 'testTag', '%{public}s', 'Ability onWindowStageCreate');
    
        windowStage.loadContent('pages/Index', (err) => {
          // windowStage.getMainWindowSync().setWindowBackgroundColor('#ff0000')
          if (err.code) {
            hilog.error(0x0000, 'testTag', 'Failed to load the content. Cause: %{public}s', JSON.stringify(err) ?? '');
            return;
          }
          
          // 以下是设置沉浸式,以及获取设备导航条、状态栏高度的代码
          windowStage.getMainWindow().then(w => {
            // 设置沉浸式 
            w.setWindowLayoutFullScreen(true)
            // 获取设备区域参数
            let avoidArea = w.getWindowAvoidArea(window.AvoidAreaType.TYPE_SYSTEM)
            let bottomRectHeight = avoidArea.bottomRect.height; // 获取到导航条区域的高度
            AppStorage.setOrCreate('bottomRectHeight', bottomRectHeight); // 存到本地存储
            let topRectHeight = avoidArea.topRect.height; // 获取状态栏区域高度
            AppStorage.setOrCreate('topRectHeight', topRectHeight); // 存到本地存储
    
            // 当区域发生改变(例如竖屏变横屏),重新获取一次再保存
            w.on('avoidAreaChange', (data) => {
              if (data.type === window.AvoidAreaType.TYPE_SYSTEM) {
                let topRectHeight = data.area.topRect.height;
                AppStorage.setOrCreate('topRectHeight', topRectHeight);
              } else if (data.type == window.AvoidAreaType.TYPE_NAVIGATION_INDICATOR) {
                let bottomRectHeight = data.area.bottomRect.height;
                AppStorage.setOrCreate('bottomRectHeight', bottomRectHeight);
              }
            });
          })
          hilog.info(0x0000, 'testTag', 'Succeeded in loading the content.');
        });
      }
    
    
  • 好了,上面每句代码都有注释,可以根据注释去理解。当然咯,我知道你们没兴趣看代码,所以需要用可以直接复制,反正这个代码是固定的

  • 然后来到页面里,先取出本地存储的值,且用@StorageProp装饰器,设置状态自动更新。

      @StorageProp('bottomRectHeight')
      bottomRectHeight: number = 0;
      @StorageProp('topRectHeight')
      topRectHeight: number = 0;
    
    
  • 然后把这两个变量,设置给根容器的padding即可

      @StorageProp('bottomRectHeight')
      bottomRectHeight: number = 0;
      @StorageProp('topRectHeight')
      topRectHeight: number = 0;
    
      build() {
        Column() {
          Button('去下一页')
            .onClick(() => {
              router.pushUrl({
                url: 'pages/Second'
              })
            })
        }
        .width('100%')
        .height('100%')
        .backgroundColor(Color.Red)
        .padding({ top: px2vp(this.topRectHeight), bottom: px2vp(this.bottomRectHeight) })
    
    
  • 同样的,其他页面也如此设置即可

  • 这样,我们通过一系列操作。以及一段代码实现了沉浸式效果。但大家也发现明显的缺点

    1. 预览器依然没效果,需要真机或模拟器查看

    2. 代码过多。好多没耐心的同学看到这,可能都已经烦躁的想打人了

    3. 这样子会让所有页面都被迫使用沉浸式,如果哪个页面不需要沉浸式,还需要再此页面的about里禁用

      aboutToAppear(): void {
          window.getLastWindow(getContext())
            .then(win => {
              win.setWindowLayoutFullScreen(false)
            })
      }
      
      
  • 因此,我们还有最为简单的一种方式,请往下继续看

使用expandSafeArea设置沉浸式(推荐)

  • expandSafeArea是一个按需方式的沉浸式方案,它能完美起到哪个页面需要沉浸式,就在哪个页面使用即可,绝对不会让整个App每个页面都强制沉浸式。而且使用起来非常简单,只需要在需要沉浸式的页面的根容器里设置即可,例

    Column() {
          Button('去下一页')
            .onClick(() => {
              router.pushUrl({
                url: 'pages/Second'
              })
            })
        }
        .width('100%')
        .height('100%')
        .backgroundColor(Color.Red)
        .expandSafeArea([SafeAreaType.SYSTEM], [SafeAreaEdge.TOP, SafeAreaEdge.BOTTOM])
    
    
  • 解释:参数1固定,参数2是设置需要沉浸式的区域,SafeAreaEdge.TOP代表上面状态栏沉浸式,SafeAreaEdge.BOTTOM代表下面导航栏沉浸式,此时效果如下

    image-20241210161011646

    • 没错,此时不需要启动模拟器,预览器也可以直接看到效果!
  • 当然,你也可以只设置让顶部沉浸式,则第二个参数只要写一个TOP即可,如下代码

    Column() {
          // 生略里面代码
      }
        .width('100%')
        .height('100%')
        .backgroundColor(Color.Red)
        .expandSafeArea([SafeAreaType.SYSTEM], [SafeAreaEdge.TOP])
    
    
  • 效果如下

    image-20241210161130556

扩展思考:setWindowLayoutFullScreen是否很鸡肋?

  • 同学们到目前为止,发现实现沉浸式我们用了三种方案。其中第一种设置Window背景色可以无视掉,这种方法我们基本不会用。而第二种方法能实现,但又比较麻烦,第三种是最容易也最推荐的方式。

  • 但此时请我们思考下:setWindowLayoutFullScreen是否真的一点应用场景都没有?

  • 要想回答这个问题,我们可以从setWindowLayoutFullScreen的特点入手,大家还记得吗?setWindowLayoutFullScreen最大的特点是让app所有页面都强制全屏(沉浸式),那么大家仔细想想,有没有哪种App是需要任意页面都强制全屏的呢?

    • 没错,答案是游戏!如下图

    点击放大

  • 像这样的,如果以后是游戏类App,我们必然需要使用setWindowLayoutFullScreen一次性设置所有页面全屏

  • 因此,这个方法,大家也需要有点印象哦!万一哪天要用到呢?

  • 请思考:还有没有除了游戏以外也可能要用到setWindowLayoutFullScreen的场景呢?把你的想法可以打在评论区

标签:两栏,topRectHeight,状态栏,沉浸,白边,HarmonyOS,setWindowLayoutFullScreen,设置,页面
From: https://www.cnblogs.com/xpzll/p/18613951

相关文章

  • HarmonyOS NEXT基础入门
    基础入门储存鸿蒙应用提供多种数据存储方式,包括:Preferences:轻量级数据存储,适用于简单键值对。SQLite:关系型数据库,适用于结构化数据存储。文件存储:通过文件系统直接存储数据。数组数组是一组有序的数据集合,可以通过索引访问每个元素。letnumbers=[1,2,3,4,5];con......
  • 「九」HarmonyOS 5 端云一体化实战项目——「M.U.」应用云侧开发云数据库
    1立意背景M.代表“我”,U.代表“你”,这是一款用于记录情侣从相识、相知、相恋、见家长、订婚直至结婚等各个阶段美好记忆留存的应用程序。它旨在为情侣们提供一个专属的空间,让他们能够将一路走来的点点滴滴,如初次相遇时的心动瞬间、逐渐了解彼此过程中的深入交流、甜蜜......
  • HarmonyOS应用开发--LazyForeach懒加载
    引言在鸿蒙应用开发中,ArkTS(ArkTypeScript)作为开发语言,提供了丰富的API来构建高效、流畅的应用程序。其中,LazyForeach作为一种懒加载技术,对于优化性能和提升用户体验具有重要作用。本文将对LazyForeach懒加载进行深入学习总结。ArkTS简介ArkTS是基于TypeScript的开发语言,专......
  • HarmonyOS 获取进程相关的信息process 常用的几个方法
    获取进程相关的信息,提供进程管理的相关功能。process1.EventListener2.isIsolatedProcess3.is64Bit4.getStartRealtime5.getPastCpuTime导入模块import{process}from'@kit.ArkTS';属性名称类型可读可写说明uidnumber是否进程的用户标识。pidnumber是......
  • HarmonyOS:@Extend装饰器:定义扩展组件样式
    一、前言使用@Styles用于样式的重用,在@Styles的基础上,使用@Extend扩展原生组件样式。说明从APIversion9开始,该装饰器支持在ArkTS卡片中使用。从APIversion11开始,该装饰器支持在元服务中使用。二、装饰器使用说明语法@Extend(UIComponentName)functionfu......
  • 华为HarmonyOS实现跨多个子系统融合的场景化服务 -- 3 打开授权设置页Button
    场景介绍本章节将向您介绍如何使用Button组件打开授权设置页功能,开发者可调用对应Button组件跳转到应用对应的权限设置页面,供用户快速进行应用权限的查看和修改。效果图展示单击“打开授权设置页”按钮,跳转至应用对应的设置页界面。开发步骤导入ScenarioFusionKit模块......
  • HarmonyOS Next 入门实战 - 文字转拼音,文字转语音
    文字转拼音安装pinyin4js三方库ohpminstall@ohos/pinyin4jspinyin4js提供了以下接口:●文字转拼音(带声调和不带声调)●文字转拼音首字母●简体繁体互转letrawText="风急天高猿萧哀,渚清沙白鸟飞回;"letpinyin1:string=pinyin4js.convertToPinyinString(rawT......
  • HarmonyOS Next分布式智能家居控制系统实战
    本文旨在深入探讨基于华为鸿蒙HarmonyOSNext系统(截止目前API12)构建分布式智能家居控制系统的技术细节,基于实际开发实践进行总结。主要作为技术分享与交流载体,难免错漏,欢迎各位同仁提出宝贵意见和问题,以便共同进步。本文为原创内容,任何形式的转载必须注明出处及原作者。一、项......
  • HarmonyOS Next 设备适配与生态拓展
    本文旨在深入探讨华为鸿蒙HarmonyOSNext系统(截止目前API12)在设备适配与生态拓展方面的技术细节,基于实际开发实践进行总结。主要作为技术分享与交流载体,难免错漏,欢迎各位同仁提出宝贵意见和问题,以便共同进步。本文为原创内容,任何形式的转载必须注明出处及原作者。一、设备适配......
  • HarmonyOS Next 应用性能优化实战
    本文旨在深入探讨华为鸿蒙HarmonyOSNext系统(截止目前API12)中应用性能优化的技术细节,基于实际开发实践进行总结。主要作为技术分享与交流载体,难免错漏,欢迎各位同仁提出宝贵意见和问题,以便共同进步。本文为原创内容,任何形式的转载必须注明出处及原作者。一、性能评估指标与工具......