Welcome to the Custom Containers workshop!
In this workshop we are building a color matching game - the game begins with a row of shuffled colors, and your goal is to drag the tiles so that the shades are in order.
In DuoColors (the first version of the game) we order the shades between two colors | In QuadColors (the second version) we order shades mixed from 4 colors! |
---|---|
There are 3 main directories:
REFERENCE/
: If in a few months you want to check how we did something - this is the place to go. For the duration of the workshop though, there's no reason to use this folder! Inside you'll find 2 projects: DuoColors is the result of both task 1 and task 2, and QuadColors is task 3.SOLVED/
: All the tasks - solved. If you are stuck and don't feel like trying to figure it out, you can peek - although you might have more fun if you try to ask folks around instead!TODO/
: This is where the party happens! Each task is an Xcode project so it doesn't matter how you finished the previous task - we're all starting fresh each time.
Everything you need to do in each task is written as placeholders inside the code, and until you clear them up you won't be able to compile, so there's no way you'll miss anything. At a high level though, here's what we're doing in each task:
Building our first custom container! We'll use ForEach(subviews:)`` to iterate the content's children, and also utilize ContainerValues and
@Entry` to pass so data down to the custom container when we iterate each subview.
Adding an "Easy Mode" to our game - this will add some optional views that are then resolved to a view (or nothing) based on the view's state. We'll also use container values again to pass a "hint" when easy mode is active.
This task is using a modified version of the previous project. We are going to have multiple sections, where each section is a row of colors. We will use ForEach(sections:)
to grab the sections, then ForEach(subviews:)
to get each subview from each section.
For simplicity, the code here does not include easy mode or any of the container values, so that we can focus on the sections.
Done early? Want some extra challenge? Here are a few things you can add to the game!
- MeshGradient is another really cool addition to SwiftUI this year! Try to use it to make a fun background for the "You Win" sheet, made from the colors of the game!
- Start from 2 random colors each time: Currently the game always starts from Red and Blue. It could be fun to make it start from 2 random colors instead!
- Start from 2 selected colors: What if the user could pick which 2 colors to start from? Hint - there's an API for that!
- Challenge mode: Easy mode is great, but maybe the game is actually too easy and not too hard? You can add an option to start from 2 colors that are closer to each other (like Orange and Red instead of Blue and Red).
- Timer: Measure the time it takes to solve the game - then you can compete with friends to see who solves it the fastest!
- High score board: if you added a timer - make a board to keep score of previous games!
ForEach(subviews:)
- iterate each subview of given view and return a modified (or other) view.ForEach(sections:)
- iterate each section of given view and return a modifier (or other) view.Group(subviews:)
- access the subviews of given view and do whatever you want with them.Group(sections:)
- access the sections of given view and do whatever you want with them.containerValue(::)
- set a container value to a subview@Entry
- Apple Sample Code: Creating custom container views
- WWDC24 Session 10146: Demystify SwiftUI containers
- VariadicViews Blog Posts: