While developing any kind of SwiftUI application it is often necessary to run some data loading when the view actually appears on the screen. In the past week, I implemented this kind of behaviour into applications at work and also on private projects. In all of these projects I am lucky enough to be able to use async/await
, therefore SwiftUI's .task
(Apple Doc) modifier is the first choice. In this small, more fundamental post I want to highlight how you can use the modifier and what you might be aware of when using it in your own projects.
Basic Usage
The .task
modifier is used to execute an asynchronous task as soon as the view, to which it is attached, appears on the screen.
With this behaviour, the modifier becomes the natural place where you could fetch some data, which needs to be displayed on your view. E.g. in the following, we have a view that displays a list of available ice cream flavours a user can choose from.
Advanced Usage
Besides the basic usage of just attaching the modifier, there are two other parameters you can use to tweak the behaviour. First is that you can define the priority under which the action you supply to the modifier is run on. Per default it will be .userInitiated
, the highest priority. If you need a lower priority you can just pass it as a parameter to the modifier:
Another version of the modifier offers the possibility to supply an id
, task(id:priority:_:)
(Apple Doc)
It behaves exactly like the previous version of the modifier, with the difference that whenever the id
changes, the eventually running task is cancelled and recreated. To detect this change the id
value needs to conform to Equatable
. This version is especially helpful if you are using NavigationSplitView
in your application.
For example, let us have a look at the following view:
In the snippet above, the init(sidebar:detail:)
version of NavigationSplitView
is used to create a 2-column split view. On an iPad the usage of task(id:priority:_:)
is crucial for our detail view now. Since, once the user has selected an ice cream, the detail view will not appear again, meaning the basic version of the task modifier will not trigger the action of the task modifier again. By using the selected ice cream as the trigger for our asynchronous action, this limitation can be mitigated and whenever the user changes their selection the detail data is fetched again. Exactly how it should be.
Comparison vs .onAppear
After having a deeper look at the task
modifier, you might ask yourself what the difference is to the .onAppear
modifier which on the surface offers the same behaviour. Both modifiers can run synchronous methods when the view appears. But .task
has two advantages:
- Native
async/await
support which eliminates the need of wrapping the task intoTask { }
- It offers automatic task cancellation when the view goes out of scope or when used in the
task(id:_:)
version, theid
changes, which makes our lives easier and we do not need to handle the lifetime of the task
Both modifiers have their use cases. It's up to you, to decide which one you need.
Conclusion
In this small post, we had a look at the three different ways of using the .task
modifier to trigger asynchronous actions when a view appears:
task(_:)
in the simplest form without any priority suppliedtask(priority:_:)
with a priority suppliedtask(id:priority:_:)
to automatically cancel and recreate tasks onid
changes
I hope this overview helps you to choose the correct version for your next project. If you find any mistake or would like to reach out to me please do so 🙂
See you next time! 👋