Published on

Layout

Authors

A type that defines the geometry of a collection of views. You traditionally arrange views in your app's user interface using built-in layout containers like HStack and Grid. If you need more complex layout behavior, you can define a custom layout container by creating a type that conforms to the Layout protocol and implementing its required methods:

  • Layout/sizeThatFits(proposal:subviews:cache:) reports the size of the composite layout view.
  • Layout/placeSubviews(in:proposal:subviews:cache:) assigns positions to the container's subviews. You can define a basic layout type with only these two methods:
    struct BasicVStack: Layout {
        func sizeThatFits(
            proposal: ProposedViewSize,
            subviews: Subviews,
            cache: inout ()
        ) -> CGSize {
            // Calculate and return the size of the layout container.
        }
        func placeSubviews(
            in bounds: CGRect,
            proposal: ProposedViewSize,
            subviews: Subviews,
            cache: inout ()
        ) {
            // Tell each subview where to appear.
        }
    }

Use your layout the same way you use a built-in layout container, by providing a ViewBuilder with the list of subviews to arrange:

    BasicVStack {
        Text("A Subview")
        Text("Another Subview")
    }

Support additional behaviors

You can optionally implement other protocol methods and properties to provide more layout container features:

  • Define explicit horizontal and vertical layout guides for the container by implementing explicitAlignment(of:in:proposal:subviews:cache:) for each dimension.
  • Establish the preferred spacing around the container by implementing spacing(subviews:cache:).
  • Indicate the axis of orientation for a container that has characteristics of a stack by implementing the layoutProperties-5rb5b static property.
  • Create and manage a cache to store computed values across different layout protocol calls by implementing makeCache(subviews:). The protocol provides default implementations for these symbols if you don't implement them. See each method or property for details.

Add input parameters

You can define parameters as inputs to the layout, like you might for a View:

    struct BasicVStack: Layout {
        var alignment: HorizontalAlignment
        // ...
    }

Set the parameters at the point where you instantiate the layout:

    BasicVStack(alignment: .leading) {
        // ...
    }

If the layout provides default values for its parameters, you can omit the parameters at the call site, but you might need to keep the parentheses after the name of the layout, depending on how you specify the defaults. For example, suppose you set a default alignment for the basic stack in the parameter declaration:

    struct BasicVStack: Layout {
        var alignment: HorizontalAlignment = .center
        // ...
    }

To instantiate this layout using the default center alignment, you don't have to specify the alignment value, but you do need to add empty parentheses:

    BasicVStack() {
        // ...
    }

The Swift compiler requires the parentheses in this case because of how the layout protocol implements this call site syntax. Specifically, the layout's callAsFunction(_:) method looks for an initializer with exactly zero input arguments when you omit the parentheses from the call site. You can enable the simpler call site for a layout that doesn't have an implicit initializer of this type by explicitly defining one:

    init() {
        self.alignment = .center
    }

For information about Swift initializers, see Initialization in The Swift Programming Language.

Interact with subviews through their proxies

To perform layout, you need information about all of its subviews, which are the views that your container arranges. While your layout can't interact directly with its subviews, it can access a set of subview proxies through the Subviews collection that each protocol method receives as an input parameter. That type is an alias for the LayoutSubviews collection type, which in turn contains LayoutSubview instances that are the subview proxies. You can get information about each subview from its proxy, like its dimensions and spacing preferences. This enables you to measure subviews before you commit to placing them. You also assign a position to each subview by calling its proxy's LayoutSubview/place(at:anchor:proposal:) method. Call the method on each subview from within your implementation of the layout's placeSubviews(in:proposal:subviews:cache:) method.

Access layout values

Views have layout values that you set with view modifiers. Layout containers can choose to condition their behavior accordingly. For example, a built-in HStack allocates space to its subviews based in part on the priorities that you set with the View/layoutPriority(_:) view modifier. Your layout container accesses this value for a subview by reading the proxy's LayoutSubview/priority property. You can also create custom layout values by creating a layout key. Set a value on a view with the View/layoutValue(key:value:) view modifier. Read the corresponding value from the subview's proxy using the key as an index on the subview. For more information about creating, setting, and accessing custom layout values, see LayoutValueKey.