GeometryReader in SwiftUI

GeometryReader 可以获取父视图的大小信息,用来适配不同尺寸的容器

再开始今天的主角前,我们先看一个使用普通手段布局的例子。

我们用两个视图来平分整个页面。

scss 复制代码
HStack(spacing: 0) {
                Rectangle()
                    .fill(Color.red)
                    .frame(width: UIScreen.main.bounds.width * 0.8)
                Rectangle()
                    .fill(Color.blue)
            }
            .ignoresSafeArea()

当我们竖屏时,它确实可以正常工作。但是当我们旋转屏幕后就出现了下

的一幕。

可以看到并没有合理的布局我们需要的视图,当我们旋转了屏幕,视图不自动更新布局。那么此时我们就需要使用GeometryReader来解决这个问题了。


利用GeometryReader布局

代码很简单,只需要把布局 视图放在GeometryReader的内部,然后内部视图布局使用GeometryReader返回的结果就可以办到。

scss 复制代码
GeometryReader { geo in
            HStack(spacing: 0) {
                Rectangle()
                    .fill(Color.red)
 .frame(width: geo.size.width * 0.8)
                Rectangle()
                    .fill(Color.blue)
            }
            .ignoresSafeArea()
        }

把设定宽度的代码从UIScreen.bounds.width 改成geo.size.width即可完成布局。

结果就是我们设定一样了。


做一个炫酷效果

我们做一个图片滚动时有一定3D旋转的效果。首先我们先用ScrollerView把图片布局起来。

scss 复制代码
VStack {
            ScrollView(.horizontal,showsIndicators: false) {
                HStack {
                    ForEach(0..<6) { i in
                        Image("(i)")
                            .resizable()
                            .scaledToFill()
                            .frame(width: 300, height: 280)
                            .cornerRadius(30)
                            .padding()
                    }
                }
            }
            Spacer()
        }

我们想在图片滑动时有一个3D效果, 我们需要得到当前图片移动的位置, 当图片在中心点时,图片的旋转角度就是0 ,否则当视图在中心点的两侧 ,就按照当前位置除以屏幕宽度的一半,来计算百分比

less 复制代码
VStack {
            ScrollView(.horizontal,showsIndicators: false) {
                HStack {
                    ForEach(0..<6) { i in
                        GeometryReader { geo in
                            Image("(i)")
                                .resizable()
                                .scaledToFill()
                                .frame(width: 300, height: 200)
                                .cornerRadius(20)
                               .rotation3DEffect(
                                    Angle(degrees: getPercentage(geo: geo) * 40),
                                    axis: (x: 0.0, y: 1.0, z: 0.0)) 
                        }
                        .frame(width: 300, height: 200)
                        .padding()
                    }
                }
            }
            
            Spacer()
        }
swift 复制代码
 func getPercentage(geo: GeometryProxy) -> Double {
        let maxDistance = UIScreen.main.bounds.width / 2
        let currentX = geo.frame(in: .global).midX
        return Double(1 - (currentX / maxDistance))
    }

geo.frame(in: .global).midX 的表达式返回了当前视图的全局坐标系中的横向中心位置(X 坐标)。这在计算偏移量、动画效果等方面非常有用。在你提供的函数中,它用于计算滑动偏移量的百分比.

GeometryReader 是一个非常消耗内存的View,它会实时计算位置,会对app性能有影响

大家有什么看法呢?欢迎留言讨论。

公众号:RobotPBQ

相关推荐
张飞签名上架1 小时前
深耕全球市场:App上架iOS与Google Play全流程指南
macos·ios·cocoa·ios上架·上架·谷歌上架
Digitally2 小时前
iPhone 无法向安卓设备发送图片:轻松解决
android·ios·iphone
阿里云云原生3 小时前
RUM 助力 iOS 应用稳定性:从异常捕获到堆栈还原的全流程分析
人工智能·阿里云·ios·云原生·rum
初级代码游戏12 小时前
iOS只剩美工了吗?时间都被遮盖看不清了
ios·界面设计·美工
2501_9159184120 小时前
iOS 开发中证书创建与管理中的常见问题
android·ios·小程序·https·uni-app·iphone·webview
00后程序员张20 小时前
IOScer 开发环境证书包括哪些,证书、描述文件与 App ID 的协同管理实践
android·ios·小程序·https·uni-app·iphone·webview
江东小bug王20 小时前
深入理解 UITabBarController:代理方法与 ViewController 生命周期的执行顺序(含 UINavigationController 场景)
ios
阿里云云原生21 小时前
RUM 赋能 iOS App 稳定:从异常体系到监控方案的全方位解析!
ios·云原生
Zfox_21 小时前
无缝穿越系统边界:节点小宝4.0如何让我的Mac/iOS像访问本地盘一样操控Windows
windows·macos·ios·节点小宝
Zender Han1 天前
Flutter 图片裁剪插件 image_cropper 最新版介绍与使用教程
android·flutter·ios