Android异形屏适配(官方方案)-程序员宅基地

技术标签: 安卓  刘海屏  Android开发  # Android设备兼容  水滴屏  异形屏  

一、前言

    Android的异形屏,包括刘海屏,水滴屏、挖孔屏,起初是iOS设备上的杰作,有吐槽,也有赞美。刚出来不久,国内的各大厂商开始效仿,起初官方并没有API进行适配,一些厂商(例如小米、vivo)自己搞了刘海屏,只能用自己的API进行检测适配,这些就是蛋疼的事,通用的做法就是利用反射调用系统API进行判断,不过这篇文章不介绍这些设备的做法,毕竟从Android 9.0 (API Level 28)开始Android官方也出了刘海屏的适配支持,这里主要将官方的(因为现在国内各大厂商出的设备都是基于新系统,都支持官方API检测和适配)

二、异形屏适配详解

2.1 官方对带异形屏设备的要求

    为了确保一致性和应用兼容性,官方对搭载 Android 9 的设备有以下要求:

  • 一条边缘最多只能包含一个刘海。
  • 一台设备不能有两个以上的刘海。
  • 设备的两条较长边缘上不能有刘海。
  • 在未设置特殊标志的竖屏模式下,状态栏的高度必须至少与刘海的高度持平。
  • 默认情况下,在全屏模式或横屏模式下,整个刘海区域必须显示黑边。

2.2 处理异形屏

2.2.1 不隐藏系统状态栏的情形

    如果应用所有界面均不隐藏状态栏,也就是应用不与系统状态栏重叠,那么就无需处理异形屏适配,系统状态栏会自动调整占据了异形切口的位置。如下图:

非全屏状态无需处理异形屏适配

2.2.2 隐藏系统状态栏的情形

    如果应用有界面是隐藏系统状态栏的(比如启动页,全屏游戏等),那么就要针对异形屏进行适配。

2.2.2.1 配置应用如何处理异形切口区域

    隐藏了系统状态栏,意味着应用的内容将扩充到系统状态栏原有的位置,但是系统提供了控制是否在异形切口区域显示内容的配置,该配置是Window(窗口)级别属性,属性名为 layoutInDisplayCutoutMode。该属性有三个可选值,分别是:

  • WindowManager.LayoutParams.LAYOUT_IN_DISPLAY_CUTOUT_MODE_DEFAULT - 这是默认行为。官方说明是在竖屏模式下,内容会呈现到刘海区域中;但在横屏模式下,内容会显示黑边。但是笔者才使用过程中,发现横屏和竖屏都是留黑边,不知道是不是跟系统有关(笔者使用官方模拟器和官方原生系统设备均是留黑边效果)
  • WindowManager.LayoutParams.LAYOUT_IN_DISPLAY_CUTOUT_MODE_SHORT_EDGES - 在竖屏模式和横屏模式下,内容都会呈现到刘海区域中。
  • WindowManager.LayoutParams.LAYOUT_IN_DISPLAY_CUTOUT_MODE_NEVER - 内容从不呈现到刘海区域中。

可以通过了两种方法配置该属性:

  • 在窗口主题样式文件中定义
    在主题样式文件中通过 android:windowLayoutInDisplayCutoutMode 定义,示例代码如下:
<style name="Theme.DisplayCutoutDemo" parent="Theme.MaterialComponents.DayNight.DarkActionBar">
    <!-- 其他属性配置 -->
    <item name="android:windowLayoutInDisplayCutoutMode">shortEdges</item>
</style>
  • 在窗口代码中定义
    在代码中定义,需要在Window(窗口)级别属性中配置,示例代码如下:
window.attributes.apply {
    
    // Window级别的全屏(这里的代码可以)
    systemUiVisibility =
        View.SYSTEM_UI_FLAG_FULLSCREEN or
                View.SYSTEM_UI_FLAG_LAYOUT_FULLSCREEN or
                View.SYSTEM_UI_FLAG_LAYOUT_STABLE or
    // 下面两个是隐藏虚拟导航栏
                View.SYSTEM_UI_FLAG_HIDE_NAVIGATION or
                View.SYSTEM_UI_FLAG_LAYOUT_HIDE_NAVIGATION

    // 设置视图内容是否显示到异形切口区域
    if(Build.VERSION.SDK_INT >= Build.VERSION_CODES.Q) {
    
        // 如果最低版本支持小于28,需要增加判断,防止在低版本系统运行时找不到系统API崩溃
        layoutInDisplayCutoutMode = WindowManager.LayoutParams.LAYOUT_IN_DISPLAY_CUTOUT_MODE_SHORT_EDGES
    }
}
  • 效果
    窗口内容全屏显示到异形切口区域
        上面的效果中,发现异形切口区域没有黑边,但是也没有达到预想的内容显示在异形切口区域内,不仅如此,下面导航栏区域也是白边。这是因为 Android 系统不允许视图内容跟系状态栏和导航栏区域重叠,要让视图内容强制延伸至系统状态栏和导航栏区域,可以通过配置视图级别的显示属性参数 systemUiVisibility,代码如下所示:
window.decorView.apply {
    
    // 视图级全屏
    systemUiVisibility =
        View.SYSTEM_UI_FLAG_FULLSCREEN or
                // 视图内容延伸到状态栏区域
                View.SYSTEM_UI_FLAG_LAYOUT_FULLSCREEN or
                View.SYSTEM_UI_FLAG_LAYOUT_STABLE or
                // 视图内容延伸到系统导航栏区域
                View.SYSTEM_UI_FLAG_HIDE_NAVIGATION or
                View.SYSTEM_UI_FLAG_LAYOUT_HIDE_NAVIGATION
}
  • 效果
    视图延伸到异形切口区域和系统导航栏区域
2.2.2.2 异形切口区域的视图适配

    如果你想将内容扩展到异形切口,如果只是遮挡部分的内容不影响体验(例如启动页图片),可不做处理,但是如果有用户可交互的操作或者文字,必须进行适配。这时,你需要知道异形切口所在的位置和大小,在 Android 9 开始,系统提供了 WindowInsets.getDisplayCutout() 接口获取 DisplayCutout 对象,这时一个描述屏幕中所有切口位置和大小的对象。

  • 如何获取 DisplayCutout 对象
        获取 DisplayCutout 对象,首先要获取 WindowInsets 对象。 WindowInsets 对象并没有直接获取方法,只能通过监听或者回调方法获取,下面介绍获取 WindowInsets 对象的两种方法。
  1. 在自定义View内部重写 View.onApplyWindowInsets(WindowInsets)方法获取
class CView(context: Context?) : LinearLayout(context) {
    
    override fun onApplyWindowInsets(insets: WindowInsets?): WindowInsets {
    
        return super.onApplyWindowInsets(insets)
    }
}

注意事项:如果是重写View.onApplyWindowInsets(WindowInsets)方法获取,请确保 WindowInsets 在之前没有被消耗,没有给 View 的父级或者 View 设置 View.OnApplyWindowInsetsListener 监听。这个大家可以去了解下 WindowInsets 分发相关知识。

  1. 通过给 View 设置 View.OnApplyWindowInsetsListener 监听获取
window.decorView.apply {
    
    if(Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT_WATCH) {
    
        // 该方法在 API 20 开始加入,需要做好版本适配
        setOnApplyWindowInsetsListener {
     v, insets ->
            insets
        }
    }
}
  • 如何获取屏幕切口区域
        上面已经介绍了 WindowInsets 对象的获取方法,有了 WindowInsets 对象,我们可以通过 WindowInsets.getDisplayCutout() 获取切口相关信息。在 API Level 28 中,该对象只能支持获取屏幕的安全区域,分别通过 WindowInsets.getSafeInsetBottom()WindowInsets.getSafeInsetLeft()WindowInsets.getSafeInsetRight()WindowInsets.getSafeInsetTop() 获取各边安全间距。在 API Level 29 开始,还支持获取具体的切口位置和大小信息,分别通过 WindowInsets.getBoundingRectBottom()WindowInsets.getBoundingRectLeft()WindowInsets.getBoundingRectRight()WindowInsets.getBoundingRectTop() 获取各边的切口信息,知道切口信息,可以更加精准地控制显示。

实用小知识:这里介绍到了 WindowInsets 对象,从 API 21开始,可以通过这个对象获取状态栏和导航栏的高度,这些接口分别是 WindowInsets.getStableInsetBottom()WindowInsets.getStableInsetLeft()WindowInsets.getStableInsetRight()WindowInsets.getStableInsetTop(),在API Level 30 开始这几个接口被废弃,使用 WindowInsets.getInsetsIgnoringVisibility (int typeMask) 替代。用这种方法获取的时候,即使状态栏和导航栏处于隐藏状态,也不影响获取。

在这里,我们演示下比较简单的安全区域的适配方法。

if(Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT_WATCH) {
    
// 如果最低版本支持小于20,需要增加判断,防止在低版本系统运行时找不到系统API崩溃
setOnApplyWindowInsetsListener {
     v, insets ->
    if(Build.VERSION.SDK_INT >= Build.VERSION_CODES.P) {
    
        insets.displayCutout?.apply {
    
            textView.apply {
    
                // 设置padding,防止内容显示到非安全区域
                setPadding(safeInsetLeft, safeInsetTop, safeInsetRight, safeInsetBottom)
            }
        }
    }
    // 不消费,直接返回原始对象
    insets
}

适配前后对比

2.2.3 异形屏适配的最佳做法与注意事项

  1. 不要让异形切口区域遮盖任何重要的文本、控件或其他信息。
  2. 不要将任何需要精细轻触识别的交互式元素放置或延伸到异形切口区域,异形切口区域中的轻触灵敏度可能要比其他区域低一些。
  3. 避免对状态栏高度进行硬编码,否则可能会导致内容重叠或被切断。
  4. 如果的应用需要进入全屏模式,使用shortEdges 模式,退出全屏是,使用 never 模式。
  5. 在全屏模式下,在使用窗口坐标与屏幕坐标时需要注意,因为在显示黑边的情况下,窗口不会占据整个屏幕。因此根据屏幕原点(屏幕左上角)得到的坐标与根据窗口原点(窗口左上角)得到的坐标不再相同。不过以根据需要使用 View.getLocationOnScreen() 接口将屏幕坐标转换为视图坐标。在处理 MotionEvent 时,应当使用 MotionEvent.getX()MotionEvent.getY() 来避免类似的坐标问题。不要使用 MotionEvent.getRawX()MotionEvent.getRawY()
版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/yingaizhu/article/details/112894902

智能推荐

计算机的外围设备简介_计算机外围固定-程序员宅基地

文章浏览阅读6.1k次,点赞3次,收藏5次。外围设备介绍计算机的外围设备(简称外设)虽然很多,但按功能分大类只有四类:输入、输出、存储、网络通讯。有些专业计算机需要的外围设备也不尽相同,并不都需要这四类外围设备。外围设备可以按需要组装,有些专业计算机甚至可以将存储设备和主芯片集成到一片芯片上,从而不再需要外加存储设备。最早的计算机(那时还只能称为计算器,只能做简单运算,如ABC机和ENIAC机)输入只是一些拨码开关,只能输入数字(还得是二进_计算机外围固定

java 图片中加文字_java怎么在图片上加文字-程序员宅基地

文章浏览阅读1.5k次。java 图片中加文字_java怎么在图片上加文字

GBase8cGDCA认证模拟题题库(三)_如果需要打开delete语句的审计功能,需要开启下面哪个参数-程序员宅基地

文章浏览阅读720次,点赞20次,收藏6次。B 选项,在创建模式时,可以不指定模式名。C 选项,兼容模式可选值为 AB、C、PG.安装GBase 8c分布式集群时所需的配置文件gbase.yml,在解压GBase8cV5 S3.0.0BXX CentOS x86 64.tar.bz2压缩包生成的目录中得到。真值的有效文本值是: TRUE、t、"true'、y、yes'、"1'TRUE'、true、整数范围内1~2^63-1、整数范围内-1~-2^63。GBase 8c 使用create table 创建表时,不指定参数,默认是astore,行存表。_如果需要打开delete语句的审计功能,需要开启下面哪个参数

xml文件中几个名词_xml文件里面的名词-程序员宅基地

文章浏览阅读334次。1 xmlns是XML Namespaces的缩写,中文名称是XML(标准通用标记语言的子集)命名空间。 web-app是web.xml的根节点标签名称 version是版本的意思 xmlns是web.xml文件用到的命名空间 xmlns:xsi是指web.xml遵守xml规范 xsi:schemaLocation是指具体用到的schema资源_xml文件里面的名词

【OpenGL】中点圆、椭圆生成算法_用setpixel函数中点画圆算法代码c++-程序员宅基地

文章浏览阅读1.6w次,点赞12次,收藏69次。OpenGL 中点圆、椭圆生成算法_用setpixel函数中点画圆算法代码c++

HTML-CSS实现背景图片出现不同的位置_css背景图高度占据一半另一半有别的背景色-程序员宅基地

文章浏览阅读2.1k次。首先在HTML中写入div,命名为img,在这个div中加入一个span标签并命名为img-bg和img50(5星为50).<div class="img"> <span class="img-bg img50"></span> <span class="img-bg img45"></span> <span class="img-bg img40"></span> </div> 在css代码._css背景图高度占据一半另一半有别的背景色

随便推点

matlab建模DNA双链,PPT绘制科研图形—DNA双链、分子细胞模型-程序员宅基地

文章浏览阅读1.3k次。原标题:PPT绘制科研图形—DNA双链、分子细胞模型 PPT绘制DNA双链 1用矩形工具画一个矩形如下,线条颜色设置为无,填充色如下图蓝色 2选中矩形框,选择菜单栏的“格式—— 编辑形状——转换为任意多边形” 3这个时候再看下“编辑形状”,可以看到“编辑顶点” 已经为可用状态 4点击“编辑顶点“,矩形框四个角变为黑色实点。可以拖动实点变为如下图示。然后在边缘上右键,选择”添加顶点“,添加如下顶点 ..._matlab双螺旋结构模型图怎么画

duilib vs2015 安装_DuiLib(1)——简单的win32窗口-程序员宅基地

文章浏览阅读169次。资源下载https://yunpan.cn/cqF6icWRN5CTc 访问密码 92e3 注:DUILIB库.7z 是vs2015下编译好的动态库及静态库,如上图所示一、新建一个win32工程项目设置中选择:debug,常规中:全程无优化-全程无优化,多线程调试 (/MTd);我的项目选择的是静态编译,使用的是静态库,就不需要带duilib.dll文件了代码如下:#include #inclu..._vs2015使用duilib

OpenGL: 渲染管线理论详解_通过此次实验你对固定渲染管线的opengl编程有什么了解。-程序员宅基地

文章浏览阅读5k次,点赞4次,收藏13次。学习着色器,并理解着色器的工作机制,就要对OpenGL的固定功能管线有深入的了解。首先要知道几个OpenGL的术语:渲染(rendering):计算机根据模型(model)创建图像的过程。模型(model):根据几何图元创建的物体(object)。几何图元:包括点、直线和多边形等,它是通过顶点(vertex)指定的。 最终完成了渲染的图像是由在屏幕上绘制的像素组成的。在内存中,和像素有关的信息(如像素的颜色)组织成位平面的形式,位平面是一块内存区域,保存了屏幕上每个像素的一个位的信息。_通过此次实验你对固定渲染管线的opengl编程有什么了解。

Android MPAndroidChart:动态添加统计数据线【8】_android 动态统计-程序员宅基地

文章浏览阅读3.9k次。Android MPAndroidChart:动态添加统计数据线【8】本文在附录相关文章6的基础上,动态的依次增加若干条统计折线(相当于批量增加数据点)。布局文件:

vmware中的linux虚拟机如何增加磁盘容量_linux虚拟机磁盘空间不足-程序员宅基地

文章浏览阅读6.3k次。vmware中 centos的磁盘大小 20G->30G现象:fdisk -l可以看到增大后的磁盘总量,但是需要增加分区并格式化然后挂载才能使用.一、vmware中的设置先关闭虚拟机vm->settings->hard disk->utilities->expand->输入大小(增加后的大小)二、启动虚拟机,进入命令行1、 fdisk /dev/sda进入命令行Comman_linux虚拟机磁盘空间不足

Hadoop2.7.3下Mysql8.0下Hive2.3.8的安装_hive2.3.8安装-程序员宅基地

文章浏览阅读927次。hive安装前提:1.基于hadoop2.7的完全分布式集群搭建完成hadoop2.7集群搭建2.MySQL8.0安装完成 安装centos7上MySQL8.0Hive2.3.8的安装下载链接:https://mirrors.tuna.tsinghua.edu.cn/apache/下滑找到hive点击进去点击hive2.3.9(hive2.3.9和hive2.3.8差别不大)下载画红线的也就是bin.tar.gz后缀的hive解压安装下载完成后通过xftp传到虚拟机上(基操不在赘述)_hive2.3.8安装

推荐文章

热门文章

相关标签