A flexible way to handle async errors in SwiftUI
May 4, 2022 ·
In this post, let’s take a look at how to handle async errors in a flexible and scalable way in SwiftUI. We’ll cover both completion block- and async/await-based use cases.
TL;DR
This post describes an observable AlertContext
that can used to present alerts from any place in the app, an ErrorAlertConvertible
protocol that can be implemented by Error
types that can map the error to an Alert
, and an ErrorAlerter
protocol to present errors.
Although the post contains a lot of text, the total amount of code is actually not that much. You can have a look at the already implemented code in SwiftUIKit.
The traditional way to show alerts
Consider the case where we use a standard .alert
modifier to show an alert if something goes wrong. In it’s easiest form, such a setup could look something like this:
struct MyView: View {
@State
private var isAlertPresented = false
enum MyError: Error {
case somethingWentWrong
}
var body: some View {
Button(action: doSomething) {
Text("Do something")
}.alert(isPresented: $isAlertPresented) {
Alert(title: Text("Something went wrong"))
}
}
}
private extension MyView {
func doSomething() {
doSomethingAsync { error in
isAlertPresented = error != nil
}
}
/// Fake an async call to test what happens.
func doSomethingAsync(completion: (Error?) -> Void) {
completion(MyError.somethingWentWrong)
}
}
Here, we fake an async call and sets an isPresented
flag to true
to simulate an error. It’s super-simple and not applicable in real life, but you get the idea.
To create a more flexible and robust setup, let’s come up with a way to have a single alert that can be used within the entire app, then find a way to trigger that alert in a single way.
Setting up the alert
I have alreaty written about an easier way to handle alerts in SwiftUI in this post. You can read it for more information, but let’s quickly just put it to work here as well.
Although we’ll only use alerts in this post, let’s create an observable class that can be used to manage any presentable content, such as alerts, full-screen covers, sheets, etc.:
open class PresentationContext<Content>: ObservableObject {
public init() {}
@Published
public var isActive = false
@Published
public internal(set) var content: (() -> Content)? {
didSet { isActive = content != nil }
}
public var isActiveBinding: Binding<Bool> {
.init(get: { self.isActive },
set: { self.isActive = $0 }
)
}
public func dismiss() {
isActive = false
}
public func presentContent(_ content: @autoclosure @escaping () -> Content) {
self.content = content
}
}
This context holds the active/presented state as well as a content
builder that generates the content view that should be presented.
Calling presentContent(...)
sets the content builder, which in turn sets isActive
to true
, while calling dismiss()
sets isActive
to false
.
The code is not clean. The binding is called isActiveBinding
instead of isActive
and the presentation function presentContent(...)
instead of present(...)
. You’ll soon see why.
We can now subclass PresentationContext
to create a context for handling alerts:
public class AlertContext: PresentationContext<Alert> {
public func present(_ alert: @autoclosure @escaping () -> Alert) {
presentContent(alert())
}
}
This class is super simple. By inheriting PresentationContext
and binding it to Alert
, we get a context that can be used to present alerts and alerts alone.
The class also has a cleaner present(...)
function that calls presentContent(...)
. This is the function that is meant to be used, so depending on how you setup these contexts, you could make presentContent
internal
to avoid exposing it altogether.
We’ll also create a view modifier soon, that will use the isActiveBinding
, which means that we don’t have to expose that either.
Having a non-generic presentContent
function in the base class also lets us avoid generic types by creating generic functions instead, for instance to present any view as a sheet:
public class SheetContext: PresentationContext<AnyView> {
public func present<Sheet: View>(_ sheet: @autoclosure @escaping () -> Sheet) {
presentContent(sheet().any())
}
}
With these things in mind, I’m happy to have a noisy base class, while the subclasses that are meant to be used are cleaner and tighter.
Binding the alert to the view
With AlertContext
in place, we can now create a view modifier to bind a context to a view:
public extension View {
func alert(_ context: AlertContext) -> some View {
alert(
isPresented: context.isActiveBinding,
content: context.content ?? { Alert(title: Text("")) }
)
}
}
This lets us use myView.alert(...)
just like before, but instead of a binding and a view or an item
that was later introduced, we can provide a context to get a more flexible setup.
Presenting an alert
With AlertContext
and the alert(...)
view modifier in place, we can clean up the code:
struct MyView: View {
@StateObject
private var alert = AlertContext()
enum MyError: String, Error {
case somethingWentWrong
}
var body: some View {
Button(action: doSomething) {
Text("Do something")
}.alert(alert)
}
}
private extension MyView {
func doSomething() {
doSomethingAsync { error in
guard let error = error else { return }
let title = Text(error.localizedDescription)
alert.present(Alert(title: title))
}
}
/**
Fake an async call to test what happens.
*/
func doSomethingAsync(completion: (Error?) -> Void) {
completion(MyError.somethingWentWrong)
}
}
Here, we create a StateObject
context and apply it to our button, then call .present(...)
to present an alert when our fake operation fails.
Although this is already more flexible, having this context gives us even more freedom. We could for instance inject a context from the outside, using .environmentObject(...)
:
struct ParentView: View {
@StateObject
private var alert = AlertContext()
var body: some View {
MyView()
.environmentObject(alert)
}
}
which can be accessed by replacing the @StateObject
property with @EnvironmentObject
:
struct MyView: View {
@EnvironmentObject
private var alert: AlertContext
We can also inject it with the MyView
initializer and set it up as an observed object instead:
struct ParentView: View {
@StateObject
private var alert = AlertContext()
var body: some View {
MyView(alert: alert)
}
}
struct MyView: View {
init(alert: AlertContext) {
self._alert = ObservedObject(wrappedValue: alert)
}
@ObservedObject
private var alert: AlertContext
...
}
Having this context gives us a lot more flexibility. In fact, we don’t even have to bind it in MyView
. We could just bind it within ParentView
and just pass it into the view hierarchy:
struct ParentView: View {
@StateObject
private var alert = AlertContext()
var body: some View {
MyView(alert: alert)
.alert(alert)
}
}
struct MyView: View {
init(alert: AlertContext) {
self._alert = ObservedObject(wrappedValue: alert)
}
...
func alert(_ text: String) {
alert.present(
Alert(title: text)
)
}
}
With this rather long detour, and with this setup in place, we can now start looking at some interesting ways to present error alerts when operations fail.
Alerting errors
We can now use AlertContext
to present any error as an alert. Given a general Error
, it could look something like this:
func alert(_ error: Error) {
alert.present(
Alert(title: error.localizedDescription)
)
}
However, we can do better. Let’s use protocols to get a more flexible way to handle alerts.
First, let’s define a protocol that can be implemented by any error that can be presented as an alert within our apps:
protocol ErrorAlertConvertible: Error {
var errorTitle: String { get }
var errorMessage: String { get }
}
With the required properties, the protocol can be extended to create an alert like this:
extension ErrorAlertConvertible {
var errorAlert: Alert {
Alert(
title: Text(errorTitle),
message: Text(errorMessage),
dismissButton: .default(Text("OK")) // Use localized strings though
)
}
}
We can now use this protocol to define app- or domain-specific errors, for instance:
enum MyError: ErrorAlertConvertible {
case general
var errorTitle: String {
switch self {
case .general:
return "Something went wrong"
}
}
var errorMessage: String {
switch self {
case .general:
return "Please try again"
}
}
}
We can also create app-specific enums, where each case can be converted to an Alert
.
Next, let’s define a protocol that can be implemented by types that can present errors:
protocol ErrorAlerter {
var alert: AlertContext { get }
}
With this single requirement, we can extend any type that implements ErrorAlerter
with functions that try to perform async operations and automatically alert any errors that occur.
For instance, adding this function lets an alerter alert any errors:
@MainActor
extension ErrorAlerter {
func alert(error: Error) {
if let error = error as? ErrorAlertConvertible {
return alert.present(error.errorAlert)
}
alert.present(
Alert(
title: Text(error.localizedDescription),
dismissButton: .default(Text("OK"))
)
)
}
}
Notice that since the function will change the context state, we should apply a @MainActor
to the extension. This lets us use it with async/await, since we can just await
the alert.
To support block-based operations, we could add a non-async function version as well:
extension ErrorAlerter {
func alertAsync(error: Error) {
DispatchQueue.main.async {
alert(error: error)
}
}
}
With the ErrorConvertible
and ErrorAlerter
protocols in place, we’re ready to put it all together. Let’s start with a block-based example.
Alert errors from block-based functions
Let’s put ErrorAlerter
to more work by adding an extension that can call any completion block-based operation and automatically alert any errors that occur:
extension ErrorAlerter {
typealias BlockResult<ErrorType: Error> = Result<Void, ErrorType>
typealias BlockCompletion<ErrorType: Error> = (BlockResult<ErrorType>) -> Void
typealias BlockOperation<ErrorType: Error> = (BlockCompletion<ErrorType>) -> Void
func tryWithErrorAlert<ErrorType: Error>(_ operation: @escaping BlockOperation<ErrorType>, completion: @escaping BlockCompletion<ErrorType>) {
operation { result in
switch result {
case .failure(let error): alertAsync(error: error)
case .success: break
}
completion(result)
}
}
}
Here, we define a generic BlockResult
that we use in a generic BlockCompletion
, which we then use in a generic BlockOperation
. This makes the rest of the code more readable.
We then define a tryWithErrorAlert
function that can use any parameterless function. If an error occurs, it calls alertAsync
. If the error implements ErrorAlertConvertible
, it has full control over how it will be presented, otherwise the localized description will be used.
Note that the provided completion
is called as well, to give us a way to handle the result. We can handle the error as well, we just don’t have to care about alerting it.
With this, any view with an AlertContext
can now present any error using the same alert within the entire app, by just implementing the ErrorAlerter
protocol. Pretty neat, right?
Let’s now look at how to achieve the same with Swift concurrency and async/await.
Alert errors from async functions
The code becomes cleaner with async/await, since we don’t need results, completions, etc. An async alternative to the block-based tryWithErrorAlert
function looks like this:
typealias AsyncOperation = () async throws -> Void
func tryWithErrorAlert(_ operation: @escaping AsyncOperation) {
Task {
do {
try await operation()
} catch {
await alert(error: error)
}
}
}
This means that we can perform any parameterless async throwing function and alert any error that is thrown, using the async alert function in the @MainActor
extension.
Using this approach, and with MyError
implementing ErrorAlertConvertible
from before, the initial example could look like this, if it implements ErrorAlerter
:
struct MyView: View, ErrorAlerter {
@StateObject
var alert = AlertContext()
var body: some View {
Button(action: doSomething) {
Text("Do something")
}.alert(alert)
}
}
extension MyView {
func doSomething() {
tryWithErrorAlert(doSomethingAsync)
}
/// Fake an async call to test what happens.
func doSomethingAsync() async throws {
throw(MyError.general)
}
}
We can also inject the alert context from the outside or into the rest of the view hierarchy, and use tryWithErrorAlert
in any view that implements ErrorAlerter
.
This is a lot cleaner than the block-based approach, but if your app targets an iOS version that doesn’t support async/await, you may have to stick with the block-based one for now.
Conclusion
In this article, we created an observable AlertContext
that can be passed around in an app to present alerts from anywhere with a single binding.
We then added an ErrorAlertConvertible
protocol that can be implemented by any Error
that can generate an Alert
, and an ErrorAlerter
protocol that adds extra functionality to anything with an AlertContext
instance.
Although the post contains a lot of text, the total amount of code is actually not that much. You can have a look in SwiftUIKit and use it in your own apps.
I hope you find the approach as usable as I do. I’d love to hear your thoughts on this, so don’t hesitate to comment or reach out with any thoughts you may have.
Discussions & More
If you found this interesting, please share your thoughts on Bluesky, Mastodon, and X. Also make sure to follow to be notified when new content is published.