在SwiftUI中自定义一个百分比Layout

前言

  1. SwiftUI的原生布局中没有提供相对布局的Layout。但是像聊天气泡这类的View基本上不会沾满容器的宽。一般会设置为占比容器宽度的80%
  2. 幸运的是SwiftUI提供了Layout布局协议,让我们可以自定义一种布局

SwiftUI中的布局原理如下图所示

  1. 容器给子视图一个建议的Size
  2. 子视图渲染并返回它的Size
  3. 容器根据对齐规则,将子视图放置到对应位置

构建相对布局

实现布局Layout

通过实现Layout协议,我们构建相对布局。通过提供宽高的最大占比,来调整传递给子视图的建议大小

swift 复制代码
// 自定义相对布局,注意只能包裹1个子视图
// 通过将布局定义为fileprivate并为View添加一个扩展,我们可以保证使用的时候只有一个子视图
fileprivate struct RelativeSizeLayout: Layout {
    // width相对百分比0~1
    var relativeWidth: Double
    // height相对百分比0~1
    var relativeHeight: Double
    
    // 这是布局的第一步和第二步。父容器传递proposal下来,我们作为自视图将Size返回
    func sizeThatFits(
        proposal: ProposedViewSize, 
        subviews: Subviews, 
        cache: inout ()
    ) -> CGSize {
        assert(subviews.count == 1, "expects a single subview")
        let resizedProposal = ProposedViewSize(
            width: proposal.width.map { $0 * relativeWidth },
            height: proposal.height.map { $0 * relativeHeight }
        )
        return subviews[0].sizeThatFits(resizedProposal)
    }

    // 在这个方法里对子视图进行布局
    func placeSubviews(
        in bounds: CGRect, 
        proposal: ProposedViewSize, 
        subviews: Subviews, 
        cache: inout ()
    ) {
        assert(subviews.count == 1, "expects a single subview")
        let resizedProposal = ProposedViewSize(
            width: proposal.width.map { $0 * relativeWidth },
            height: proposal.height.map { $0 * relativeHeight }
        )
        subviews[0].place(
            at: CGPoint(x: bounds.midX, y: bounds.midY), 
            anchor: .center, 
            proposal: resizedProposal
        )
    }
}

给 View 扩展一个便利的使用方法

swift 复制代码
extension View {
    /// Proposes a percentage of its received proposed size to `self`.
    public func relativeProposed(width: Double = 1, 
                                height: Double = 1) -> some View {
        RelativeSizeLayout(relativeWidth: width,
                          relativeHeight: height) {
            // Wrap content view in a container to make sure the layout only
            // receives a single subview. Because views are lists!
            VStack { // alternatively: `_UnaryViewAdaptor(self)`
                self
            }
        }
    }
}

使用示例

swift 复制代码
let alignment: Alignment = message.sender == .me ? .trailing : .leading
chatBubble
    // 这里我们可以设置气泡的最大宽度,然后再应用我们的relativeProposed。这样可以保证气泡先限制在400以内,然后再走我们的自定义Layout
    // .frame(maxWidth: 400)
    .relativeProposed(width: 0.8)
    .frame(maxWidth: .infinity, alignment: alignment)

说明

  1. 我们自定义的Layout只是将proposalSize进行了百分比计算后传递给了子视图,实际上最后布局的时候子视图的大小依然是子视图返回的size。
  2. SwiftUI中每应用一次modifier,你可以认为是在原有视图上加了一层父视图。而整个布局是从最顶层的屏幕大小开始逐层往下建议尺寸。所以在relativeProposed之前就已经限制了大小的modifier会先决定它的size。就和上面.frame(maxWidth: 400) .relativeProposed(width: 0.8)一样,会先限制最大400,然后才是父容器80%

资料

oleb.net/2023/swiftu...

相关推荐
CYpdpjRnUE1 天前
光伏电池PV建模及其基于Boost Buck电路的最大功率追踪MPPT算法研究及仿真效果探究
swiftui
初级代码游戏2 天前
iOS开发 SwiftUI 15:手势 拖动 缩放 旋转
ios·swiftui·swift
zhyongrui4 天前
SnipTrip 菜单 Liquid Glass 实现方案:结构、材质、交互与深浅色策略
ios·性能优化·swiftui·交互·开源软件·材质
zhyongrui4 天前
SnipTrip 不发烫的实现路径:局部刷新 + 合成缓存 + 峰值削减
ios·swiftui
初级代码游戏5 天前
iOS开发 SwiftUI 14:ScrollView 滚动视图
ios·swiftui·swift
初级代码游戏5 天前
iOS开发 SwitftUI 13:提示、弹窗、上下文菜单
ios·swiftui·swift·弹窗·消息框
zhyongrui5 天前
托盘删除手势与引导体验修复:滚动冲突、画布消失动画、气泡边框
ios·性能优化·swiftui·swift
zhyongrui6 天前
SnipTrip 发热优化实战:从 60Hz 到 30Hz 的性能之旅
ios·swiftui·swift
大熊猫侯佩7 天前
赛博深渊(上):用 Apple Foundation Models 提炼“禁忌知识”的求生指南
llm·swiftui·大语言模型·foundationmodel·apple ai·apple 人工智能·summarize
zhyongrui8 天前
SwiftUI 光晕动画性能优化:消除托盘缩放卡顿的实战方案
ios·性能优化·swiftui