Scrolling pickers in SwiftUI
July 24, 2024
The idea behind the implementation is pretty straightforward. We need to generate a bunch of vertical segments. Some of them are taller and have a text block, some are just shorter. This is determined by the index of each segment.
First, we define a foundation for the solution.
Thus, we display a dozen vertical segments horizontally. Each vertical segment consists of two rectangles aligned at the top and bottom respectively.
Next, we will improve this solution step-by-step by defining the logic for showing numbers and the behavior for scrolling.
Primaries And Others
Before we continue, let's look at a standard ruler. The longer segments drawn are those whose sequence number is a multiple of the number of steps between the main values.
That is, for example, between numbers 0 and 10 there are 10 steps, the segments with indices 0 and 10 will be long and the rest will be short.
Finishing this part with adding a red line overlay on top of the scroll view so the picker can indicate where the current selection is.
Scrolling The Things
We definitely don't want the default scroll indicator be displayed, so let's hide it:
Next, let's add some content padding to make the beginning of the wheel be aligned with the red line.
This can be done with the .safeAreaPadding
modifier. To calculate a required value for the padding, we use GeometryReader
.
Other option is to pass width as a parameter to view initialiser, but it seems to be not so SwiftUI-ty.
Now let's add some dynamism to the wheel and apply specific effects to segments and numbers.
Segments to the left of the red line will be slightly dimmed, while the segments to the right, on the contrary, will appear completely opaque. We achieve this effect by using the scrollTransition
modifier.
It's second parameter is a closure which provides two values:
- content - instance of
EmptyVisualEffect
type, which by conforming toVisualEffect
protocol provides a plenty of options for applying various visual effects, including scaling, blur, blend mode and others - phase - represents different states of an element placed in scroll view, basically it can tell us whether the element in the center, left or right part of the scroll container
With this knowledge we can implement our opacity effect: for the segments to the left side we will set their opacity to 0.2, for others - to 1.0
Make sure to put this transition modifier before number overlay so that the effect is only applied to segments.
The numbers will be darkened everywhere except in the centre. To do this, we use scrollTransition
again, but apply it directly to the text view to limit the area of influence.
Since we are working in a range of integers, it makes sense to stop scrolling only on the segments themselves, not on the space between them. In other words, make the wheel a bit snappy to the segments.
This is where the ScrollTargetBehavior
protocol and its namesake modifier come into play. By default it comes with two options:
- paging - makes scrolling look like pagination
- viewAligned - with this behavior scroll tries to align the final position with it's child views
ScrollTargetBehavior
works in conjunction with the scrollTargetLayout
modifier. The latter helps to tell SwiftUI which views to consider when calculating the stopping point.
Sadly, both options will not help us achieve the desired behavior. Happily, we can define our own by implementing a custom type conforming the ScrollTargetBehavior
protocol.
Start by defining a new type called SnapScrollTargetBehaviour
.
The updateTarget
method is the only one required by the ScrolLTargetBehavior
protocol. Here, the target
parameter is the means by which we can tell scroll view where it should stop. Initially, it contains some size values that can tell where SwiftUI expects a scroll view to stop. We can use these to customise the behavior.
Math corner.
We divide the entire length of the scroll view into some number of small slices. Each slice represents the distance the red line needs to travel to change the value by one. This distance consists of the width of a single segment and the distance between two separate segments.
We need to find for x1 (the expected stopping point) such that x2 (the desired, snapping stopping point) is a multiple of the slice length and nearest to x1.
Wuka-chika-buka-boom, here's the code.
Now all that remains is to define a convenient method to create the instance.
And use it in the scrollTargetBehavior
modifier. We will consider the step length as the distance between segments and the length of one segment.
Since we call it "WheelPicker", it should pass the value somewhere outside. Let's define a binding for it.
Bind the previously defined property with the scrollPosition
modifier. We simply define the binding in place with a custom setter to filter out optional values.
The scrollPosition
modifier also works closely with scroll targets and uses their identifiers to pass values. Since we are working with integer ranges, we can expect integer to be returned via the respective binding.
In addition to the visual part, we can add a tactile response. Personally, I really like it when applications implement this detail in their UI components.
5th revision of SwiftUI makes it incredibly easy to add haptic feedback. Instead of UIFeedbackGenerator
we can use sensoryFeedback
modifier.
Conclusion
Further it is possible to improve this component, for example, by adding the ability to work not only with integers. Or various customisations of segment size, colours, etc.
In general, it's obvious how much SwiftUI simplifies the handling of fairly non-trivial behaviors by taking away most of the computation.
We'll continue to experiment in future articles.
See you soon 🙌