iOS界面布局:屏幕尺寸与安全区域全面指南

news2025/1/23 12:57:42

引言

随着iPhone和iPad的更新迭代,iOS设备的屏幕尺寸和设计也在不断变化。无论是iPhone X系列的刘海屏,还是最新的iPhone 14,开发者都需要面对适配不同设备布局的问题。在项目开发中,导航栏、状态栏、TabBar的高度以及安全区域的处理,直接影响到应用界面的美观与用户的交互体验。如果没有正确处理这些元素,应用在不同设备上的显示效果可能会失衡,甚至影响用户操作。

本文将详细介绍在iOS项目开发中获取这些屏幕的相关参数,帮助你打造在所有设备上都能正常运行的应用界面。

尺寸获取

在本篇博客中我们以Swift为例,在Swift中我们可以直接定义全局常量和全局的计算属性,在项目的任何地方进行引用。

屏幕宽高

屏幕的宽度和高度是开发中最常用的两个尺寸,几乎所有的UI适配都需要以屏幕的宽高为基础进行等分,或者减去左右,上下间距的方式来计算位置。我们可以直接通过主屏幕的bounds属性来获取屏幕的宽高:

/// 屏幕宽度
let ZMSCREEN_WIDTH = UIScreen.main.bounds.width
/// 屏幕高度
let ZMSCREEN_HEIGHT = UIScreen.main.bounds.height

状态栏高度

状态栏的高度也同样重要,状态栏主要用于显示电池和信号信息,在刘海屏没有出现之前状态栏的高度固定为20点(points),这是iOS 7到iOS 10设备上的标准高度。

  • 在iPhone4/4S和iPhone 5/5S/5C:状态栏高度是20点。
  • iPhone6/6S/7/8(包括Plus版本)和其他早起iPhone模型:状态栏高度也是20点。
  • iPhone X及其之后型号(带有刘海屏):状态栏的高度增加到了44点,以适应刘海屏设计。

我们可以通过UIApplecation.shared.statusBarFrame.height来获取当前状态栏的高度。但是在iOS 13之后推荐使用UIViewController的viewSafeAreaInsets来处理与安全区域相关的布局。后面我们也会介绍到。

/// 状态栏高度
var ZMSTATUSBAR_HEIGHT:CGFloat {
    get {
        if #available(iOS 13.0, *) {
            return UIApplication.shared.windows.first?.windowScene?.statusBarManager?.statusBarFrame.height ?? 0
        } else {
            return UIApplication.shared.statusBarFrame.height
        }
    }
}

顶部安全区域

在iOS开发中,顶部安全区域是指页面中从屏幕底部到底部安全区域之间的空间。自iPhone X引入刘海屏以来,状态栏不再仅仅占据20点的固定高度,而是随着设备的变化可能更高。在刘海屏设备(如iPhoneX、iPhone 11、iPhone 12等,包括iPhone 14以后的灵动岛)上,顶部安全区域的高度增加,用于容纳状态栏和刘海,通常是44点。而在没有刘海的设备上,顶部安全渔区的高度则保持20点。

因此,在适配不同屏幕是,开发这需要通过系统提供的安全区域属性safeAreaInsets来获取这些动态变化的安全区域,以确保UI元素不会被刘海或者状态栏遮挡。

/// 顶部安全区域高度
var ZMTopSafeAreaHeight: CGFloat {
    get {
        return UIApplication.shared.windows.first?.safeAreaInsets.top ?? 0
    }
}

底部安全区域

自从iPhone X开始,底部的物理Home键被移除,取而代之的是一个用于导航手势的区域,为了适配这一变化,底部安全区域(Safe Area Inset)应留出一定的空间,以防止手势操作与应用的UI元素冲突。不同设备的地步安全区域高度可能会有所不同,尤其是在全面屏设备上。

没有物理Home键的设备(如iPhone X及其之后的系列)上,底部安全区域会更大,通常为为34点,而在带有物理Home键的设备商,这个安全区域的高度为0。我们在设计界面时,必须考虑到这个底部安全区域。

可以通过safeAreaInsets.bottom来获取底部安全区域的高度。

/// 底部安全区域高度
var ZMBottomSafeAreaHeight: CGFloat {
    get {
        return UIApplication.shared.windows.first?.safeAreaInsets.bottom ?? 0
    }
}

导航栏高度

导航栏是iOS应用中最常见的UI组件之一,通常用于显示页面的标题、返回按钮等交互元素。在布局时,导航栏的高度是一个极其重要的参数。

在iPhone X出现之前,导航栏的高度是一个固定值,通常是44点,加上状态栏的高度20点。那么我们所看见的总高度就为64点。

iPhone X出现之后,状态栏的高度增肌了,虽然导航栏的核心部分依然保持44点的高度,但我们需要考虑状态栏的高度,其中状态栏约为44点,那么总高度通常为88点。

但是iPhone X之后我们还需要考虑安全区域的高度,因为这些设备的状态栏高度可能会随着安全区域的变化而有所不同,所以我们优先使用安全区域的topInset来确保正确布局。

/// 导航栏高度()
var ZMNAVGATION_HEIGHT: CGFloat {
    guard let window = UIApplication.shared.windows.first else {
        return 0
    }

    let topSafeAreaInset = window.safeAreaInsets.top
    let navigationBarHeight = 44.0

    // 如果设备没有安全区域(即没有刘海),状态栏高度可能是 20
    if topSafeAreaInset > 0 {
        // 对于有刘海的设备,安全区域的 topInset 已经包含状态栏的高度
        return topSafeAreaInset + navigationBarHeight
    } else {
        // 没有刘海的设备,状态栏通常是 20,高度为状态栏 + 导航栏
        let statusBarHeight = UIApplication.shared.statusBarFrame.height
        return statusBarHeight + navigationBarHeight
    }
}

Tab Bar高度

Tab Bar的标准高度为49点,这是iPhone X之前所有设备上的固定高度。然而,自iPhone X引入全面屏设计并取消了物理Home键后,底部安全区域(Safe Area Inset)使得Tab Bar的高度在全面屏设备上有所增加。

因此我们需要通过计算底部底部安全区域的高度来动态设置Tab Bar的高度。

/// tabbar高度
var ZMTABBAR_HEIGHT:CGFloat {
    get {
        return ZMBottomSafeAreaHeight + 49
    }
}

尺寸示例

下面我们就来打印几个机型的屏幕尺寸信息。

        // 屏幕高
        print("屏幕高 = \(ZMSCREEN_HEIGHT)")
        // 屏幕宽
        print("屏幕宽 = \(ZMSCREEN_WIDTH)")
        // 状态栏高度
        print("状态栏高度 = \(ZMSTATUSBAR_HEIGHT)")
        // 顶部安全距离
        print("顶部安全距离 = \(ZMTopSafeAreaHeight)")
        // 底部安全距离
        print("底部安全距离 = \(ZMBottomSafeAreaHeight)")
        // 导航栏+安全距离
        print("导航栏+安全高度= \(ZMNAVGATION_HEIGHT)")
        // tabbar高度
        print("Tab Bar高度 = \(ZMTABBAR_HEIGHT)")

iPhone 15

屏幕高 = 852.0

屏幕宽 = 393.0

状态栏高度 = 54.0

顶部安全距离 = 59.0

底部安全距离 = 34.0

导航栏+安全高度= 103.0

Tab Bar高度 = 83.0

iPhone 11

屏幕高 = 896.0

屏幕宽 = 414.0

状态栏高度 = 48.0

顶部安全距离 = 48.0

底部安全距离 = 34.0

导航栏+安全高度= 92.0

Tab Bar高度 = 83.0

iPhone SE

屏幕高 = 667.0

屏幕宽 = 375.0

状态栏高度 = 20.0

顶部安全距离 = 20.0

底部安全距离 = 0.0

导航栏+安全高度= 64.0

Tab Bar高度 = 49.0

完整代码

/// 屏幕宽度
let ZMSCREEN_WIDTH = UIScreen.main.bounds.width
/// 屏幕高度
let ZMSCREEN_HEIGHT = UIScreen.main.bounds.height
/// 状态栏高度
var ZMSTATUSBAR_HEIGHT:CGFloat {
    get {
        if #available(iOS 13.0, *) {
            return UIApplication.shared.windows.first?.windowScene?.statusBarManager?.statusBarFrame.height ?? 0
        } else {
            return UIApplication.shared.statusBarFrame.height
        }
    }
}

/// 顶部安全距离
var ZMTopSafeAreaHeight:CGFloat {
    get {
        return UIApplication.shared.windows.first?.safeAreaInsets.top ?? 0
    }
}

/// 底部安全距离
var ZMBottomSafeAreaHeight:CGFloat {
    get {
        return UIApplication.shared.windows.first?.safeAreaInsets.bottom ?? 0
    }
}

/// 导航栏高度()
var ZMNAVGATION_HEIGHT: CGFloat {
    guard let window = UIApplication.shared.windows.first else {
        return 0
    }

    let topSafeAreaInset = window.safeAreaInsets.top
    let navigationBarHeight = 44.0

    // 如果设备没有安全区域(即没有刘海),状态栏高度可能是 20
    if topSafeAreaInset > 0 {
        // 对于有刘海的设备,安全区域的 topInset 已经包含状态栏的高度
        return topSafeAreaInset + navigationBarHeight
    } else {
        // 没有刘海的设备,状态栏通常是 20,高度为状态栏 + 导航栏
        let statusBarHeight = UIApplication.shared.statusBarFrame.height
        return statusBarHeight + navigationBarHeight
    }
}

/// tabbar高度
var ZMTABBAR_HEIGHT:CGFloat {
    get {
        return ZMBottomSafeAreaHeight + 49
    }
}

结语

以上就是在iOS项目开发中关于屏幕尺寸以及导航栏高度Tab Bar高度适配的所有内容。通过动态获取屏幕尺寸、导航栏、状态栏和安全区域的高度,我们能够适配不同设备,确保应用在各种屏幕上都有最佳的用户体验。随着iPhone屏幕设计的演进,特别是全面屏和灵动岛的引入,开发者必须灵活运用这额布局方法,以适应不断变化的屏幕形态。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2148367.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

rabbitmq整合skywalking并编写自定义插件增强

rabbitmq整合skywalking 首先先下载准备好skywalking 的服务端和ui控制台,java-agent https://skywalking.apache.org/downloads/ 整合skywalking 我的流程是在生产者和消费者服务中去引入一个mq的sdk,具体SDK的内容可以查看这篇文章 在sdk的pom文件…

合理使用布局

一、ArkUI框架执行流程 在使用ArkUI开发中,我们通过布局组件和基础组件进行界面描述,这些描述会呈现出一个组件树的结构,基础组件在其中为叶子结点,布局组件则是中间节点,可以把这棵树称之为应用组件树。当用户执行交互…

VMware ESXi 8.0U3b macOS Unlocker OEM BIOS 2.7 集成网卡驱动和 NVMe 驱动 (集成驱动版)

VMware ESXi 8.0U3b macOS Unlocker & OEM BIOS 2.7 集成网卡驱动和 NVMe 驱动 (集成驱动版) 发布 ESXi 8.0U3 集成驱动版,在个人电脑上运行企业级工作负载 请访问原文链接:https://sysin.org/blog/vmware-esxi-8-u3-sysin/,查看最新版…

用多种编程语言绘制爱心图案的艺术

目录 一、Python:用 Matplotlib 绘制静态爱心 代码示例: 效果展示: 二、JavaScript:用 HTML5 Canvas 绘制跃动的爱心 代码示例: 效果展示: 三、Java:用 Swing 绘制静态爱心 代码示例&…

【新手上路】衡石分析平台系统管理手册-安全管理

安全策略​ 安全策略是针对系统中所有用户进行的安全控制,只有系统管理员可以设置。 打开设置->安全策略页面。 登录安全策略​ 启用复杂密码​ 启用之后,用户修改密码时,必须输入符合要求的复杂密码。 密码90天失效​ 密码的有效期…

江协科技STM32学习- P14 示例程序(定时器定时中断和定时器外部时钟)

🚀write in front🚀 🔎大家好,我是黄桃罐头,希望你看完之后,能对你有所帮助,不足请指正!共同学习交流 🎁欢迎各位→点赞👍 收藏⭐️ 留言📝​…

【嵌入式】二进制参数配置生效策略引发数据类型校验失败问题

背景 嵌入式产品发布后,可以通过升级二进制小文件进行产品参数配置。因为预留配置问题,当二进制转化为内部结构体架构化数据时,会判断如果值为255则表示无需配置生效。但是因为笔误,代码不严谨,调试的时候发现数值校验…

用代码生成代码之Roslyn-C#代码分析和生成工具

Roslyn 是什么? Roslyn是微软公司开源的.NET编译器,它提供了丰富的代码分析API,并支持C#和Visual Basic代码的编译。 Roslyn 的主要功能 编译器扩展: 使用Roslyn可以创建自定义的编译器扩展,如语法分析器、重构器、…

50页PPT麦肯锡精益运营转型五步法

读者朋友大家好,最近有会员朋友咨询晓雯,需要《 50页PPT麦肯锡精益运营转型五步法》资料,欢迎大家下载学习。 知识星球已上传的资料链接: 企业架构 企业架构 (EA) 设计咨询项目-企业架构治理(EAM)现状诊断 105页PPTHW企业架构设…

JavaSE--集合总览02:单列集合Collection的体系之一:List

Collection体系的特点 分为 list 和set集合,这篇文章主要讲述List,下篇讲述Set。 简单认识单列集合collection集合的特点 : list集合的特点: 有序 可重复 有索引 set集合的特点:无序 不重复 无索引 其中LinkedHashSet有序 TreeS…

基于springboot的智慧社区微信小程序

文未可获取一份本项目的java源码和数据库参考。 本课题研究目标 本文主要对小区生活服务平台的功能和非功能需求进行了分析,系统除了提供物业保修、小区资讯、投诉留言、常用电话等基础功能外,为了满足用户的多样化需求,还提供邻里圈子和有…

仪表放大器AD620

AD623 是一款低功耗、高精度的仪表放大器,而不是轨到轨运算放大器。它的输入电压范围并不覆盖整个电源电压(轨到轨),但在单电源供电下可以处理接近地电位的输入信号。 AD620 和 AD623 都是仪表放大器,但它们在一些关键…

【OceanBase诊断调优】—— 断连接问题根因分析

背景 当前用户请求执行的链路主要如下,请求从客户端发送到ObProxy,ObProxy将请求路由到对应的ObServer节点,ObServer处理请求发送回包给ObProxy,ObProxy回给客户端。目前整条链路上都可能发生断连接的场景,比如请求处…

Java 每日一刊(第12期):面向对象

“任何复杂的程序,都可以通过分解成若干个简单的问题来解决。” 前言 这里是分享 Java 相关内容的专刊,每日一更。 本期将为大家带来以下内容: 类对象类与对象的关系Java 中的三种变量类型OOP 的三大特性 类 类 是对现实世界中某类事物…

修改Docker默认存储路径,解决系统盘占用90%+问题(修改docker root dir)

随着Docker技术的广泛应用,它极大地简化了复杂项目的部署与维护流程,仅凭单一镜像即可轻松运行。然而,随着数据量不断增长,Docker的默认数据存储方式可能逐渐成为挑战,尤其是当默认安装于根目录(“/”&…

计算机的错误计算(九十八)

摘要 探讨 的计算精度问题。 由计算机的错误计算(九十六)知,IEEE 754-2019标准中含有 运算。 另外,似乎没有语言直接编程实现内置了该运算。 例1. 已知 x-0.9999999999321 . 计算 不妨用Java编程计算: import…

微服务注册中⼼1

1. 微服务的注册中⼼ 注册中⼼可以说是微服务架构中的”通讯录“ ,它记录了服务和服务地址的映射关系。在分布式架构中, 服务会注册到这⾥,当服务需要调⽤其它服务时,就这⾥找到服务的地址,进⾏调⽤。 1.1 注册中⼼的…

前端面试题——token安全问题处理与大数据列表展示

1.长时间保存token问题 长时间保存Token涉及多个方面的问题,包括安全性、性能、以及Token的管理策略等。以下是对长时间保存Token问题的详细分析: 一、安全性问题 Token泄露风险: Token是用户身份验证的凭证,如果长时间保存且未…

矿场工程车检测数据集 4900张 工程车 带标注voc yolo

矿场工程车检测数据集 数据集描述 该数据集旨在用于矿场工程车的检测和分类任务,涵盖了多种常见的工程车辆类型。数据集包含了大量的工程车图像及其对应的标注信息,可用于训练计算机视觉模型,以识别和定位矿场中的不同工程车辆。 数据规模 …

【速成Redis】02 Redis 五大基本数据类型常用命令

前言: 上一节课,我们对redis进行了初步了解,和安装好了redis。【速成Redis】01 Redis简介及windows上如何安装redishttps://blog.csdn.net/weixin_71246590/article/details/142319358?spm1001.2014.3001.5501 该篇博客,我们正…